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 = "Return what's dirty in the working tree: staged adds/modifies/deletes, \
working-tree modifications, and untracked files. `is_clean: true` if all five \
buckets are empty. Requires `basemind serve` to be run inside a git repository."
)]
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 commit comes with \
sha, summary (first line of message), author, unix timestamp, and — when \
`include_files=true` (default) — the per-file change list relative to its first \
parent. `limit` is the page size (default 20, max 100). Pass `cursor` from a \
previous response to fetch the next page; cursors invalidate when HEAD moves \
(caller restarts on `cursor_invalidated`). Cached by HEAD sha."
)]
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. Returns the same per-commit \
shape as `recent_changes` without the per-file list (the path is implicit). \
`limit` is the page size (default 20, max 100). Pass `cursor` from a previous \
response to fetch the next page; cursors invalidate when HEAD moves."
)]
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 another revision \
(`rev`, defaults to HEAD). Returns three lists: `added` (in the current view, \
not at `rev`), `removed` (at `rev`, not in current view), and `common`. Useful \
for 'what symbols did this branch add' style questions without reading source."
)]
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 = "Walk recent commits (default last 200, max 1000) and return those whose \
changed-file list contains any path matching the regex `pattern`. Cheaper \
than `git log -G` because we only match paths, not patch text. `limit` is \
the page size (default 50, max 500). Pass `cursor` from a previous response \
to fetch the next page; cursors invalidate when HEAD moves. Cached via \
the same `commit_files` cache as `recent_changes`."
)]
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 returns the per-kind \
breakdown (added/modified/deleted). Useful for a churn map of where the \
activity is concentrated."
)]
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 reports old/new \
1-based line ranges plus the changed text (lines prefixed with '-'/'+' for \
modifications). When the file is absent on one side, `present_at_old` or \
`present_at_new` indicates so and hunks describe the full add/remove."
)]
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 = "List commits where the named symbol's body bytes changed (or where it was \
added/removed). Combines tree-sitter outline extraction with the commit log: \
a `recent_changes` filtered by symbol identity rather than file identity. \
`limit` is the page size (default 20, max 100). Pass `cursor` from a \
previous response to fetch the next page; cursors invalidate when HEAD moves."
)]
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 the given revision (default HEAD), returning one hunk per \
consecutive run of lines that share a source commit. Optional 1-based \
`line_start`/`line_end` clamp to a range. Each hunk carries commit sha, \
author, unix time, summary, and the renamed source path if applicable. \
Pass `limit` (default unbounded, max 1000) to page through hunks; the \
returned `next_cursor` encodes the last hunk's `start_line`. Cached \
forever by (suspect_sha, path, range)."
)]
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 that belong to 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 of different kinds. Pass `limit` (default unbounded, max 1000) \
to page through hunks; the returned `next_cursor` encodes the last hunk's \
`start_line`."
)]
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
}
}