aboutsummaryrefslogtreecommitdiff
path: root/runtime/lua/vim/treesitter/query.lua
diff options
context:
space:
mode:
authorJaehwang Jung <tomtomjhj@gmail.com>2023-07-03 00:44:21 +0900
committerGitHub <noreply@github.com>2023-07-02 16:44:21 +0100
commit4fd852b8cb88ed035203d3f9ae2e6a8258244974 (patch)
treedc212f0ab08efabc7e951683337edcce1233002a /runtime/lua/vim/treesitter/query.lua
parent2d40f5e843a99a67efd55304d17ccd48d47d9013 (diff)
downloadrneovim-4fd852b8cb88ed035203d3f9ae2e6a8258244974.tar.gz
rneovim-4fd852b8cb88ed035203d3f9ae2e6a8258244974.tar.bz2
rneovim-4fd852b8cb88ed035203d3f9ae2e6a8258244974.zip
perf(treesitter): cache fold query (#24222)
perf(treesitter): cache vim.treesitter.query.get Problem: vim.treesitter.query.get searches and reads query files every time it's called, if user hasn't overridden the query. So this can incur slowdown when called frequently. This can happen when using treesitter foldexpr. For example, when using `:h :range!` in markdown file to format fenced codeblock, on_changedtree in _fold.lua is triggered many times despite that the tree doesn't have syntactic changes (might be a bug in LanguageTree). (Incidentally, the resulting fold is incorrect due to a bug in `:h range!`.) on_changedtree calls vim.treesitter.query.get for each tree changes. In addition, it may request folds queries for injected languages without fold queries, such as markdown_inline. Solution: * Cache the result of vim.treesitter.query.get. * If query file was not found, fail quickly at later calls.
Diffstat (limited to 'runtime/lua/vim/treesitter/query.lua')
-rw-r--r--runtime/lua/vim/treesitter/query.lua30
1 files changed, 24 insertions, 6 deletions
diff --git a/runtime/lua/vim/treesitter/query.lua b/runtime/lua/vim/treesitter/query.lua
index 7f90fa10e8..7610ef7b7f 100644
--- a/runtime/lua/vim/treesitter/query.lua
+++ b/runtime/lua/vim/treesitter/query.lua
@@ -195,6 +195,12 @@ function M.set(lang, query_name, text)
explicit_queries[lang][query_name] = M.parse(lang, text)
end
+--- `false` if query files didn't exist or were empty
+---@type table<string, table<string, Query|false>>
+local query_get_cache = vim.defaulttable(function()
+ return setmetatable({}, { __mode = 'v' })
+end)
+
---@deprecated
function M.get_query(...)
vim.deprecate('vim.treesitter.query.get_query()', 'vim.treesitter.query.get()', '0.10')
@@ -212,16 +218,28 @@ function M.get(lang, query_name)
return explicit_queries[lang][query_name]
end
+ local cached = query_get_cache[lang][query_name]
+ if cached then
+ return cached
+ elseif cached == false then
+ return nil
+ end
+
local query_files = M.get_files(lang, query_name)
local query_string = read_query_files(query_files)
- if #query_string > 0 then
- return M.parse(lang, query_string)
+ if #query_string == 0 then
+ query_get_cache[lang][query_name] = false
+ return nil
end
+
+ local query = M.parse(lang, query_string)
+ query_get_cache[lang][query_name] = query
+ return query
end
----@type {[string]: {[string]: Query}}
-local query_cache = vim.defaulttable(function()
+---@type table<string, table<string, Query>>
+local query_parse_cache = vim.defaulttable(function()
return setmetatable({}, { __mode = 'v' })
end)
@@ -250,7 +268,7 @@ end
---@return Query Parsed query
function M.parse(lang, query)
language.add(lang)
- local cached = query_cache[lang][query]
+ local cached = query_parse_cache[lang][query]
if cached then
return cached
end
@@ -259,7 +277,7 @@ function M.parse(lang, query)
self.query = vim._ts_parse_query(lang, query)
self.info = self.query:inspect()
self.captures = self.info.captures
- query_cache[lang][query] = self
+ query_parse_cache[lang][query] = self
return self
end