* Refactor Docker mount to mount only user home Mounting the whole filesystem expose the user to security risks, considering the container is running are root. This mounts only the user home directory in the container, to mitigate the security risks. The user home directory is mounted in read only mode to even reduce the risks of accidental or malicious modifications. Mounting the whole should allow the user to have multiple neovim instances runinng at the same time and sharing the same rag_service. Also the container is started with the --rm flag to remove it after it stops. * RAG mount point is not configurable * Remove useless filter.lua file * Use Path to join paths This should be more safe than just concatenating strings.
552 lines
19 KiB
Lua
552 lines
19 KiB
Lua
---NOTE: user will be merged with defaults and
|
|
---we add a default var_accessor for this table to config values.
|
|
|
|
---@alias WebSearchEngineProviderResponseBodyFormatter fun(body: table): (string, string?)
|
|
|
|
local Utils = require("avante.utils")
|
|
|
|
---@class avante.file_selector.IParams
|
|
---@field public title string
|
|
---@field public filepaths string[]
|
|
---@field public handler fun(filepaths: string[]|nil): nil
|
|
|
|
---@class avante.file_selector.opts.IGetFilepathsParams
|
|
---@field public cwd string
|
|
---@field public selected_filepaths string[]
|
|
|
|
---@class avante.CoreConfig: avante.Config
|
|
local M = {}
|
|
---@class avante.Config
|
|
---@field custom_tools AvanteLLMToolPublic[]
|
|
M._defaults = {
|
|
debug = false,
|
|
---@alias Provider "claude" | "openai" | "azure" | "gemini" | "vertex" | "cohere" | "copilot" | string
|
|
provider = "claude",
|
|
-- WARNING: Since auto-suggestions are a high-frequency operation and therefore expensive,
|
|
-- currently designating it as `copilot` provider is dangerous because: https://github.com/yetone/avante.nvim/issues/1048
|
|
-- Of course, you can reduce the request frequency by increasing `suggestion.debounce`.
|
|
auto_suggestions_provider = "claude",
|
|
cursor_applying_provider = nil,
|
|
---@alias Tokenizer "tiktoken" | "hf"
|
|
-- Used for counting tokens and encoding text.
|
|
-- By default, we will use tiktoken.
|
|
-- For most providers that we support we will determine this automatically.
|
|
-- If you wish to use a given implementation, then you can override it here.
|
|
tokenizer = "tiktoken",
|
|
rag_service = {
|
|
enabled = false, -- Enables the rag service, requires OPENAI_API_KEY to be set
|
|
host_mount = os.getenv("HOME"), -- Host mount path for the rag service (docker will mount this path)
|
|
runner = "docker", -- The runner for the rag service, (can use docker, or nix)
|
|
provider = "openai", -- The provider to use for RAG service. eg: openai or ollama
|
|
llm_model = "", -- The LLM model to use for RAG service
|
|
embed_model = "", -- The embedding model to use for RAG service
|
|
endpoint = "https://api.openai.com/v1", -- The API endpoint for RAG service
|
|
},
|
|
web_search_engine = {
|
|
provider = "tavily",
|
|
providers = {
|
|
tavily = {
|
|
api_key_name = "TAVILY_API_KEY",
|
|
extra_request_body = {
|
|
include_answer = "basic",
|
|
},
|
|
---@type WebSearchEngineProviderResponseBodyFormatter
|
|
format_response_body = function(body) return body.answer, nil end,
|
|
},
|
|
serpapi = {
|
|
api_key_name = "SERPAPI_API_KEY",
|
|
extra_request_body = {
|
|
engine = "google",
|
|
google_domain = "google.com",
|
|
},
|
|
---@type WebSearchEngineProviderResponseBodyFormatter
|
|
format_response_body = function(body)
|
|
if body.answer_box ~= nil then return body.answer_box.result, nil end
|
|
if body.organic_results ~= nil then
|
|
local jsn = vim
|
|
.iter(body.organic_results)
|
|
:map(
|
|
function(result)
|
|
return {
|
|
title = result.title,
|
|
link = result.link,
|
|
snippet = result.snippet,
|
|
date = result.date,
|
|
}
|
|
end
|
|
)
|
|
:take(10)
|
|
:totable()
|
|
return vim.json.encode(jsn), nil
|
|
end
|
|
return "", nil
|
|
end,
|
|
},
|
|
searchapi = {
|
|
api_key_name = "SEARCHAPI_API_KEY",
|
|
extra_request_body = {
|
|
engine = "google",
|
|
},
|
|
---@type WebSearchEngineProviderResponseBodyFormatter
|
|
format_response_body = function(body)
|
|
if body.answer_box ~= nil then return body.answer_box.result, nil end
|
|
if body.organic_results ~= nil then
|
|
local jsn = vim
|
|
.iter(body.organic_results)
|
|
:map(
|
|
function(result)
|
|
return {
|
|
title = result.title,
|
|
link = result.link,
|
|
snippet = result.snippet,
|
|
date = result.date,
|
|
}
|
|
end
|
|
)
|
|
:take(10)
|
|
:totable()
|
|
return vim.json.encode(jsn), nil
|
|
end
|
|
return "", nil
|
|
end,
|
|
},
|
|
google = {
|
|
api_key_name = "GOOGLE_SEARCH_API_KEY",
|
|
engine_id_name = "GOOGLE_SEARCH_ENGINE_ID",
|
|
extra_request_body = {},
|
|
---@type WebSearchEngineProviderResponseBodyFormatter
|
|
format_response_body = function(body)
|
|
if body.items ~= nil then
|
|
local jsn = vim
|
|
.iter(body.items)
|
|
:map(
|
|
function(result)
|
|
return {
|
|
title = result.title,
|
|
link = result.link,
|
|
snippet = result.snippet,
|
|
}
|
|
end
|
|
)
|
|
:take(10)
|
|
:totable()
|
|
return vim.json.encode(jsn), nil
|
|
end
|
|
return "", nil
|
|
end,
|
|
},
|
|
kagi = {
|
|
api_key_name = "KAGI_API_KEY",
|
|
extra_request_body = {
|
|
limit = "10",
|
|
},
|
|
---@type WebSearchEngineProviderResponseBodyFormatter
|
|
format_response_body = function(body)
|
|
if body.data ~= nil then
|
|
local jsn = vim
|
|
.iter(body.data)
|
|
-- search results only
|
|
:filter(function(result) return result.t == 0 end)
|
|
:map(
|
|
function(result)
|
|
return {
|
|
title = result.title,
|
|
url = result.url,
|
|
snippet = result.snippet,
|
|
}
|
|
end
|
|
)
|
|
:take(10)
|
|
:totable()
|
|
return vim.json.encode(jsn), nil
|
|
end
|
|
return "", nil
|
|
end,
|
|
},
|
|
brave = {
|
|
api_key_name = "BRAVE_API_KEY",
|
|
extra_request_body = {
|
|
count = "10",
|
|
result_filter = "web",
|
|
},
|
|
format_response_body = function(body)
|
|
if body.web == nil then return "", nil end
|
|
|
|
local jsn = vim.iter(body.web.results):map(
|
|
function(result)
|
|
return {
|
|
title = result.title,
|
|
url = result.url,
|
|
snippet = result.description,
|
|
}
|
|
end
|
|
)
|
|
|
|
return vim.json.encode(jsn), nil
|
|
end,
|
|
},
|
|
},
|
|
},
|
|
---@type AvanteSupportedProvider
|
|
openai = {
|
|
endpoint = "https://api.openai.com/v1",
|
|
model = "gpt-4o",
|
|
timeout = 30000, -- Timeout in milliseconds
|
|
temperature = 0,
|
|
max_tokens = 4096,
|
|
},
|
|
---@type AvanteSupportedProvider
|
|
copilot = {
|
|
endpoint = "https://api.githubcopilot.com",
|
|
model = "gpt-4o-2024-08-06",
|
|
proxy = nil, -- [protocol://]host[:port] Use this proxy
|
|
allow_insecure = false, -- Allow insecure server connections
|
|
timeout = 30000, -- Timeout in milliseconds
|
|
temperature = 0,
|
|
max_tokens = 4096,
|
|
},
|
|
---@type AvanteAzureProvider
|
|
azure = {
|
|
endpoint = "", -- example: "https://<your-resource-name>.openai.azure.com"
|
|
deployment = "", -- Azure deployment name (e.g., "gpt-4o", "my-gpt-4o-deployment")
|
|
api_version = "2024-06-01",
|
|
timeout = 30000, -- Timeout in milliseconds
|
|
temperature = 0,
|
|
max_tokens = 4096,
|
|
},
|
|
---@type AvanteSupportedProvider
|
|
claude = {
|
|
endpoint = "https://api.anthropic.com",
|
|
model = "claude-3-7-sonnet-20250219",
|
|
timeout = 30000, -- Timeout in milliseconds
|
|
temperature = 0,
|
|
max_tokens = 8000,
|
|
},
|
|
---@type AvanteSupportedProvider
|
|
bedrock = {
|
|
model = "anthropic.claude-3-5-sonnet-20240620-v1:0",
|
|
timeout = 30000, -- Timeout in milliseconds
|
|
temperature = 0,
|
|
max_tokens = 8000,
|
|
},
|
|
---@type AvanteSupportedProvider
|
|
gemini = {
|
|
endpoint = "https://generativelanguage.googleapis.com/v1beta/models",
|
|
model = "gemini-1.5-flash-latest",
|
|
timeout = 30000, -- Timeout in milliseconds
|
|
temperature = 0,
|
|
max_tokens = 4096,
|
|
},
|
|
---@type AvanteSupportedProvider
|
|
vertex = {
|
|
endpoint = "https://LOCATION-aiplatform.googleapis.com/v1/projects/PROJECT_ID/locations/LOCATION/publishers/google/models",
|
|
model = "gemini-1.5-flash-002",
|
|
timeout = 30000, -- Timeout in milliseconds
|
|
temperature = 0,
|
|
max_tokens = 4096,
|
|
},
|
|
---@type AvanteSupportedProvider
|
|
cohere = {
|
|
endpoint = "https://api.cohere.com/v2",
|
|
model = "command-r-plus-08-2024",
|
|
timeout = 30000, -- Timeout in milliseconds
|
|
temperature = 0,
|
|
max_tokens = 4096,
|
|
},
|
|
---To add support for custom provider, follow the format below
|
|
---See https://github.com/yetone/avante.nvim/wiki#custom-providers for more details
|
|
---@type {[string]: AvanteProvider}
|
|
vendors = {
|
|
---@type AvanteSupportedProvider
|
|
["claude-haiku"] = {
|
|
__inherited_from = "claude",
|
|
model = "claude-3-5-haiku-20241022",
|
|
timeout = 30000, -- Timeout in milliseconds
|
|
temperature = 0,
|
|
max_tokens = 8000,
|
|
},
|
|
---@type AvanteSupportedProvider
|
|
["claude-opus"] = {
|
|
__inherited_from = "claude",
|
|
model = "claude-3-opus-20240229",
|
|
timeout = 30000, -- Timeout in milliseconds
|
|
temperature = 0,
|
|
max_tokens = 8000,
|
|
},
|
|
},
|
|
---Specify the special dual_boost mode
|
|
---1. enabled: Whether to enable dual_boost mode. Default to false.
|
|
---2. first_provider: The first provider to generate response. Default to "openai".
|
|
---3. second_provider: The second provider to generate response. Default to "claude".
|
|
---4. prompt: The prompt to generate response based on the two reference outputs.
|
|
---5. timeout: Timeout in milliseconds. Default to 60000.
|
|
---How it works:
|
|
--- When dual_boost is enabled, avante will generate two responses from the first_provider and second_provider respectively. Then use the response from the first_provider as provider1_output and the response from the second_provider as provider2_output. Finally, avante will generate a response based on the prompt and the two reference outputs, with the default Provider as normal.
|
|
---Note: This is an experimental feature and may not work as expected.
|
|
dual_boost = {
|
|
enabled = false,
|
|
first_provider = "openai",
|
|
second_provider = "claude",
|
|
prompt = "Based on the two reference outputs below, generate a response that incorporates elements from both but reflects your own judgment and unique perspective. Do not provide any explanation, just give the response directly. Reference Output 1: [{{provider1_output}}], Reference Output 2: [{{provider2_output}}]",
|
|
timeout = 60000, -- Timeout in milliseconds
|
|
},
|
|
---Specify the behaviour of avante.nvim
|
|
---1. auto_focus_sidebar : Whether to automatically focus the sidebar when opening avante.nvim. Default to true.
|
|
---2. auto_suggestions = false, -- Whether to enable auto suggestions. Default to false.
|
|
---3. auto_apply_diff_after_generation: Whether to automatically apply diff after LLM response.
|
|
--- This would simulate similar behaviour to cursor. Default to false.
|
|
---4. auto_set_keymaps : Whether to automatically set the keymap for the current line. Default to true.
|
|
--- Note that avante will safely set these keymap. See https://github.com/yetone/avante.nvim/wiki#keymaps-and-api-i-guess for more details.
|
|
---5. auto_set_highlight_group : Whether to automatically set the highlight group for the current line. Default to true.
|
|
---6. jump_result_buffer_on_finish = false, -- Whether to automatically jump to the result buffer after generation
|
|
---7. support_paste_from_clipboard : Whether to support pasting image from clipboard. This will be determined automatically based whether img-clip is available or not.
|
|
---8. minimize_diff : Whether to remove unchanged lines when applying a code block
|
|
---9. enable_token_counting : Whether to enable token counting. Default to true.
|
|
behaviour = {
|
|
auto_focus_sidebar = true,
|
|
auto_suggestions = false, -- Experimental stage
|
|
auto_suggestions_respect_ignore = false,
|
|
auto_set_highlight_group = true,
|
|
auto_set_keymaps = true,
|
|
auto_apply_diff_after_generation = false,
|
|
jump_result_buffer_on_finish = false,
|
|
support_paste_from_clipboard = false,
|
|
minimize_diff = true,
|
|
enable_token_counting = true,
|
|
enable_cursor_planning_mode = false,
|
|
},
|
|
history = {
|
|
max_tokens = 4096,
|
|
storage_path = vim.fn.stdpath("state") .. "/avante",
|
|
paste = {
|
|
extension = "png",
|
|
filename = "pasted-%Y-%m-%d-%H-%M-%S",
|
|
},
|
|
},
|
|
highlights = {
|
|
diff = {
|
|
current = nil,
|
|
incoming = nil,
|
|
},
|
|
},
|
|
mappings = {
|
|
---@class AvanteConflictMappings
|
|
diff = {
|
|
ours = "co",
|
|
theirs = "ct",
|
|
all_theirs = "ca",
|
|
both = "cb",
|
|
cursor = "cc",
|
|
next = "]x",
|
|
prev = "[x",
|
|
},
|
|
suggestion = {
|
|
accept = "<M-l>",
|
|
next = "<M-]>",
|
|
prev = "<M-[>",
|
|
dismiss = "<C-]>",
|
|
},
|
|
jump = {
|
|
next = "]]",
|
|
prev = "[[",
|
|
},
|
|
submit = {
|
|
normal = "<CR>",
|
|
insert = "<C-s>",
|
|
},
|
|
-- NOTE: The following will be safely set by avante.nvim
|
|
ask = "<leader>aa",
|
|
edit = "<leader>ae",
|
|
refresh = "<leader>ar",
|
|
focus = "<leader>af",
|
|
toggle = {
|
|
default = "<leader>at",
|
|
debug = "<leader>ad",
|
|
hint = "<leader>ah",
|
|
suggestion = "<leader>as",
|
|
repomap = "<leader>aR",
|
|
},
|
|
sidebar = {
|
|
apply_all = "A",
|
|
apply_cursor = "a",
|
|
retry_user_request = "r",
|
|
edit_user_request = "e",
|
|
switch_windows = "<Tab>",
|
|
reverse_switch_windows = "<S-Tab>",
|
|
remove_file = "d",
|
|
add_file = "@",
|
|
close = { "<Esc>", "q" },
|
|
},
|
|
files = {
|
|
add_current = "<leader>ac", -- Add current buffer to selected files
|
|
},
|
|
select_model = "<leader>a?", -- Select model command
|
|
},
|
|
windows = {
|
|
---@alias AvantePosition "right" | "left" | "top" | "bottom" | "smart"
|
|
position = "right",
|
|
wrap = true, -- similar to vim.o.wrap
|
|
width = 30, -- default % based on available width in vertical layout
|
|
height = 30, -- default % based on available height in horizontal layout
|
|
sidebar_header = {
|
|
enabled = true, -- true, false to enable/disable the header
|
|
align = "center", -- left, center, right for title
|
|
rounded = true,
|
|
},
|
|
input = {
|
|
prefix = "> ",
|
|
height = 8, -- Height of the input window in vertical layout
|
|
},
|
|
edit = {
|
|
border = "rounded",
|
|
start_insert = true, -- Start insert mode when opening the edit window
|
|
},
|
|
ask = {
|
|
floating = false, -- Open the 'AvanteAsk' prompt in a floating window
|
|
border = "rounded",
|
|
start_insert = true, -- Start insert mode when opening the ask window
|
|
---@alias AvanteInitialDiff "ours" | "theirs"
|
|
focus_on_apply = "ours", -- which diff to focus after applying
|
|
},
|
|
},
|
|
--- @class AvanteConflictConfig
|
|
diff = {
|
|
autojump = true,
|
|
--- Override the 'timeoutlen' setting while hovering over a diff (see :help timeoutlen).
|
|
--- Helps to avoid entering operator-pending mode with diff mappings starting with `c`.
|
|
--- Disable by setting to -1.
|
|
override_timeoutlen = 500,
|
|
},
|
|
--- @class AvanteHintsConfig
|
|
hints = {
|
|
enabled = true,
|
|
},
|
|
--- @class AvanteRepoMapConfig
|
|
repo_map = {
|
|
ignore_patterns = { "%.git", "%.worktree", "__pycache__", "node_modules" }, -- ignore files matching these
|
|
negate_patterns = {}, -- negate ignore files matching these.
|
|
},
|
|
--- @class AvanteFileSelectorConfig
|
|
file_selector = {
|
|
--- @alias FileSelectorProvider "native" | "fzf" | "mini.pick" | "snacks" | "telescope" | string | fun(params: avante.file_selector.IParams|nil): nil
|
|
provider = "native",
|
|
-- Options override for custom providers
|
|
provider_opts = {},
|
|
},
|
|
suggestion = {
|
|
debounce = 600,
|
|
throttle = 600,
|
|
},
|
|
disabled_tools = {}, ---@type string[]
|
|
---@type AvanteLLMToolPublic[]
|
|
custom_tools = {},
|
|
}
|
|
|
|
---@type avante.Config
|
|
---@diagnostic disable-next-line: missing-fields
|
|
M._options = {}
|
|
|
|
---@type Provider[]
|
|
M.providers = {}
|
|
|
|
---@param opts? avante.Config
|
|
function M.setup(opts)
|
|
vim.validate({ opts = { opts, "table", true } })
|
|
|
|
local merged = vim.tbl_deep_extend(
|
|
"force",
|
|
M._defaults,
|
|
opts or {},
|
|
---@type avante.Config
|
|
{
|
|
behaviour = {
|
|
support_paste_from_clipboard = M.support_paste_image(),
|
|
},
|
|
}
|
|
)
|
|
|
|
M._options = merged
|
|
M.providers = vim
|
|
.iter(M._defaults)
|
|
:filter(function(_, value) return type(value) == "table" and value.endpoint ~= nil end)
|
|
:fold({}, function(acc, k)
|
|
acc = vim.list_extend({}, acc)
|
|
acc = vim.list_extend(acc, { k })
|
|
return acc
|
|
end)
|
|
|
|
vim.validate({ provider = { M._options.provider, "string", false } })
|
|
|
|
if next(M._options.vendors) ~= nil then
|
|
for k, v in pairs(M._options.vendors) do
|
|
M._options.vendors[k] = type(v) == "function" and v() or v
|
|
end
|
|
vim.validate({ vendors = { M._options.vendors, "table", true } })
|
|
M.providers = vim.list_extend(M.providers, vim.tbl_keys(M._options.vendors))
|
|
end
|
|
end
|
|
|
|
---@param opts table<string, any>
|
|
function M.override(opts)
|
|
vim.validate({ opts = { opts, "table", true } })
|
|
|
|
M._options = vim.tbl_deep_extend("force", M._options, opts or {})
|
|
|
|
if next(M._options.vendors) ~= nil then
|
|
for k, v in pairs(M._options.vendors) do
|
|
M._options.vendors[k] = type(v) == "function" and v() or v
|
|
if not vim.tbl_contains(M.providers, k) then M.providers = vim.list_extend(M.providers, { k }) end
|
|
end
|
|
vim.validate({ vendors = { M._options.vendors, "table", true } })
|
|
end
|
|
end
|
|
|
|
M = setmetatable(M, {
|
|
__index = function(_, k)
|
|
if M._options[k] then return M._options[k] end
|
|
end,
|
|
})
|
|
|
|
function M.support_paste_image() return Utils.has("img-clip.nvim") or Utils.has("img-clip") end
|
|
|
|
function M.get_window_width() return math.ceil(vim.o.columns * (M.windows.width / 100)) end
|
|
|
|
---@param provider Provider
|
|
---@return boolean
|
|
function M.has_provider(provider) return M._options[provider] ~= nil or M.vendors[provider] ~= nil end
|
|
|
|
---get supported providers
|
|
---@param provider Provider
|
|
---@return AvanteProviderFunctor
|
|
function M.get_provider(provider)
|
|
if M._options[provider] ~= nil then
|
|
return vim.deepcopy(M._options[provider], true)
|
|
elseif M.vendors and M.vendors[provider] ~= nil then
|
|
return vim.deepcopy(M.vendors[provider], true)
|
|
else
|
|
error("Failed to find provider: " .. provider, 2)
|
|
end
|
|
end
|
|
|
|
M.BASE_PROVIDER_KEYS = {
|
|
"endpoint",
|
|
"model",
|
|
"deployment",
|
|
"api_version",
|
|
"proxy",
|
|
"allow_insecure",
|
|
"api_key_name",
|
|
"timeout",
|
|
-- internal
|
|
"local",
|
|
"_shellenv",
|
|
"tokenizer_id",
|
|
"use_xml_format",
|
|
"role_map",
|
|
"__inherited_from",
|
|
"disable_tools",
|
|
"entra",
|
|
}
|
|
|
|
return M
|