use rmcp::ErrorData as McpError;
use rmcp::handler::server::wrapper::Parameters;
use rmcp::model::CallToolResult;
use rmcp::tool;
use serde_json::Value;
use super::BasemindServer;
use super::helpers::*;
use super::types::*;
#[rmcp::tool_router(vis = "pub(super)", router = "tool_router_git")]
impl BasemindServer {
#[tool(
description = "What's dirty in the working tree: staged adds/modifies/deletes, working-tree \
modifications, untracked files. `is_clean: true` if all five buckets are \
empty. Requires a git repo.",
annotations(read_only_hint = true, open_world_hint = false)
)]
pub(crate) async fn working_tree_status(
&self,
Parameters(_): Parameters<WorkingTreeStatusParams>,
) -> Result<CallToolResult, McpError> {
let __started = std::time::Instant::now();
let __params_json = Value::Null;
let __result: Result<CallToolResult, McpError> = async {
let repo = require_git_repo(&self.state)?;
let s = repo
.status_porcelain()
.map_err(|e| McpError::internal_error(format!("git status: {e}"), None))?;
let is_clean = s.staged_added.is_empty()
&& s.staged_modified.is_empty()
&& s.staged_deleted.is_empty()
&& s.modified.is_empty()
&& s.untracked.is_empty();
json_result(&WorkingTreeStatusView {
staged_added: s.staged_added,
staged_modified: s.staged_modified,
staged_deleted: s.staged_deleted,
modified: s.modified,
untracked: s.untracked,
is_clean,
})
}
.await;
record_call(
&self.state,
"working_tree_status",
&__params_json,
__started,
&__result,
);
__result
}
#[tool(
description = "Last N commits on the current branch, newest first. Each: sha, summary \
(first message line), author, unix timestamp, and — when \
`include_files=true` (default) — the per-file change list vs first parent. \
`limit` is page size (default 20, max 100). `cursor` pages results \
(invalidate when HEAD moves, `cursor_invalidated`). Cached by HEAD sha.",
annotations(read_only_hint = true, open_world_hint = false)
)]
pub(crate) async fn recent_changes(
&self,
Parameters(params): Parameters<RecentChangesParams>,
) -> Result<CallToolResult, McpError> {
let __started = std::time::Instant::now();
let __params_json = serde_json::to_value(¶ms).unwrap_or(Value::Null);
let __result: Result<CallToolResult, McpError> = async {
let repo = require_git_repo(&self.state)?;
let limit = params.limit.unwrap_or(LOG_LIMIT_DEFAULT).min(LOG_LIMIT_MAX) as usize;
let head = head_sha(repo)?;
let snapshot = head_snapshot_id(&head);
let skip = match params.cursor.as_ref() {
Some(c) => {
let (offset, snapshot_id) = c.decode_in_memory()?;
if snapshot_id != snapshot {
return json_result(&RecentChangesResponse {
commits: Vec::new(),
truncated: false,
truncated_reason: None,
next_cursor: None,
cursor_invalidated: true,
});
}
offset as usize
}
None => 0,
};
let walk_depth =
(skip.saturating_add(limit).saturating_add(1)).min(LOG_WALK_MAX) as u32;
let commits = self
.state
.git_cache
.log(repo, &head, None, walk_depth, params.include_files)
.map_err(|e| McpError::internal_error(format!("log: {e}"), None))?;
let page: Vec<CommitView> = commits
.iter()
.skip(skip)
.take(limit)
.cloned()
.map(|c| commit_to_view(c, params.include_files))
.collect();
let has_more = commits.len() > skip + page.len();
let next_cursor = has_more.then(|| {
super::cursor::Cursor::encode_in_memory((skip + page.len()) as u64, snapshot)
});
let truncated = repo.is_shallow();
json_result(&RecentChangesResponse {
commits: page,
truncated,
truncated_reason: truncated.then_some("shallow_clone"),
next_cursor,
cursor_invalidated: false,
})
}
.await;
record_call(
&self.state,
"recent_changes",
&__params_json,
__started,
&__result,
);
__result
}
#[tool(
description = "Commits that modified `path`, newest first. Same per-commit shape as \
`recent_changes` minus the per-file list (path is implicit). `limit` is \
page size (default 20, max 100). `cursor` pages results (invalidate when \
HEAD moves).",
annotations(read_only_hint = true, open_world_hint = false)
)]
pub(crate) async fn commits_touching(
&self,
Parameters(params): Parameters<CommitsTouchingParams>,
) -> Result<CallToolResult, McpError> {
let __started = std::time::Instant::now();
let __params_json = serde_json::to_value(¶ms).unwrap_or(Value::Null);
let __result: Result<CallToolResult, McpError> = async {
let repo = require_git_repo(&self.state)?;
let limit = params.limit.unwrap_or(LOG_LIMIT_DEFAULT).min(LOG_LIMIT_MAX) as usize;
let head = head_sha(repo)?;
let snapshot = head_snapshot_id(&head);
let skip = match params.cursor.as_ref() {
Some(c) => {
let (offset, snapshot_id) = c.decode_in_memory()?;
if snapshot_id != snapshot {
return json_result(&CommitsTouchingResponse {
path: params.path,
commits: Vec::new(),
truncated: false,
truncated_reason: None,
next_cursor: None,
cursor_invalidated: true,
});
}
offset as usize
}
None => 0,
};
let walk_depth =
(skip.saturating_add(limit).saturating_add(1)).min(LOG_WALK_MAX) as u32;
let commits = self
.state
.git_cache
.log(repo, &head, Some(¶ms.path), walk_depth, false)
.map_err(|e| McpError::internal_error(format!("log: {e}"), None))?;
let page: Vec<CommitView> = commits
.iter()
.skip(skip)
.take(limit)
.cloned()
.map(|c| commit_to_view(c, false))
.collect();
let has_more = commits.len() > skip + page.len();
let next_cursor = has_more.then(|| {
super::cursor::Cursor::encode_in_memory((skip + page.len()) as u64, snapshot)
});
let truncated = repo.is_shallow();
json_result(&CommitsTouchingResponse {
path: params.path,
commits: page,
truncated,
truncated_reason: truncated.then_some("shallow_clone"),
next_cursor,
cursor_invalidated: false,
})
}
.await;
record_call(
&self.state,
"commits_touching",
&__params_json,
__started,
&__result,
);
__result
}
#[tool(
description = "Diff the symbol set of `path` between the current view and `rev` (default \
HEAD): `added` (in view, not at `rev`), `removed` (at `rev`, not in view), \
`common` — 'what symbols did this branch add' without reading source.",
annotations(read_only_hint = true, open_world_hint = false)
)]
pub(crate) async fn diff_outline(
&self,
Parameters(params): Parameters<DiffOutlineParams>,
) -> Result<CallToolResult, McpError> {
let __started = std::time::Instant::now();
let __params_json = serde_json::to_value(¶ms).unwrap_or(Value::Null);
let __result: Result<CallToolResult, McpError> = async {
let repo = require_git_repo(&self.state)?;
let rev_spec = params.rev.as_deref().unwrap_or("HEAD");
let rev_sha = repo.resolve_rev(rev_spec).map_err(|e| {
McpError::invalid_params(format!("resolve_rev({rev_spec}): {e}"), None)
})?;
let cache = self.state.cache.load_full();
let here = cache.by_path.get(¶ms.path).map(|l1| {
l1.symbols
.iter()
.map(|s| (s.name.clone(), kind_to_str(s.kind)))
.collect::<Vec<(String, &'static str)>>()
});
let rev_blob = repo.read_blob_at_rev(&rev_sha, ¶ms.path).map_err(|e| {
McpError::internal_error(format!("read blob {rev_sha}:{}: {e}", params.path), None)
})?;
let there: Option<Vec<(String, &'static str)>> = match rev_blob {
Some(bytes) => {
let lang = crate::lang::detect(std::path::Path::new(¶ms.path)).ok_or_else(
|| {
McpError::invalid_params(
format!("unsupported language for {}", params.path),
None,
)
},
)?;
let l1 = crate::extract::l1::extract_l1(lang, &bytes).map_err(|e| {
McpError::internal_error(
format!("extract {rev_sha}:{}: {e}", params.path),
None,
)
})?;
Some(
l1.symbols
.into_iter()
.map(|s| (s.name, kind_to_str(s.kind)))
.collect(),
)
}
None => None,
};
let (added, removed, common, note) = match (here, there) {
(Some(h), Some(t)) => {
let hs: ahash::AHashSet<(String, &'static str)> = h.iter().cloned().collect();
let ts: ahash::AHashSet<(String, &'static str)> = t.iter().cloned().collect();
let added = h
.iter()
.filter(|p| !ts.contains(*p))
.cloned()
.map(|(n, k)| DiffSymbolView {
name: n,
kind: k.to_string(),
})
.collect();
let removed = t
.iter()
.filter(|p| !hs.contains(*p))
.cloned()
.map(|(n, k)| DiffSymbolView {
name: n,
kind: k.to_string(),
})
.collect();
let common = h
.iter()
.filter(|p| ts.contains(*p))
.cloned()
.map(|(n, k)| DiffSymbolView {
name: n,
kind: k.to_string(),
})
.collect();
(added, removed, common, None)
}
(Some(h), None) => (
h.into_iter()
.map(|(n, k)| DiffSymbolView {
name: n,
kind: k.to_string(),
})
.collect(),
Vec::new(),
Vec::new(),
Some(format!(
"path absent at {rev_spec}; entire file treated as added"
)),
),
(None, Some(t)) => (
Vec::new(),
t.into_iter()
.map(|(n, k)| DiffSymbolView {
name: n,
kind: k.to_string(),
})
.collect(),
Vec::new(),
Some(
"path not indexed in the current view; entire file treated as removed"
.to_string(),
),
),
(None, None) => {
return Err(McpError::invalid_params(
format!(
"path not present in current view or at {rev_spec}: {}",
params.path
),
None,
));
}
};
json_result(&DiffOutlineResponse {
path: params.path,
rev: rev_sha,
added,
removed,
common,
note,
})
}
.await;
record_call(
&self.state,
"diff_outline",
&__params_json,
__started,
&__result,
);
__result
}
#[tool(
description = "Recent commits (default last 200, max 1000) whose changed-file list has a \
path matching the regex `pattern`. Matches paths only, not patch text \
(cheaper than `git log -G`). `limit` is page size (default 50, max 500). \
`cursor` pages results (invalidate when HEAD moves). Shares the \
`recent_changes` commit-files cache.",
annotations(read_only_hint = true, open_world_hint = false)
)]
pub(crate) async fn find_commits_by_path(
&self,
Parameters(params): Parameters<FindCommitsByPathParams>,
) -> Result<CallToolResult, McpError> {
let __started = std::time::Instant::now();
let __params_json = serde_json::to_value(¶ms).unwrap_or(Value::Null);
let __result: Result<CallToolResult, McpError> = async {
let repo = require_git_repo(&self.state)?;
let re = regex::Regex::new(¶ms.pattern)
.map_err(|e| McpError::invalid_params(format!("invalid regex: {e}"), None))?;
let window = params.window.unwrap_or(200).min(1000);
let limit = params.limit.unwrap_or(50).min(500) as usize;
let head = head_sha(repo)?;
let snapshot = head_snapshot_id(&head);
let skip = match params.cursor.as_ref() {
Some(c) => {
let (offset, snapshot_id) = c.decode_in_memory()?;
if snapshot_id != snapshot {
return json_result(&FindCommitsByPathResponse {
pattern: params.pattern,
window_inspected: window,
commits: Vec::new(),
next_cursor: None,
cursor_invalidated: true,
});
}
offset as usize
}
None => 0,
};
let commits = self
.state
.git_cache
.log(repo, &head, None, window, true)
.map_err(|e| McpError::internal_error(format!("log: {e}"), None))?;
let mut hits: Vec<CommitView> = Vec::new();
let mut seen: usize = 0;
let mut has_more = false;
for c in commits.iter() {
if !c.files.iter().any(|(p, _)| re.is_match(&p.to_str_lossy())) {
continue;
}
if seen < skip {
seen += 1;
continue;
}
if hits.len() >= limit {
has_more = true;
break;
}
seen += 1;
hits.push(commit_to_view(c.clone(), true));
}
let next_cursor = has_more.then(|| {
super::cursor::Cursor::encode_in_memory((skip + hits.len()) as u64, snapshot)
});
json_result(&FindCommitsByPathResponse {
pattern: params.pattern,
window_inspected: window,
commits: hits,
next_cursor,
cursor_invalidated: false,
})
}
.await;
record_call(
&self.state,
"find_commits_by_path",
&__params_json,
__started,
&__result,
);
__result
}
#[tool(
description = "Top-K files most-frequently modified in the last `window` commits on the \
current branch (default 200, max 2000). Each entry: per-kind breakdown \
(added/modified/deleted) — a repo churn map.",
annotations(read_only_hint = true, open_world_hint = false)
)]
pub(crate) async fn hot_files(
&self,
Parameters(params): Parameters<HotFilesParams>,
) -> Result<CallToolResult, McpError> {
let __started = std::time::Instant::now();
let __params_json = serde_json::to_value(¶ms).unwrap_or(Value::Null);
let __result: Result<CallToolResult, McpError> = async {
let repo = require_git_repo(&self.state)?;
let window = params.window.unwrap_or(200).min(2000);
let top_k = params.top_k.unwrap_or(20).min(200) as usize;
let head = head_sha(repo)?;
let commits = self
.state
.git_cache
.log(repo, &head, None, window, true)
.map_err(|e| McpError::internal_error(format!("log: {e}"), None))?;
let mut counts: ahash::AHashMap<crate::path::RelPath, (u32, u32, u32, u32)> =
ahash::AHashMap::new();
for c in commits.iter() {
for (path, kind) in &c.files {
let entry = counts.entry(path.clone()).or_insert((0, 0, 0, 0));
entry.0 += 1;
match kind {
crate::git::ChangeKind::Added => entry.1 += 1,
crate::git::ChangeKind::Modified | crate::git::ChangeKind::Renamed => {
entry.2 += 1
}
crate::git::ChangeKind::Deleted => entry.3 += 1,
}
}
}
let total_files_changed = counts.len() as u32;
let mut ranked: Vec<HotFileEntry> = counts
.into_iter()
.map(|(path, (n, added, modified, deleted))| HotFileEntry {
path,
commits_touching: n,
added,
modified,
deleted,
})
.collect();
ranked.sort_by(|a, b| {
b.commits_touching
.cmp(&a.commits_touching)
.then(a.path.cmp(&b.path))
});
ranked.truncate(top_k);
json_result(&HotFilesResponse {
window_inspected: window,
total_files_changed,
files: ranked,
})
}
.await;
record_call(
&self.state,
"hot_files",
&__params_json,
__started,
&__result,
);
__result
}
#[tool(
description = "Hunks for `path` between `rev_old` and `rev_new`. Each hunk: old/new 1-based \
line ranges plus changed text ('-'/'+' prefixed). If the file is absent on \
one side, `present_at_old` / `present_at_new` flag it and hunks describe \
the full add/remove.",
annotations(read_only_hint = true, open_world_hint = false)
)]
pub(crate) async fn diff_file(
&self,
Parameters(params): Parameters<DiffFileParams>,
) -> Result<CallToolResult, McpError> {
let __started = std::time::Instant::now();
let __params_json = serde_json::to_value(¶ms).unwrap_or(Value::Null);
let __result: Result<CallToolResult, McpError> = async {
let repo = require_git_repo(&self.state)?;
let old_sha = repo.resolve_rev(¶ms.rev_old).map_err(|e| {
McpError::invalid_params(format!("resolve_rev({}): {e}", params.rev_old), None)
})?;
let new_sha = repo.resolve_rev(¶ms.rev_new).map_err(|e| {
McpError::invalid_params(format!("resolve_rev({}): {e}", params.rev_new), None)
})?;
let result = repo
.diff_file(&old_sha, &new_sha, ¶ms.path)
.map_err(|e| McpError::internal_error(format!("diff: {e}"), None))?;
let (hunks, present_old, present_new) = result.unwrap_or((Vec::new(), false, false));
let hunks = hunks
.into_iter()
.map(|h| HunkView {
kind: h.kind.as_str(),
old_line_start: h.old_line_start,
old_line_count: h.old_line_count,
new_line_start: h.new_line_start,
new_line_count: h.new_line_count,
text: h.text,
})
.collect();
json_result(&DiffFileResponse {
path: params.path,
rev_old: old_sha,
rev_new: new_sha,
present_at_old: present_old,
present_at_new: present_new,
hunks,
})
}
.await;
record_call(
&self.state,
"diff_file",
&__params_json,
__started,
&__result,
);
__result
}
#[tool(
description = "Commits where the named symbol's body bytes changed (or it was \
added/removed): `recent_changes` filtered by symbol identity, not file \
identity, via tree-sitter outlines. `limit` is page size (default 20, max \
100). `cursor` pages results (invalidate when HEAD moves).",
annotations(read_only_hint = true, open_world_hint = false)
)]
pub(crate) async fn symbol_history(
&self,
Parameters(params): Parameters<SymbolHistoryParams>,
) -> Result<CallToolResult, McpError> {
let __started = std::time::Instant::now();
let __params_json = serde_json::to_value(¶ms).unwrap_or(Value::Null);
let __result: Result<CallToolResult, McpError> = async {
let repo = require_git_repo(&self.state)?;
let kind = params.kind.as_deref().map(parse_kind).transpose()?;
let limit = params.limit.unwrap_or(20).min(100) as usize;
let lang =
crate::lang::detect(std::path::Path::new(¶ms.path)).ok_or_else(|| {
McpError::invalid_params(format!("unsupported language: {}", params.path), None)
})?;
let hash_mode = match params.hash_mode.as_deref() {
Some(s) => parse_hash_mode(s)?,
None => HashMode::Normalized,
};
let head = head_sha(repo)?;
let snapshot = head_snapshot_id(&head);
let skip = match params.cursor.as_ref() {
Some(c) => {
let (offset, snapshot_id) = c.decode_in_memory()?;
if snapshot_id != snapshot {
return json_result(&SymbolHistoryResponse {
path: params.path,
name: params.name,
kind: kind.map(|k| kind_to_str(k).to_string()),
commits_inspected: 0,
history: Vec::new(),
hash_mode: hash_mode.as_str(),
truncated: false,
truncated_reason: None,
next_cursor: None,
cursor_invalidated: true,
});
}
offset as usize
}
None => 0,
};
let walk_depth = (skip
.saturating_add(limit)
.saturating_add(1)
.saturating_mul(4))
.min(LOG_WALK_MAX) as u32;
let commits = self
.state
.git_cache
.log(repo, &head, Some(¶ms.path), walk_depth, false)
.map_err(|e| McpError::internal_error(format!("log: {e}"), None))?;
let chronological: Vec<crate::git::CommitInfo> =
commits.iter().cloned().rev().collect();
let mut history = Vec::new();
let mut prev_fp: Option<Vec<u8>> = None;
let mut prev_existed = false;
let mut inspected: u32 = 0;
for c in chronological {
inspected += 1;
let blob = repo
.read_blob_at_rev_with_oid(&c.sha, ¶ms.path)
.map_err(|e| McpError::internal_error(format!("blob: {e}"), None))?;
let fingerprint = match blob {
Some((bytes, oid)) => {
outline_entry_for_blob(&self.state.outline_cache, oid, lang, bytes)
.and_then(|entry| {
symbol_fingerprint(&entry, ¶ms.name, kind, lang, hash_mode)
})
}
None => None,
};
let change = match (prev_existed, fingerprint.as_ref()) {
(false, Some(_)) => Some("introduced"),
(true, None) => Some("removed"),
(true, Some(curr)) => {
if prev_fp.as_deref() != Some(curr.as_slice()) {
Some("modified")
} else {
None
}
}
(false, None) => None,
};
if let Some(kind_str) = change {
history.push(SymbolHistoryEntry {
sha: c.sha.clone(),
short_sha: c.short_sha.clone(),
summary: c.summary.clone(),
author: c.author.clone(),
author_time_unix: c.author_time_unix,
change: kind_str,
});
}
prev_existed = fingerprint.is_some();
prev_fp = fingerprint;
}
history.reverse();
let total_history = history.len();
let page: Vec<SymbolHistoryEntry> =
history.into_iter().skip(skip).take(limit).collect();
let has_more = total_history > skip + page.len();
let next_cursor = has_more.then(|| {
super::cursor::Cursor::encode_in_memory((skip + page.len()) as u64, snapshot)
});
let truncated = repo.is_shallow();
json_result(&SymbolHistoryResponse {
path: params.path,
name: params.name,
kind: kind.map(|k| kind_to_str(k).to_string()),
commits_inspected: inspected,
history: page,
hash_mode: hash_mode.as_str(),
truncated,
truncated_reason: truncated.then_some("shallow_clone"),
next_cursor,
cursor_invalidated: false,
})
}
.await;
record_call(
&self.state,
"symbol_history",
&__params_json,
__started,
&__result,
);
__result
}
#[tool(
description = "Blame the file at `rev` (default HEAD): one hunk per consecutive run of \
lines sharing a source commit. Optional 1-based `line_start`/`line_end` \
clamp a range. Each hunk: commit sha, author, unix time, summary, renamed \
source path if any. `limit` (default unbounded, max 1000) pages hunks; \
`next_cursor` encodes the last hunk's `start_line`. Cached by \
(suspect_sha, path, range).",
annotations(read_only_hint = true, open_world_hint = false)
)]
pub(crate) async fn blame_file(
&self,
Parameters(params): Parameters<BlameFileParams>,
) -> Result<CallToolResult, McpError> {
let __started = std::time::Instant::now();
let __params_json = serde_json::to_value(¶ms).unwrap_or(Value::Null);
let __result: Result<CallToolResult, McpError> = async {
let repo = require_git_repo(&self.state)?;
let suspect_sha = match params.rev.as_deref() {
Some(r) => repo.resolve_rev(r).map_err(|e| {
McpError::invalid_params(format!("resolve_rev({r}): {e}"), None)
})?,
None => head_sha(repo)?,
};
let range = match (params.line_start, params.line_end) {
(Some(lo), Some(hi)) => Some((lo, hi)),
(None, None) => None,
_ => {
return Err(McpError::invalid_params(
"line_start and line_end must be provided together",
None,
));
}
};
let resume_after: u32 = match params.cursor.as_ref() {
Some(c) => c.decode_in_memory()?.0.min(u32::MAX as u64) as u32,
None => 0,
};
let result = match self
.state
.git_cache
.blame(repo, &suspect_sha, ¶ms.path, range)
{
Ok(r) => r,
Err(e) => {
if let Some(too_large) =
blame_too_large_response(¶ms.path, &suspect_sha, &e)
{
return json_result(&too_large);
}
return Err(McpError::internal_error(format!("blame: {e}"), None));
}
};
let (hunks, next_cursor) =
paginate_blame_hunks(result.hunks.iter(), resume_after, params.limit);
let truncated_reason: Option<&'static str> = match result.truncated_reason.as_deref() {
Some("shallow_clone") => Some("shallow_clone"),
Some(_) => Some("truncated"),
None if repo.is_shallow() => Some("shallow_clone"),
None => None,
};
json_result(&BlameResponse {
path: result.path.clone(),
suspect_sha: result.suspect_sha.clone(),
hunks,
truncated: truncated_reason.is_some(),
truncated_reason,
next_cursor,
})
}
.await;
record_call(
&self.state,
"blame_file",
&__params_json,
__started,
&__result,
);
__result
}
#[tool(
description = "Blame only the lines of a named symbol in a file. Resolves the symbol via \
the cached L1 outline (must be indexed in the current view) and feeds its \
line range to `blame_file`. `kind` disambiguates same-named symbols. \
`limit` (default unbounded, max 1000) pages hunks; `next_cursor` encodes \
the last hunk's `start_line`.",
annotations(read_only_hint = true, open_world_hint = false)
)]
pub(crate) async fn blame_symbol(
&self,
Parameters(params): Parameters<BlameSymbolParams>,
) -> Result<CallToolResult, McpError> {
let __started = std::time::Instant::now();
let __params_json = serde_json::to_value(¶ms).unwrap_or(Value::Null);
let __result: Result<CallToolResult, McpError> = async {
let repo = require_git_repo(&self.state)?;
let kind = params.kind.as_deref().map(parse_kind).transpose()?;
let cache = self.state.cache.load_full();
let l1 = cache.by_path.get(¶ms.path).ok_or_else(|| {
McpError::invalid_params(
format!("file not indexed in current view: {}", params.path),
None,
)
})?;
let sym = l1
.symbols
.iter()
.find(|s| s.name == params.name && kind.is_none_or(|k| s.kind == k))
.ok_or_else(|| {
McpError::invalid_params(
format!(
"symbol `{}`{} not found in {}",
params.name,
kind.map(|k| format!(" (kind={})", kind_to_str(k)))
.unwrap_or_default(),
params.path
),
None,
)
})?;
let (line_start, line_end) = symbol_line_range(repo, ¶ms.path, sym);
let suspect_sha = match params.rev.as_deref() {
Some(r) => repo.resolve_rev(r).map_err(|e| {
McpError::invalid_params(format!("resolve_rev({r}): {e}"), None)
})?,
None => head_sha(repo)?,
};
let resume_after: u32 = match params.cursor.as_ref() {
Some(c) => c.decode_in_memory()?.0.min(u32::MAX as u64) as u32,
None => 0,
};
let result = match self.state.git_cache.blame(
repo,
&suspect_sha,
¶ms.path,
Some((line_start, line_end)),
) {
Ok(r) => r,
Err(e) => {
if let Some(too_large) = blame_symbol_too_large_response(
¶ms.path,
&suspect_sha,
sym,
line_start,
line_end,
&e,
) {
return json_result(&too_large);
}
return Err(McpError::internal_error(format!("blame: {e}"), None));
}
};
let (hunks, next_cursor) =
paginate_blame_hunks(result.hunks.iter(), resume_after, params.limit);
let truncated_reason: Option<&'static str> = match result.truncated_reason.as_deref() {
Some("shallow_clone") => Some("shallow_clone"),
Some(_) => Some("truncated"),
None if repo.is_shallow() => Some("shallow_clone"),
None => None,
};
json_result(&BlameSymbolResponse {
path: result.path.clone(),
suspect_sha: result.suspect_sha.clone(),
name: sym.name.clone(),
kind: kind_to_str(sym.kind).to_string(),
line_start,
line_end,
hunks,
truncated: truncated_reason.is_some(),
truncated_reason,
next_cursor,
})
}
.await;
record_call(
&self.state,
"blame_symbol",
&__params_json,
__started,
&__result,
);
__result
}
}