Review and cleanup

This commit is contained in:
Jonas Widen 2025-03-18 18:38:11 +01:00
parent 050f9bd3b5
commit 2b7717bccf
2 changed files with 136 additions and 26 deletions

View File

@ -16,30 +16,116 @@ local current_suggestion = {
-- Add at the top of the file with other state variables -- Add at the top of the file with other state variables
local completion_cache = { local completion_cache = {
last_line = nil, entries = {},
last_col = nil, max_entries = 1000,
suggestions = {}, ttl = 30 * 60 * 1000, -- 30 minutes TTL
timestamp = 0, last_cleanup = 0
ttl = 30000 -- Cache TTL in milliseconds (30 seconds)
} }
-- Add this function -- Add completion triggers detection
local function get_cached_suggestion(line, col) local function should_trigger_completion(line_text, col)
local config = require("gemini.config")
local trigger_chars = config.options.completion.trigger_characters or "."
-- Don't trigger on empty lines or spaces
if line_text:match("^%s*$") then return false end
-- Check minimum characters
local text_before_cursor = line_text:sub(1, col)
if #text_before_cursor < current_suggestion.min_chars then return false end
-- Check if we're in a comment
local filetype = vim.bo.filetype
local comment_string = vim.bo.commentstring
if comment_string then
local comment_start = comment_string:match("^(.-)%%s")
if comment_start and text_before_cursor:match(vim.pesc(comment_start)) then
return false
end
end
-- Check for trigger characters
local last_char = text_before_cursor:sub(-1)
if trigger_chars:find(last_char, 1, true) then
return true
end
-- Check for meaningful context
local meaningful_pattern = "[%w_][%w_%.%:]*$" -- Matches identifiers and dot/colon access
local context = text_before_cursor:match(meaningful_pattern)
if context and #context >= 2 then
return true
end
return false
end
-- Improved cache key generation
local function get_cache_key(context, line_text, col)
local prefix = line_text:sub(1, col)
local filetype = vim.bo.filetype
local key_parts = {
filetype,
prefix,
vim.inspect(context):sub(1, 100) -- Limited context hash
}
return vim.fn.sha256(table.concat(key_parts, "||"))
end
-- Cache management
local function cleanup_cache()
local now = vim.loop.now() local now = vim.loop.now()
if completion_cache.last_line == line if now - completion_cache.last_cleanup < 60000 then return end -- Cleanup max once per minute
and completion_cache.last_col == col
and (now - completion_cache.timestamp) < completion_cache.ttl then local count = 0
return completion_cache.suggestions local expired = {}
for key, entry in pairs(completion_cache.entries) do
if now - entry.timestamp > completion_cache.ttl then
table.insert(expired, key)
end
count = count + 1
end
-- Remove expired entries
for _, key in ipairs(expired) do
completion_cache.entries[key] = nil
end
-- If still too many entries, remove oldest
if count > completion_cache.max_entries then
local entries = {}
for k, v in pairs(completion_cache.entries) do
table.insert(entries, {key = k, timestamp = v.timestamp})
end
table.sort(entries, function(a, b) return a.timestamp < b.timestamp end)
for i = 1, count - completion_cache.max_entries do
completion_cache.entries[entries[i].key] = nil
end
end
completion_cache.last_cleanup = now
end
-- Cache access
local function get_cached_completion(context, line_text, col)
cleanup_cache()
local key = get_cache_key(context, line_text, col)
local entry = completion_cache.entries[key]
if entry and (vim.loop.now() - entry.timestamp) < completion_cache.ttl then
return entry.completion
end end
return nil return nil
end end
-- Add this function local function cache_completion(context, line_text, col, completion)
local function cache_suggestion(line, col, suggestions) local key = get_cache_key(context, line_text, col)
completion_cache.last_line = line completion_cache.entries[key] = {
completion_cache.last_col = col completion = completion,
completion_cache.suggestions = suggestions timestamp = vim.loop.now()
completion_cache.timestamp = vim.loop.now() }
end end
-- Helper function to get visible lines around cursor -- Helper function to get visible lines around cursor
@ -279,11 +365,29 @@ function M.trigger_completion()
local col = cursor[2] local col = cursor[2]
local current_line = vim.api.nvim_get_current_line() local current_line = vim.api.nvim_get_current_line()
-- Check if we should trigger completion
if not should_trigger_completion(current_line, col) then
clear_suggestion()
return
end
-- Check rate limiting
if should_rate_limit() then return end
-- Get context -- Get context
local visible_lines = vim.api.nvim_buf_get_lines(0, math.max(0, line - 10), line + 10, false) local visible_lines = vim.api.nvim_buf_get_lines(0, math.max(0, line - 10), line + 10, false)
local context, detected_file_type = get_context_around_cursor(visible_lines, 10, current_suggestion.max_context_lines) local context = get_context_around_cursor(visible_lines, 10, current_suggestion.max_context_lines)
-- Create a very specific prompt for raw completion -- Check cache first
local cached_completion = get_cached_completion(context, current_line, col)
if cached_completion then
vim.schedule(function()
show_suggestion(cached_completion, col)
end)
return
end
-- Create prompt for completion
local prompt = string.format([[ local prompt = string.format([[
You are an autocomplete engine. Respond ONLY with the direct completion text. You are an autocomplete engine. Respond ONLY with the direct completion text.
DO NOT include: DO NOT include:
@ -297,19 +401,23 @@ Language: %s
Context: Context:
%s %s
Complete this line: Complete this line:
%s]], detected_file_type, table.concat(context, "\n"), current_line) %s]], vim.bo.filetype, table.concat(context, "\n"), current_line)
-- Make API request -- Make API request
api.get_response(prompt, nil, function(response, error) api.get_response(prompt, nil, function(response, error)
if error then if error then
vim.notify("Completion error: " .. error, vim.log.levels.ERROR) if error:match("code = 429") then
handle_rate_limit()
return
end
return return
end end
if response then if response then
-- Show the raw suggestion directly -- Cache the successful completion
cache_completion(context, current_line, col, response)
vim.schedule(function() vim.schedule(function()
-- Only show if cursor position hasn't changed
local new_cursor = vim.api.nvim_win_get_cursor(0) local new_cursor = vim.api.nvim_win_get_cursor(0)
if new_cursor[1] - 1 == line and new_cursor[2] == col then if new_cursor[1] - 1 == line and new_cursor[2] == col then
show_suggestion(response, col) show_suggestion(response, col)

View File

@ -22,11 +22,13 @@ M.defaults = {
completion = { completion = {
enabled = true, enabled = true,
debounce_ms = 1000, debounce_ms = 1000,
min_chars = 0, min_chars = 2,
max_context_lines = 10, max_context_lines = 10,
style = "ghost", -- or "inline" style = "ghost",
trigger_characters = ".", -- Add trigger characters trigger_characters = ".:", -- Trigger on dot and colon
exclude_filetypes = { "TelescopePrompt", "neo-tree" }, exclude_filetypes = { "TelescopePrompt", "neo-tree" },
cache_ttl = 30 * 60 * 1000, -- 30 minutes
max_cache_entries = 1000,
suggestion_highlight = { suggestion_highlight = {
fg = '#666666', fg = '#666666',
italic = true, italic = true,