175 lines
4.5 KiB
Bash
Executable File
175 lines
4.5 KiB
Bash
Executable File
#!/usr/bin/env bash
|
|
set -eo pipefail
|
|
|
|
# shellcheck disable=SC1090
|
|
source "$LLM_PROMPT_UTILS_FILE"
|
|
source "$LLM_ROOT_DIR/agents/.shared/utils.sh"
|
|
|
|
# @env LLM_OUTPUT=/dev/stdout
|
|
# @env LLM_AGENT_VAR_PROJECT_DIR=.
|
|
# @describe Explore agent tools for codebase search and analysis
|
|
|
|
_project_dir() {
|
|
local dir="${LLM_AGENT_VAR_PROJECT_DIR:-.}"
|
|
(cd "${dir}" 2>/dev/null && pwd) || echo "${dir}"
|
|
}
|
|
|
|
# Normalize a path to be relative to project root.
|
|
# Strips the project_dir prefix if the LLM passes an absolute path.
|
|
_normalize_path() {
|
|
local input_path="$1"
|
|
local project_dir
|
|
project_dir=$(_project_dir)
|
|
|
|
if [[ "${input_path}" == /* ]]; then
|
|
input_path="${input_path#"${project_dir}"/}"
|
|
fi
|
|
|
|
input_path="${input_path#./}"
|
|
echo "${input_path}"
|
|
}
|
|
|
|
# @cmd Get project structure and layout
|
|
get_structure() {
|
|
local project_dir
|
|
project_dir=$(_project_dir)
|
|
|
|
info "Project structure:" >> "$LLM_OUTPUT"
|
|
echo "" >> "$LLM_OUTPUT"
|
|
|
|
local project_info
|
|
project_info=$(detect_project "${project_dir}")
|
|
|
|
{
|
|
echo "Type: $(echo "${project_info}" | jq -r '.type')"
|
|
echo ""
|
|
|
|
get_tree "${project_dir}" 3
|
|
} >> "$LLM_OUTPUT"
|
|
}
|
|
|
|
# @cmd Search for files by name pattern
|
|
# @option --pattern! File name pattern (e.g., "*.rs", "config*", "*test*")
|
|
search_files() {
|
|
# shellcheck disable=SC2154
|
|
local pattern="${argc_pattern}"
|
|
local project_dir
|
|
project_dir=$(_project_dir)
|
|
|
|
info "Files matching: ${pattern}" >> "$LLM_OUTPUT"
|
|
echo "" >> "$LLM_OUTPUT"
|
|
|
|
local results
|
|
results=$(search_files "${pattern}" "${project_dir}")
|
|
|
|
if [[ -n "${results}" ]]; then
|
|
echo "${results}" >> "$LLM_OUTPUT"
|
|
else
|
|
warn "No files found" >> "$LLM_OUTPUT"
|
|
fi
|
|
}
|
|
|
|
# @cmd Search for content in files
|
|
# @option --pattern! Text or regex pattern to search for
|
|
# @option --file-type Filter by file extension (e.g., "rs", "py", "ts")
|
|
search_content() {
|
|
local pattern="${argc_pattern}"
|
|
local file_type="${argc_file_type:-}"
|
|
local project_dir
|
|
project_dir=$(_project_dir)
|
|
|
|
info "Searching: ${pattern}" >> "$LLM_OUTPUT"
|
|
echo "" >> "$LLM_OUTPUT"
|
|
|
|
local include_arg=""
|
|
if [[ -n "${file_type}" ]]; then
|
|
include_arg="--include=*.${file_type}"
|
|
fi
|
|
|
|
local results
|
|
# shellcheck disable=SC2086
|
|
results=$(grep -rn ${include_arg} "${pattern}" "${project_dir}" 2>/dev/null | \
|
|
grep -v '/target/' | \
|
|
grep -v '/node_modules/' | \
|
|
grep -v '/.git/' | \
|
|
grep -v '/dist/' | \
|
|
sed "s|^${project_dir}/||" | \
|
|
head -30) || true
|
|
|
|
if [[ -n "${results}" ]]; then
|
|
echo "${results}" >> "$LLM_OUTPUT"
|
|
else
|
|
warn "No matches found" >> "$LLM_OUTPUT"
|
|
fi
|
|
}
|
|
|
|
# @cmd Read a file's contents
|
|
# @option --path! Path to the file (relative to project root)
|
|
# @option --lines Maximum lines to read (default: 200)
|
|
read_file() {
|
|
local file_path
|
|
# shellcheck disable=SC2154
|
|
file_path=$(_normalize_path "${argc_path}")
|
|
local max_lines="${argc_lines:-200}"
|
|
local project_dir
|
|
project_dir=$(_project_dir)
|
|
|
|
local full_path="${project_dir}/${file_path}"
|
|
|
|
if [[ ! -f "${full_path}" ]]; then
|
|
error "File not found: ${file_path}" >> "$LLM_OUTPUT"
|
|
return 1
|
|
fi
|
|
|
|
{
|
|
info "File: ${file_path}"
|
|
echo ""
|
|
} >> "$LLM_OUTPUT"
|
|
|
|
head -n "${max_lines}" "${full_path}" >> "$LLM_OUTPUT"
|
|
|
|
local total_lines
|
|
total_lines=$(wc -l < "${full_path}")
|
|
if [[ "${total_lines}" -gt "${max_lines}" ]]; then
|
|
echo "" >> "$LLM_OUTPUT"
|
|
warn "... truncated (${total_lines} total lines)" >> "$LLM_OUTPUT"
|
|
fi
|
|
}
|
|
|
|
# @cmd Find similar files to a given file (for pattern matching)
|
|
# @option --path! Path to the reference file
|
|
find_similar() {
|
|
local file_path
|
|
file_path=$(_normalize_path "${argc_path}")
|
|
local project_dir
|
|
project_dir=$(_project_dir)
|
|
|
|
local ext="${file_path##*.}"
|
|
local dir
|
|
dir=$(dirname "${file_path}")
|
|
|
|
info "Files similar to: ${file_path}" >> "$LLM_OUTPUT"
|
|
echo "" >> "$LLM_OUTPUT"
|
|
|
|
local results
|
|
results=$(find "${project_dir}/${dir}" -maxdepth 1 -type f -name "*.${ext}" \
|
|
! -name "$(basename "${file_path}")" \
|
|
! -name "*test*" \
|
|
! -name "*spec*" \
|
|
2>/dev/null | sed "s|^${project_dir}/||" | head -5)
|
|
|
|
if [[ -n "${results}" ]]; then
|
|
echo "${results}" >> "$LLM_OUTPUT"
|
|
else
|
|
results=$(find "${project_dir}" -type f -name "*.${ext}" \
|
|
! -name "$(basename "${file_path}")" \
|
|
! -name "*test*" \
|
|
-not -path '*/target/*' \
|
|
2>/dev/null | sed "s|^${project_dir}/||" | head -5)
|
|
if [[ -n "${results}" ]]; then
|
|
echo "${results}" >> "$LLM_OUTPUT"
|
|
else
|
|
warn "No similar files found" >> "$LLM_OUTPUT"
|
|
fi
|
|
fi
|
|
} |