Was this page helpful?
Lors du travail sur une tâche, Claude a parfois besoin de vérifier auprès des utilisateurs. Il peut avoir besoin d'une permission avant de supprimer des fichiers, ou avoir besoin de demander quelle base de données utiliser pour un nouveau projet. Votre application doit présenter ces demandes aux utilisateurs afin que Claude puisse continuer avec leurs entrées.
Claude demande une entrée utilisateur dans deux situations : lorsqu'il a besoin d'une permission pour utiliser un outil (comme supprimer des fichiers ou exécuter des commandes), et lorsqu'il a des questions de clarification (via l'outil AskUserQuestion). Les deux déclenchent votre rappel canUseTool, qui met en pause l'exécution jusqu'à ce que vous renvoyiez une réponse. C'est différent des tours de conversation normaux où Claude termine et attend votre prochain message.
Pour les questions de clarification, Claude génère les questions et les options. Votre rôle est de les présenter aux utilisateurs et de retourner leurs sélections. Vous ne pouvez pas ajouter vos propres questions à ce flux ; si vous avez besoin de poser une question aux utilisateurs vous-même, faites-le séparément dans votre logique d'application.
Ce guide vous montre comment détecter chaque type de demande et répondre de manière appropriée.
Passez un rappel canUseTool dans vos options de requête. Le rappel se déclenche chaque fois que Claude a besoin d'une entrée utilisateur, en recevant le nom de l'outil et l'entrée comme arguments :
async def handle_tool_request(tool_name, input_data, context):
# Demander à l'utilisateur et retourner autoriser ou refuser
...
options = ClaudeAgentOptions(can_use_tool=handle_tool_request)Le rappel se déclenche dans deux cas :
tool_name pour l'outil (par exemple, "Bash", "Write").AskUserQuestion. Vérifiez si tool_name == "AskUserQuestion" pour le gérer différemment. Si vous spécifiez un tableau tools, incluez AskUserQuestion pour que cela fonctionne. Voir Gérer les questions de clarification pour plus de détails.Pour autoriser ou refuser automatiquement les outils sans inviter les utilisateurs, utilisez plutôt les hooks. Les hooks s'exécutent avant canUseTool et peuvent autoriser, refuser ou modifier les demandes en fonction de votre propre logique. Vous pouvez également utiliser le hook PermissionRequest pour envoyer des notifications externes (Slack, e-mail, push) lorsque Claude attend une approbation.
Une fois que vous avez passé un rappel canUseTool dans vos options de requête, il se déclenche lorsque Claude veut utiliser un outil qui n'est pas auto-approuvé. Votre rappel reçoit deux arguments :
| Argument | Description |
|---|---|
toolName | Le nom de l'outil que Claude veut utiliser (par exemple, "Bash", "Write", "Edit") |
input | Les paramètres que Claude passe à l'outil. Le contenu varie selon l'outil. |
L'objet input contient des paramètres spécifiques à l'outil. Exemples courants :
| Outil | Champs d'entrée |
|---|---|
Bash | command, description, timeout |
Write | file_path, content |
Edit | file_path, old_string, new_string |
Read | file_path, offset, limit |
Consultez la référence du SDK pour les schémas d'entrée complets : Python | TypeScript.
Vous pouvez afficher ces informations à l'utilisateur afin qu'il puisse décider d'autoriser ou de rejeter l'action, puis retourner la réponse appropriée.
L'exemple suivant demande à Claude de créer et de supprimer un fichier de test. Lorsque Claude tente chaque opération, le rappel imprime la demande d'outil sur le terminal et demande une approbation o/n.
En Python, can_use_tool nécessite le mode streaming et un hook PreToolUse qui retourne {"continue_": True} pour garder le flux ouvert. Sans ce hook, le flux se ferme avant que le rappel de permission puisse être invoqué.
Cet exemple utilise un flux o/n où toute entrée autre que o est traitée comme un refus. En pratique, vous pourriez construire une interface utilisateur plus riche qui permet aux utilisateurs de modifier la demande, de fournir des commentaires ou de rediriger Claude entièrement. Voir Répondre aux demandes d'outil pour tous les moyens de répondre.
Votre rappel retourne l'un de deux types de réponse :
| Réponse | Python | TypeScript |
|---|---|---|
| Autoriser | PermissionResultAllow(updated_input=...) | { behavior: "allow", updatedInput } |
| Refuser | PermissionResultDeny(message=...) | { behavior: "deny", message } |
Lors de l'autorisation, passez l'entrée de l'outil (originale ou modifiée). Lors du refus, fournissez un message expliquant pourquoi. Claude voit ce message et peut ajuster son approche.
from claude_agent_sdk.types import PermissionResultAllow, PermissionResultDeny
# Autoriser l'outil à s'exécuter
return PermissionResultAllow(updated_input=input_data)
# Bloquer l'outil
return PermissionResultDeny(message="User rejected this action")Au-delà d'autoriser ou de refuser, vous pouvez modifier l'entrée de l'outil ou fournir un contexte qui aide Claude à ajuster son approche :
Lorsque Claude a besoin de plus de direction sur une tâche avec plusieurs approches valides, il appelle l'outil AskUserQuestion. Cela déclenche votre rappel canUseTool avec toolName défini sur AskUserQuestion. L'entrée contient les questions de Claude sous forme d'options à choix multiples, que vous affichez à l'utilisateur et retournez ses sélections.
Les questions de clarification sont particulièrement courantes en mode plan, où Claude explore la base de code et pose des questions avant de proposer un plan. Cela rend le mode plan idéal pour les flux de travail interactifs où vous voulez que Claude rassemble les exigences avant de faire des modifications.
Les étapes suivantes montrent comment gérer les questions de clarification :
L'entrée contient les questions générées par Claude dans un tableau questions. Chaque question a ces champs :
| Champ | Description |
|---|---|
question | Le texte complet de la question à afficher |
header | Étiquette courte pour la question (max 12 caractères) |
options | Tableau de 2-4 choix, chacun avec label et description |
multiSelect | Si true, les utilisateurs peuvent sélectionner plusieurs options |
Voici un exemple de la structure que vous recevrez :
{
"questions": [
{
"question": "How should I format the output?",
"header": "Format",
"options": [
{ "label": "Summary", "description": "Brief overview of key points" },
{ "label": "Detailed", "description": "Full explanation with examples" }
],
"multiSelect": false
}
]
}Retournez un objet answers mappant le champ question de chaque question au label de l'option sélectionnée :
| Champ | Description |
|---|---|
questions | Passez le tableau de questions original (requis pour le traitement de l'outil) |
answers | Objet où les clés sont le texte de la question et les valeurs sont les labels sélectionnés |
Pour les questions à sélection multiple, joignez plusieurs labels avec ", ". Pour l'entrée de texte libre, utilisez le texte personnalisé de l'utilisateur directement.
{
"questions": [...],
"answers": {
"How should I format the output?": "Summary",
"Which sections should I include?": "Introduction, Conclusion"
}
}Les options prédéfinies de Claude ne couvriront pas toujours ce que les utilisateurs veulent. Pour permettre aux utilisateurs de taper leur propre réponse :
Voir l'exemple complet ci-dessous pour une implémentation complète.
Claude pose des questions de clarification lorsqu'il a besoin d'une entrée utilisateur pour continuer. Par exemple, lorsqu'on lui demande d'aider à décider d'une pile technologique pour une application mobile, Claude peut poser des questions sur le cross-plateforme par rapport au natif, les préférences de backend ou les plates-formes cibles. Ces questions aident Claude à prendre des décisions qui correspondent aux préférences de l'utilisateur plutôt que de deviner.
Cet exemple gère ces questions dans une application de terminal. Voici ce qui se passe à chaque étape :
canUseTool vérifie si le nom de l'outil est "AskUserQuestion" et route vers un gestionnaire dédiéquestions et imprime chaque question avec des options numérotéesquestions original et le mapping answersAskUserQuestion n'est actuellement pas disponible dans les sous-agents générés via l'outil TaskAskUserQuestion supporte 1-4 questions avec 2-4 options chacuneLe rappel canUseTool et l'outil AskUserQuestion couvrent la plupart des scénarios d'approbation et de clarification, mais le SDK offre d'autres façons d'obtenir une entrée des utilisateurs :
Utilisez l'entrée streaming lorsque vous avez besoin de :
L'entrée streaming est idéale pour les interfaces conversationnelles où les utilisateurs interagissent avec l'agent tout au long de l'exécution, pas seulement aux points d'approbation.
Utilisez les outils personnalisés lorsque vous avez besoin de :
AskUserQuestionLes outils personnalisés vous donnent un contrôle total sur l'interaction, mais nécessitent plus de travail d'implémentation que d'utiliser le rappel canUseTool intégré.
import asyncio
from claude_agent_sdk import ClaudeAgentOptions, query
from claude_agent_sdk.types import (
HookMatcher,
PermissionResultAllow,
PermissionResultDeny,
ToolPermissionContext,
)
async def can_use_tool(
tool_name: str, input_data: dict, context: ToolPermissionContext
) -> PermissionResultAllow | PermissionResultDeny:
# Afficher la demande d'outil
print(f"\nTool: {tool_name}")
if tool_name == "Bash":
print(f"Command: {input_data.get('command')}")
if input_data.get("description"):
print(f"Description: {input_data.get('description')}")
else:
print(f"Input: {input_data}")
# Obtenir l'approbation de l'utilisateur
response = input("Allow this action? (y/n): ")
# Retourner autoriser ou refuser en fonction de la réponse de l'utilisateur
if response.lower() == "y":
# Autoriser : l'outil s'exécute avec l'entrée originale (ou modifiée)
return PermissionResultAllow(updated_input=input_data)
else:
# Refuser : l'outil ne s'exécute pas, Claude voit le message
return PermissionResultDeny(message="User denied this action")
# Contournement requis : un hook factice garde le flux ouvert pour can_use_tool
async def dummy_hook(input_data, tool_use_id, context):
return {"continue_": True}
async def prompt_stream():
yield {
"type": "user",
"message": {
"role": "user",
"content": "Create a test file in /tmp and then delete it",
},
}
async def main():
async for message in query(
prompt=prompt_stream(),
options=ClaudeAgentOptions(
can_use_tool=can_use_tool,
hooks={"PreToolUse": [HookMatcher(matcher=None, hooks=[dummy_hook])]},
),
):
if hasattr(message, "result"):
print(message.result)
asyncio.run(main())Passer un rappel canUseTool
Passez un rappel canUseTool dans vos options de requête. Par défaut, AskUserQuestion est disponible. Si vous spécifiez un tableau tools pour restreindre les capacités de Claude (par exemple, un agent en lecture seule avec seulement Read, Glob et Grep), incluez AskUserQuestion dans ce tableau. Sinon, Claude ne pourra pas poser de questions de clarification :
async for message in query(
prompt="Analyze this codebase",
options=ClaudeAgentOptions(
# Inclure AskUserQuestion dans votre liste d'outils
tools=["Read", "Glob", "Grep", "AskUserQuestion"],
can_use_tool=can_use_tool,
),
):
# ...Détecter AskUserQuestion
Dans votre rappel, vérifiez si toolName est égal à AskUserQuestion pour le gérer différemment des autres outils :
async def can_use_tool(tool_name: str, input_data: dict, context):
if tool_name == "AskUserQuestion":
# Votre implémentation pour collecter les réponses de l'utilisateur
return await handle_clarifying_questions(input_data)
# Gérer les autres outils normalement
return await prompt_for_approval(tool_name, input_data)Analyser l'entrée de la question
L'entrée contient les questions de Claude dans un tableau questions. Chaque question a une question (le texte à afficher), des options (les choix) et multiSelect (si plusieurs sélections sont autorisées) :
{
"questions": [
{
"question": "How should I format the output?",
"header": "Format",
"options": [
{ "label": "Summary", "description": "Brief overview" },
{ "label": "Detailed", "description": "Full explanation" }
],
"multiSelect": false
},
{
"question": "Which sections should I include?",
"header": "Sections",
"options": [
{ "label": "Introduction", "description": "Opening context" },
{ "label": "Conclusion", "description": "Final summary" }
],
"multiSelect": true
}
]
}Voir Format de question pour les descriptions complètes des champs.
Collecter les réponses de l'utilisateur
Présentez les questions à l'utilisateur et collectez ses sélections. La façon dont vous le faites dépend de votre application : une invite de terminal, un formulaire web, une boîte de dialogue mobile, etc.
Retourner les réponses à Claude
Construisez l'objet answers comme un enregistrement où chaque clé est le texte de question et chaque valeur est le label de l'option sélectionnée :
| De l'objet question | Utiliser comme |
|---|---|
Champ question (par exemple, "How should I format the output?") | Clé |
Champ label de l'option sélectionnée (par exemple, "Summary") | Valeur |
Pour les questions à sélection multiple, joignez plusieurs labels avec ", ". Si vous supportez l'entrée de texte libre, utilisez le texte personnalisé de l'utilisateur comme valeur.
return PermissionResultAllow(
updated_input={
"questions": input_data.get("questions", []),
"answers": {
"How should I format the output?": "Summary",
"Which sections should I include?": "Introduction, Conclusion"
}
}
)import asyncio
from claude_agent_sdk import ClaudeAgentOptions, query
from claude_agent_sdk.types import HookMatcher, PermissionResultAllow
def parse_response(response: str, options: list) -> str:
"""Analyser l'entrée de l'utilisateur comme numéro(s) d'option ou texte libre."""
try:
indices = [int(s.strip()) - 1 for s in response.split(",")]
labels = [options[i]["label"] for i in indices if 0 <= i < len(options)]
return ", ".join(labels) if labels else response
except ValueError:
return response
async def handle_ask_user_question(input_data: dict) -> PermissionResultAllow:
"""Afficher les questions de Claude et collecter les réponses de l'utilisateur."""
answers = {}
for q in input_data.get("questions", []):
print(f"\n{q['header']}: {q['question']}")
options = q["options"]
for i, opt in enumerate(options):
print(f" {i + 1}. {opt['label']} - {opt['description']}")
if q.get("multiSelect"):
print(" (Enter numbers separated by commas, or type your own answer)")
else:
print(" (Enter a number, or type your own answer)")
response = input("Your choice: ").strip()
answers[q["question"]] = parse_response(response, options)
return PermissionResultAllow(
updated_input={
"questions": input_data.get("questions", []),
"answers": answers,
}
)
async def can_use_tool(tool_name: str, input_data: dict, context) -> PermissionResultAllow:
# Router AskUserQuestion vers notre gestionnaire de questions
if tool_name == "AskUserQuestion":
return await handle_ask_user_question(input_data)
# Auto-approuver les autres outils pour cet exemple
return PermissionResultAllow(updated_input=input_data)
async def prompt_stream():
yield {
"type": "user",
"message": {"role": "user", "content": "Help me decide on the tech stack for a new mobile app"},
}
# Contournement requis : un hook factice garde le flux ouvert pour can_use_tool
async def dummy_hook(input_data, tool_use_id, context):
return {"continue_": True}
async def main():
async for message in query(
prompt=prompt_stream(),
options=ClaudeAgentOptions(
can_use_tool=can_use_tool,
hooks={"PreToolUse": [HookMatcher(matcher=None, hooks=[dummy_hook])]},
),
):
if hasattr(message, "result"):
print(message.result)
asyncio.run(main())