mirror of
				https://github.com/neovim/neovim.git
				synced 2025-10-26 04:17:01 +00:00 
			
		
		
		
	perf(diagnostic): avoid table copies to filter by severity (#28491)
Instead of adding all diagnostics matching lnum filters to a table, and then copying that table to another table while applying the severity filter, this changes the flow to only add diagnostics matching both filters in the first pass.
This commit is contained in:
		 Mathias Fußenegger
					Mathias Fußenegger
				
			
				
					committed by
					
						 GitHub
						GitHub
					
				
			
			
				
	
			
			
			 GitHub
						GitHub
					
				
			
						parent
						
							a1550dbf0a
						
					
				
				
					commit
					7f084770c2
				
			| @@ -371,6 +371,39 @@ local function to_severity(severity) | |||||||
|   return severity |   return severity | ||||||
| end | end | ||||||
|  |  | ||||||
|  | --- @param severity vim.diagnostic.SeverityFilter | ||||||
|  | --- @return fun(vim.Diagnostic):boolean | ||||||
|  | local function severity_predicate(severity) | ||||||
|  |   if type(severity) ~= 'table' then | ||||||
|  |     severity = assert(to_severity(severity)) | ||||||
|  |     ---@param d vim.Diagnostic | ||||||
|  |     return function(d) | ||||||
|  |       return d.severity == severity | ||||||
|  |     end | ||||||
|  |   end | ||||||
|  |   if severity.min or severity.max then | ||||||
|  |     --- @cast severity {min:vim.diagnostic.Severity,max:vim.diagnostic.Severity} | ||||||
|  |     local min_severity = to_severity(severity.min) or M.severity.HINT | ||||||
|  |     local max_severity = to_severity(severity.max) or M.severity.ERROR | ||||||
|  |  | ||||||
|  |     --- @param d vim.Diagnostic | ||||||
|  |     return function(d) | ||||||
|  |       return d.severity <= min_severity and d.severity >= max_severity | ||||||
|  |     end | ||||||
|  |   end | ||||||
|  |  | ||||||
|  |   --- @cast severity vim.diagnostic.Severity[] | ||||||
|  |   local severities = {} --- @type table<vim.diagnostic.Severity,true> | ||||||
|  |   for _, s in ipairs(severity) do | ||||||
|  |     severities[assert(to_severity(s))] = true | ||||||
|  |   end | ||||||
|  |  | ||||||
|  |   --- @param d vim.Diagnostic | ||||||
|  |   return function(d) | ||||||
|  |     return severities[d.severity] | ||||||
|  |   end | ||||||
|  | end | ||||||
|  |  | ||||||
| --- @param severity vim.diagnostic.SeverityFilter | --- @param severity vim.diagnostic.SeverityFilter | ||||||
| --- @param diagnostics vim.Diagnostic[] | --- @param diagnostics vim.Diagnostic[] | ||||||
| --- @return vim.Diagnostic[] | --- @return vim.Diagnostic[] | ||||||
| @@ -378,37 +411,7 @@ local function filter_by_severity(severity, diagnostics) | |||||||
|   if not severity then |   if not severity then | ||||||
|     return diagnostics |     return diagnostics | ||||||
|   end |   end | ||||||
|  |   return vim.tbl_filter(severity_predicate(severity), diagnostics) | ||||||
|   if type(severity) ~= 'table' then |  | ||||||
|     severity = assert(to_severity(severity)) |  | ||||||
|     --- @param t vim.Diagnostic |  | ||||||
|     return vim.tbl_filter(function(t) |  | ||||||
|       return t.severity == severity |  | ||||||
|     end, diagnostics) |  | ||||||
|   end |  | ||||||
|  |  | ||||||
|   if severity.min or severity.max then |  | ||||||
|     --- @cast severity {min:vim.diagnostic.Severity,max:vim.diagnostic.Severity} |  | ||||||
|     local min_severity = to_severity(severity.min) or M.severity.HINT |  | ||||||
|     local max_severity = to_severity(severity.max) or M.severity.ERROR |  | ||||||
|  |  | ||||||
|     --- @param t vim.Diagnostic |  | ||||||
|     return vim.tbl_filter(function(t) |  | ||||||
|       return t.severity <= min_severity and t.severity >= max_severity |  | ||||||
|     end, diagnostics) |  | ||||||
|   end |  | ||||||
|  |  | ||||||
|   --- @cast severity vim.diagnostic.Severity[] |  | ||||||
|  |  | ||||||
|   local severities = {} --- @type table<vim.diagnostic.Severity,true> |  | ||||||
|   for _, s in ipairs(severity) do |  | ||||||
|     severities[assert(to_severity(s))] = true |  | ||||||
|   end |  | ||||||
|  |  | ||||||
|   --- @param t vim.Diagnostic |  | ||||||
|   return vim.tbl_filter(function(t) |  | ||||||
|     return severities[t.severity] |  | ||||||
|   end, diagnostics) |  | ||||||
| end | end | ||||||
|  |  | ||||||
| --- @param bufnr integer | --- @param bufnr integer | ||||||
| @@ -714,10 +717,18 @@ local function get_diagnostics(bufnr, opts, clamp) | |||||||
|     end, |     end, | ||||||
|   }) |   }) | ||||||
|  |  | ||||||
|  |   local match_severity = opts.severity and severity_predicate(opts.severity) | ||||||
|  |     or function(_) | ||||||
|  |       return true | ||||||
|  |     end | ||||||
|  |  | ||||||
|   ---@param b integer |   ---@param b integer | ||||||
|   ---@param d vim.Diagnostic |   ---@param d vim.Diagnostic | ||||||
|   local function add(b, d) |   local function add(b, d) | ||||||
|     if not opts.lnum or (opts.lnum >= d.lnum and opts.lnum <= (d.end_lnum or d.lnum)) then |     if | ||||||
|  |       match_severity(d) | ||||||
|  |       and (not opts.lnum or (opts.lnum >= d.lnum and opts.lnum <= (d.end_lnum or d.lnum))) | ||||||
|  |     then | ||||||
|       if clamp and api.nvim_buf_is_loaded(b) then |       if clamp and api.nvim_buf_is_loaded(b) then | ||||||
|         local line_count = buf_line_count[b] - 1 |         local line_count = buf_line_count[b] - 1 | ||||||
|         if |         if | ||||||
| @@ -771,10 +782,6 @@ local function get_diagnostics(bufnr, opts, clamp) | |||||||
|     end |     end | ||||||
|   end |   end | ||||||
|  |  | ||||||
|   if opts.severity then |  | ||||||
|     diagnostics = filter_by_severity(opts.severity, diagnostics) |  | ||||||
|   end |  | ||||||
|  |  | ||||||
|   return diagnostics |   return diagnostics | ||||||
| end | end | ||||||
|  |  | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user