feat: support acp slash commands (#2775)

This commit is contained in:
yetone
2025-10-16 19:05:57 +08:00
committed by GitHub
parent 216ba4a8b2
commit 487972386e
8 changed files with 107 additions and 33 deletions

View File

@@ -437,6 +437,8 @@ setmetatable(M.toggle, {
__call = function() M.toggle_sidebar() end,
})
M.slash_commands_id = nil
---@param opts? avante.Config
function M.setup(opts)
---PERF: we can still allow running require("avante").setup() multiple times to override config if users wish to
@@ -496,7 +498,7 @@ function M.setup(opts)
local has_cmp, cmp = pcall(require, "cmp")
if has_cmp then
cmp.register_source("avante_commands", require("cmp_avante.commands"):new())
M.slash_commands_id = cmp.register_source("avante_commands", require("cmp_avante.commands"):new())
cmp.register_source("avante_mentions", require("cmp_avante.mentions"):new(Utils.get_chat_mentions))

View File

@@ -124,8 +124,13 @@ local Utils = require("avante.utils")
---@class avante.acp.Plan
---@field entries avante.acp.PlanEntry[]
---@class avante.acp.AvailableCommand
---@field name string
---@field description string
---@field input? table<string, any>
---@class avante.acp.BaseSessionUpdate
---@field sessionUpdate "user_message_chunk" | "agent_message_chunk" | "agent_thought_chunk" | "tool_call" | "tool_call_update" | "plan"
---@field sessionUpdate "user_message_chunk" | "agent_message_chunk" | "agent_thought_chunk" | "tool_call" | "tool_call_update" | "plan" | "available_commands_update"
---@class avante.acp.UserMessageChunk : avante.acp.BaseSessionUpdate
---@field sessionUpdate "user_message_chunk"
@@ -154,6 +159,10 @@ local Utils = require("avante.utils")
---@field sessionUpdate "plan"
---@field entries avante.acp.PlanEntry[]
---@class avante.acp.AvailableCommandsUpdate : avante.acp.BaseSessionUpdate
---@field sessionUpdate "available_commands_update"
---@field availableCommands avante.acp.AvailableCommand[]
---@class avante.acp.PermissionOption
---@field optionId string
---@field name string
@@ -196,7 +205,7 @@ ACPClient.ERROR_CODES = {
}
---@class ACPHandlers
---@field on_session_update? fun(update: avante.acp.UserMessageChunk | avante.acp.AgentMessageChunk | avante.acp.AgentThoughtChunk | avante.acp.ToolCallUpdate | avante.acp.PlanUpdate)
---@field on_session_update? fun(update: avante.acp.UserMessageChunk | avante.acp.AgentMessageChunk | avante.acp.AgentThoughtChunk | avante.acp.ToolCallUpdate | avante.acp.PlanUpdate | avante.acp.AvailableCommandsUpdate)
---@field on_request_permission? fun(tool_call: table, options: table[], callback: fun(option_id: string | nil)): nil
---@field on_read_file? fun(path: string, line: integer | nil, limit: integer | nil, callback: fun(content: string)): nil
---@field on_write_file? fun(path: string, content: string, callback: fun(error: string|nil)): nil

View File

@@ -1054,6 +1054,32 @@ function M._stream_acp(opts)
if tool_result_message then table.insert(messages, tool_result_message) end
on_messages_add(messages)
end
if update.sessionUpdate == "available_commands_update" then
local commands = update.availableCommands
local has_cmp, cmp = pcall(require, "cmp")
if has_cmp then
local slash_commands_id = require("avante").slash_commands_id
if slash_commands_id ~= nil then cmp.unregister_source(slash_commands_id) end
for _, command in ipairs(commands) do
local exists = false
for _, command_ in ipairs(Config.slash_commands) do
if command_.name == command.name then
exists = true
break
end
end
if not exists then
table.insert(Config.slash_commands, {
name = command.name,
description = command.description,
details = command.description,
})
end
end
local avante = require("avante")
avante.slash_commands_id = cmp.register_source("avante_commands", require("cmp_avante.commands"):new())
end
end
end,
on_request_permission = function(tool_call, options, callback)
local sidebar = require("avante").get()
@@ -1171,6 +1197,7 @@ function M._stream_acp(opts)
session_id = session_id_
if opts.on_save_acp_session_id then opts.on_save_acp_session_id(session_id) end
end
if opts.just_connect_acp_client then return end
local prompt = {}
local donot_use_builtin_system_prompt = opts.history_messages ~= nil and #opts.history_messages > 0
if donot_use_builtin_system_prompt then

View File

@@ -195,6 +195,9 @@ function Sidebar:open(opts)
vim.g.avante_login = true
end
local acp_provider = Config.acp_providers[Config.provider]
if acp_provider then self:handle_submit("") end
return self
end
@@ -2626,6 +2629,20 @@ function Sidebar:get_generate_prompts_options(request, cb)
if cb then cb(prompts_opts) end
end
function Sidebar:submit_input()
if not vim.g.avante_login then
Utils.warn("Sending message to fast!, API key is not yet set", { title = "Avante" })
return
end
if not Utils.is_valid_container(self.containers.input) then return end
local lines = api.nvim_buf_get_lines(self.containers.input.bufnr, 0, -1, false)
local request = table.concat(lines, "\n")
if request == "" then return end
api.nvim_buf_set_lines(self.containers.input.bufnr, 0, -1, false, {})
api.nvim_win_set_cursor(self.containers.input.winid, { 1, 0 })
self:handle_submit(request)
end
---@param request string
function Sidebar:handle_submit(request)
if Config.prompt_logger.enabled then PromptLogger.log_prompt(request) end
@@ -2650,16 +2667,18 @@ function Sidebar:handle_submit(request)
---@type AvanteSlashCommand
local cmd = vim.iter(cmds):filter(function(cmd) return cmd.name == command end):totable()[1]
if cmd then
if command == "lines" then
cmd.callback(self, args, function(args_)
local _, _, question = args_:match("(%d+)-(%d+)%s+(.*)")
request = question
end)
elseif command == "commit" then
cmd.callback(self, args, function(question) request = question end)
else
cmd.callback(self, args)
return
if cmd.callback then
if command == "lines" then
cmd.callback(self, args, function(args_)
local _, _, question = args_:match("(%d+)-(%d+)%s+(.*)")
request = question
end)
elseif command == "commit" then
cmd.callback(self, args, function(question) request = question end)
else
cmd.callback(self, args)
return
end
end
else
self:update_content("Unknown command: " .. command, { focus = false, scroll = false })
@@ -2681,10 +2700,12 @@ function Sidebar:handle_submit(request)
content = self.code.selection.content,
}
--- HACK: we need to set focus to true and scroll to false to
--- prevent the cursor from jumping to the bottom of the
--- buffer at the beginning
self:update_content("", { focus = true, scroll = false })
if request ~= "" then
--- HACK: we need to set focus to true and scroll to false to
--- prevent the cursor from jumping to the bottom of the
--- buffer at the beginning
self:update_content("", { focus = true, scroll = false })
end
---stop scroll when user presses j/k keys
local function on_j()
@@ -2809,6 +2830,7 @@ function Sidebar:handle_submit(request)
---@type AvanteLLMStreamOptions
---@diagnostic disable-next-line: assign-type-mismatch
local stream_options = vim.tbl_deep_extend("force", generate_prompts_options, {
just_connect_acp_client = request == "",
on_start = on_start,
on_stop = on_stop,
on_tool_log = on_tool_log,
@@ -2860,7 +2882,7 @@ function Sidebar:handle_submit(request)
stream_options.on_memory_summarize = on_memory_summarize
on_state_change("generating")
if request ~= "" then on_state_change("generating") end
Llm.stream(stream_options)
end)
end
@@ -2909,19 +2931,7 @@ function Sidebar:create_input_container()
size = get_size(),
})
local function on_submit()
if not vim.g.avante_login then
Utils.warn("Sending message to fast!, API key is not yet set", { title = "Avante" })
return
end
if not Utils.is_valid_container(self.containers.input) then return end
local lines = api.nvim_buf_get_lines(self.containers.input.bufnr, 0, -1, false)
local request = table.concat(lines, "\n")
if request == "" then return end
api.nvim_buf_set_lines(self.containers.input.bufnr, 0, -1, false, {})
api.nvim_win_set_cursor(self.containers.input.winid, { 1, 0 })
self:handle_submit(request)
end
local function on_submit() self:submit_input() end
self.containers.input:mount()
PromptLogger.init()

View File

@@ -417,6 +417,7 @@ vim.g.avante_login = vim.g.avante_login
---@class AvanteLLMStreamOptions: AvanteGeneratePromptsOptions
---@field acp_client? avante.acp.ACPClient
---@field on_save_acp_client? fun(client: avante.acp.ACPClient): nil
---@field just_connect_acp_client? boolean
---@field acp_session_id? string
---@field on_save_acp_session_id? fun(session_id: string): nil
---@field on_start AvanteLLMStartCallback

View File

@@ -1626,7 +1626,22 @@ function M.get_commands()
)
:totable()
return vim.list_extend(builtin_commands, Config.slash_commands)
local commands = {}
local seen = {}
for _, command in ipairs(Config.slash_commands) do
if not seen[command.name] then
table.insert(commands, command)
seen[command.name] = true
end
end
for _, command in ipairs(builtin_commands) do
if not seen[command.name] then
table.insert(commands, command)
seen[command.name] = true
end
end
return commands
end
function M.get_timestamp() return tostring(os.date("%Y-%m-%d %H:%M:%S")) end

View File

@@ -39,6 +39,7 @@ function M.init()
end
function M.log_prompt(request)
if request == "" then return end
local log_dir = Config.prompt_logger.log_dir
local log_file = Utils.join_paths(log_dir, "avante_prompts.log")

View File

@@ -56,9 +56,18 @@ function CommandsSource:execute(item, callback)
local commands = Utils.get_commands()
local command = vim.iter(commands):find(function(command) return command.name == item.data.name end)
if not command then return end
if not command then
callback()
return
end
local sidebar = require("avante").get()
if not command.callback then
if sidebar then sidebar:submit_input() end
callback()
return
end
command.callback(sidebar, nil, function()
local bufnr = sidebar.containers.input.bufnr ---@type integer
local content = table.concat(api.nvim_buf_get_lines(bufnr, 0, -1, false), "\n")