use async_trait::async_trait;
use serde_json::{Value, json};
use crate::skills::{Skill, discover_in_workspace, skills_directories};
use super::spec::{
ApprovalRequirement, ToolCapability, ToolContext, ToolError, ToolResult, ToolSpec,
};
pub struct LoadSkillTool;
#[async_trait]
impl ToolSpec for LoadSkillTool {
fn name(&self) -> &'static str {
"load_skill"
}
fn description(&self) -> &'static str {
"Load a skill (SKILL.md body + companion file list) into the next turn's context. \
Use this when the user names a skill or the task clearly matches a skill listed in the system prompt's `## Skills` section. Faster than read_file + list_dir."
}
fn input_schema(&self) -> Value {
json!({
"type": "object",
"properties": {
"name": {
"type": "string",
"description": "Skill id (the `name` field from the SKILL.md frontmatter, also shown in the `## Skills` listing)."
}
},
"required": ["name"],
"additionalProperties": false
})
}
fn capabilities(&self) -> Vec<ToolCapability> {
vec![ToolCapability::ReadOnly]
}
fn approval_requirement(&self) -> ApprovalRequirement {
ApprovalRequirement::Auto
}
fn supports_parallel(&self) -> bool {
true
}
async fn execute(&self, input: Value, context: &ToolContext) -> Result<ToolResult, ToolError> {
let name = input
.get("name")
.and_then(Value::as_str)
.ok_or_else(|| ToolError::missing_field("name"))?
.trim();
if name.is_empty() {
return Err(ToolError::invalid_input(
"`name` must be a non-empty string",
));
}
let registry = discover_in_workspace(&context.workspace);
let Some(skill) = registry.get(name) else {
let available: Vec<&str> = registry.list().iter().map(|s| s.name.as_str()).collect();
let hint = if available.is_empty() {
let dirs: Vec<String> = skills_directories(&context.workspace)
.iter()
.map(|p| p.display().to_string())
.collect();
if dirs.is_empty() {
"no skills directories found; install skills under `<workspace>/.agents/skills/<name>/SKILL.md`, `~/.agents/skills/<name>/SKILL.md`, or `~/.deepseek/skills/<name>/SKILL.md`"
.to_string()
} else {
format!("no skills installed. Searched: {}", dirs.join(", "))
}
} else {
format!(
"skill `{name}` not found. Available: {}",
available.join(", ")
)
};
return Err(ToolError::execution_failed(hint));
};
let body = format_skill_body(skill);
Ok(ToolResult::success(body).with_metadata(json!({
"skill_name": skill.name,
"skill_path": skill.path.display().to_string(),
"companion_files": collect_companion_files(skill)
.into_iter()
.map(|p| p.display().to_string())
.collect::<Vec<String>>(),
})))
}
}
fn format_skill_body(skill: &Skill) -> String {
let mut out = String::new();
out.push_str(&format!("# Skill: {}\n\n", skill.name));
if !skill.description.trim().is_empty() {
out.push_str(&format!("> {}\n\n", skill.description.trim()));
}
out.push_str(&format!("Source: `{}`\n\n", skill.path.display()));
out.push_str("## SKILL.md\n\n");
out.push_str(skill.body.trim());
out.push('\n');
let companions = collect_companion_files(skill);
if !companions.is_empty() {
out.push_str("\n## Companion files\n\n");
out.push_str(
"Sibling files in the skill directory. Use `read_file` to open them when the task requires.\n\n",
);
for path in &companions {
out.push_str(&format!("- `{}`\n", path.display()));
}
}
out
}
fn collect_companion_files(skill: &Skill) -> Vec<std::path::PathBuf> {
let Some(dir) = skill.path.parent() else {
return Vec::new();
};
let mut entries: Vec<std::path::PathBuf> = match std::fs::read_dir(dir) {
Ok(rd) => rd
.flatten()
.filter_map(|entry| {
let path = entry.path();
let is_file = entry.file_type().is_ok_and(|ft| ft.is_file());
let is_skill_md = path.file_name().and_then(|s| s.to_str()) == Some("SKILL.md");
if is_file && !is_skill_md {
Some(path)
} else {
None
}
})
.collect(),
Err(_) => Vec::new(),
};
entries.sort();
entries
}
#[cfg(test)]
mod tests {
use super::*;
use crate::skills::SkillRegistry;
use std::fs;
use tempfile::tempdir;
fn write_skill(dir: &std::path::Path, name: &str, description: &str, body: &str) {
let skill_dir = dir.join(name);
fs::create_dir_all(&skill_dir).unwrap();
fs::write(
skill_dir.join("SKILL.md"),
format!("---\nname: {name}\ndescription: {description}\n---\n{body}\n"),
)
.unwrap();
}
#[test]
fn load_skill_returns_skill_body_with_description_header() {
let tmp = tempdir().unwrap();
write_skill(
tmp.path(),
"review-pr",
"Run a focused PR review",
"# Steps\n1. Read the diff.\n2. Comment.\n",
);
let skill = SkillRegistry::discover(tmp.path())
.get("review-pr")
.unwrap()
.clone();
let body = format_skill_body(&skill);
assert!(body.contains("# Skill: review-pr"));
assert!(body.contains("Run a focused PR review"));
assert!(body.contains("# Steps"));
assert!(body.contains("Read the diff."));
}
#[test]
fn collect_companion_files_lists_siblings_excluding_skill_md() {
let tmp = tempdir().unwrap();
let skill_dir = tmp.path().join("rich-skill");
fs::create_dir_all(&skill_dir).unwrap();
fs::write(
skill_dir.join("SKILL.md"),
"---\nname: rich-skill\ndescription: x\n---\nbody\n",
)
.unwrap();
fs::write(skill_dir.join("script.py"), "print('hi')").unwrap();
fs::write(skill_dir.join("data.json"), "{}").unwrap();
fs::create_dir_all(skill_dir.join("subdir")).unwrap();
let registry = SkillRegistry::discover(tmp.path());
let skill = registry.get("rich-skill").unwrap();
let files = collect_companion_files(skill);
let names: Vec<String> = files
.iter()
.filter_map(|p| p.file_name().and_then(|s| s.to_str().map(str::to_string)))
.collect();
assert_eq!(
names,
vec!["data.json".to_string(), "script.py".to_string()]
);
}
#[test]
fn collect_companion_files_returns_empty_for_solo_skill() {
let tmp = tempdir().unwrap();
write_skill(tmp.path(), "solo", "Just a skill", "body");
let registry = SkillRegistry::discover(tmp.path());
let skill = registry.get("solo").unwrap();
assert!(collect_companion_files(skill).is_empty());
}
#[test]
fn format_skill_body_emits_companion_files_section_when_present() {
let tmp = tempdir().unwrap();
let skill_dir = tmp.path().join("skill-with-friends");
fs::create_dir_all(&skill_dir).unwrap();
fs::write(
skill_dir.join("SKILL.md"),
"---\nname: skill-with-friends\ndescription: x\n---\nbody\n",
)
.unwrap();
fs::write(skill_dir.join("helper.sh"), "#!/bin/sh\necho hi").unwrap();
let registry = SkillRegistry::discover(tmp.path());
let skill = registry.get("skill-with-friends").unwrap();
let body = format_skill_body(skill);
assert!(body.contains("## Companion files"));
assert!(body.contains("helper.sh"));
}
#[test]
fn format_skill_body_skips_companion_section_when_solo() {
let tmp = tempdir().unwrap();
write_skill(tmp.path(), "solo", "x", "body");
let registry = SkillRegistry::discover(tmp.path());
let skill = registry.get("solo").unwrap();
let body = format_skill_body(skill);
assert!(
!body.contains("## Companion files"),
"solo skills shouldn't emit an empty Companion files section"
);
}
#[tokio::test]
async fn execute_finds_skills_in_opencode_dir_via_workspace_discovery() {
let tmp = tempdir().unwrap();
let workspace = tmp.path().to_path_buf();
let opencode_dir = workspace.join(".opencode").join("skills");
std::fs::create_dir_all(&opencode_dir).unwrap();
write_skill(
&opencode_dir,
"from-opencode",
"Skill installed under .opencode/skills",
"Body content marker.",
);
let mut context = ToolContext::new(workspace);
context.workspace = tmp.path().to_path_buf();
let tool = LoadSkillTool;
let result = tool
.execute(json!({"name": "from-opencode"}), &context)
.await
.expect("load_skill should succeed");
assert!(result.success);
assert!(
result.content.contains("# Skill: from-opencode"),
"body header missing: {}",
&result.content
);
assert!(result.content.contains("Body content marker."));
let metadata = result.metadata.expect("metadata stamped");
assert_eq!(
metadata
.get("skill_name")
.and_then(serde_json::Value::as_str),
Some("from-opencode")
);
let path_str = metadata
.get("skill_path")
.and_then(serde_json::Value::as_str)
.expect("skill_path stamped");
assert!(
path_str.contains(".opencode"),
"skill_path should point at the .opencode dir: {path_str}"
);
}
#[tokio::test]
async fn execute_returns_helpful_error_for_unknown_skill() {
let tmp = tempdir().unwrap();
let workspace = tmp.path().to_path_buf();
write_skill(
&workspace.join(".agents").join("skills"),
"real-one",
"x",
"body",
);
let context = ToolContext::new(workspace);
let tool = LoadSkillTool;
let err = tool
.execute(json!({"name": "imaginary"}), &context)
.await
.expect_err("unknown skill should error");
let msg = err.to_string();
assert!(
msg.contains("imaginary") && msg.contains("real-one"),
"error must name the missing skill and list available ones: {msg}"
);
}
}