aboutsummaryrefslogtreecommitdiff
path: root/runtime/lua/man.lua
diff options
context:
space:
mode:
authorJosh Rahm <joshuarahm@gmail.com>2023-11-30 20:35:25 +0000
committerJosh Rahm <joshuarahm@gmail.com>2023-11-30 20:35:25 +0000
commit1b7b916b7631ddf73c38e3a0070d64e4636cb2f3 (patch)
treecd08258054db80bb9a11b1061bb091c70b76926a /runtime/lua/man.lua
parenteaa89c11d0f8aefbb512de769c6c82f61a8baca3 (diff)
parent4a8bf24ac690004aedf5540fa440e788459e5e34 (diff)
downloadrneovim-aucmd_textputpost.tar.gz
rneovim-aucmd_textputpost.tar.bz2
rneovim-aucmd_textputpost.zip
Merge remote-tracking branch 'upstream/master' into aucmd_textputpostaucmd_textputpost
Diffstat (limited to 'runtime/lua/man.lua')
-rw-r--r--runtime/lua/man.lua236
1 files changed, 129 insertions, 107 deletions
diff --git a/runtime/lua/man.lua b/runtime/lua/man.lua
index 0956022ac6..dcdfc2b87f 100644
--- a/runtime/lua/man.lua
+++ b/runtime/lua/man.lua
@@ -2,6 +2,8 @@ local api, fn = vim.api, vim.fn
local FIND_ARG = '-w'
local localfile_arg = true -- Always use -l if possible. #6683
+
+---@type table[]
local buf_hls = {}
local M = {}
@@ -12,83 +14,32 @@ local function man_error(msg)
end
-- Run a system command and timeout after 30 seconds.
-local function system(cmd_, silent, env)
- local stdout_data = {}
- local stderr_data = {}
- local stdout = vim.loop.new_pipe(false)
- local stderr = vim.loop.new_pipe(false)
-
- local done = false
- local exit_code
-
- -- We use the `env` command here rather than the env option to vim.loop.spawn since spawn will
- -- completely overwrite the environment when we just want to modify the existing one.
- --
- -- Overwriting mainly causes problems NixOS which relies heavily on a non-standard environment.
- local cmd
- if env then
- cmd = { 'env' }
- vim.list_extend(cmd, env)
- vim.list_extend(cmd, cmd_)
- else
- cmd = cmd_
- end
-
- local handle
- handle = vim.loop.spawn(cmd[1], {
- args = vim.list_slice(cmd, 2),
- stdio = { nil, stdout, stderr },
- }, function(code)
- exit_code = code
- stdout:close()
- stderr:close()
- handle:close()
- done = true
- end)
-
- if handle then
- stdout:read_start(function(_, data)
- stdout_data[#stdout_data + 1] = data
- end)
- stderr:read_start(function(_, data)
- stderr_data[#stderr_data + 1] = data
- end)
- else
- stdout:close()
- stderr:close()
- if not silent then
- local cmd_str = table.concat(cmd, ' ')
- man_error(string.format('command error: %s', cmd_str))
- end
- end
-
- vim.wait(30000, function()
- return done
- end)
-
- if not done then
- if handle then
- handle:close()
- stdout:close()
- stderr:close()
- end
- local cmd_str = table.concat(cmd, ' ')
- man_error(string.format('command timed out: %s', cmd_str))
- end
-
- if exit_code ~= 0 and not silent then
+---@param cmd string[]
+---@param silent boolean?
+---@param env? table<string,string|number>
+---@return string
+local function system(cmd, silent, env)
+ local r = vim.system(cmd, { env = env, timeout = 10000 }):wait()
+
+ if r.code ~= 0 and not silent then
local cmd_str = table.concat(cmd, ' ')
- man_error(string.format("command error '%s': %s", cmd_str, table.concat(stderr_data)))
+ man_error(string.format("command error '%s': %s", cmd_str, r.stderr))
end
- return table.concat(stdout_data)
+ return assert(r.stdout)
end
+---@param line string
+---@param linenr integer
local function highlight_line(line, linenr)
+ ---@type string[]
local chars = {}
local prev_char = ''
local overstrike, escape = false, false
+
+ ---@type table<integer,{attr:integer,start:integer,final:integer}>
local hls = {} -- Store highlight groups as { attr, start, final }
+
local NONE, BOLD, UNDERLINE, ITALIC = 0, 1, 2, 3
local hl_groups = { [BOLD] = 'manBold', [UNDERLINE] = 'manUnderline', [ITALIC] = 'manItalic' }
local attr = NONE
@@ -194,11 +145,12 @@ local function highlight_line(line, linenr)
-- We only want to match against SGR sequences, which consist of ESC
-- followed by '[', then a series of parameter and intermediate bytes in
-- the range 0x20 - 0x3f, then 'm'. (See ECMA-48, sections 5.4 & 8.3.117)
+ ---@type string?
local sgr = prev_char:match('^%[([\032-\063]*)m$')
-- Ignore escape sequences with : characters, as specified by ITU's T.416
-- Open Document Architecture and interchange format.
if sgr and not string.find(sgr, ':') then
- local match
+ local match ---@type string?
while sgr and #sgr > 0 do
-- Match against SGR parameters, which may be separated by ';'
match, sgr = sgr:match('^(%d*);?(.*)')
@@ -261,11 +213,16 @@ end
-- intended for PostgreSQL, which has man pages like 'CREATE_TABLE(7)';
-- while editing SQL source code, it's nice to visually select 'CREATE TABLE'
-- and hit 'K', which requires this transformation
+---@param str string
+---@return string
local function spaces_to_underscores(str)
local res = str:gsub('%s', '_')
return res
end
+---@param sect string|nil
+---@param name string|nil
+---@param silent boolean
local function get_path(sect, name, silent)
name = name or ''
sect = sect or ''
@@ -287,7 +244,7 @@ local function get_path(sect, name, silent)
--
-- Finally, we can avoid relying on -S or -s here since they are very
-- inconsistently supported. Instead, call -w with a section and a name.
- local cmd
+ local cmd ---@type string[]
if sect == '' then
cmd = { 'man', FIND_ARG, name }
else
@@ -295,7 +252,7 @@ local function get_path(sect, name, silent)
end
local lines = system(cmd, silent)
- local results = vim.split(lines or {}, '\n', { trimempty = true })
+ local results = vim.split(lines, '\n', { trimempty = true })
if #results == 0 then
return
@@ -310,12 +267,15 @@ local function get_path(sect, name, silent)
end
-- find any that match the specified name
+ ---@param v string
local namematches = vim.tbl_filter(function(v)
- return fn.fnamemodify(v, ':t'):match(name)
+ local tail = fn.fnamemodify(v, ':t')
+ return string.find(tail, name, 1, true)
end, results) or {}
local sectmatches = {}
if #namematches > 0 and sect ~= '' then
+ ---@param v string
sectmatches = vim.tbl_filter(function(v)
return fn.fnamemodify(v, ':e') == sect
end, namematches)
@@ -324,9 +284,12 @@ local function get_path(sect, name, silent)
return fn.substitute(sectmatches[1] or namematches[1] or results[1], [[\n\+$]], '', '')
end
+---@param text string
+---@param pat_or_re string
local function matchstr(text, pat_or_re)
local re = type(pat_or_re) == 'string' and vim.regex(pat_or_re) or pat_or_re
+ ---@type integer, integer
local s, e = re:match_str(text)
if s == nil then
@@ -338,6 +301,8 @@ end
-- attempt to extract the name and sect out of 'name(sect)'
-- otherwise just return the largest string of valid characters in ref
+---@param ref string
+---@return string, string
local function extract_sect_and_name_ref(ref)
ref = ref or ''
if ref:sub(1, 1) == '-' then -- try ':Man -pandoc' with this disabled.
@@ -349,18 +314,18 @@ local function extract_sect_and_name_ref(ref)
if not name then
man_error('manpage reference cannot contain only parentheses: ' .. ref)
end
- return '', spaces_to_underscores(name)
+ return '', name
end
local parts = vim.split(ref1, '(', { plain = true })
-- see ':Man 3X curses' on why tolower.
-- TODO(nhooyr) Not sure if this is portable across OSs
-- but I have not seen a single uppercase section.
local sect = vim.split(parts[2] or '', ')', { plain = true })[1]:lower()
- local name = spaces_to_underscores(parts[1])
+ local name = parts[1]
return sect, name
end
--- verify_exists attempts to find the path to a manpage
+-- find_path attempts to find the path to a manpage
-- based on the passed section and name.
--
-- 1. If manpage could not be found with the given sect and name,
@@ -368,7 +333,10 @@ end
-- 2. If it still could not be found, then we try again without a section.
-- 3. If still not found but $MANSECT is set, then we try again with $MANSECT
-- unset.
-local function verify_exists(sect, name, silent)
+-- 4. If a path still wasn't found, return nil.
+---@param sect string?
+---@param name string
+function M.find_path(sect, name)
if sect and sect ~= '' then
local ret = get_path(sect, name, true)
if ret then
@@ -404,10 +372,8 @@ local function verify_exists(sect, name, silent)
end
end
- if not silent then
- -- finally, if that didn't work, there is no hope
- man_error('no manual entry for ' .. name)
- end
+ -- finally, if that didn't work, there is no hope
+ return nil
end
local EXT_RE = vim.regex([[\.\%([glx]z\|bz2\|lzma\|Z\)$]])
@@ -416,6 +382,8 @@ local EXT_RE = vim.regex([[\.\%([glx]z\|bz2\|lzma\|Z\)$]])
-- more specific than what we provided to `man` (try `:Man 3 App::CLI`).
-- Also on linux, name seems to be case-insensitive. So for `:Man PRIntf`, we
-- still want the name of the buffer to be 'printf'.
+---@param path string
+---@return string, string
local function extract_sect_and_name_path(path)
local tail = fn.fnamemodify(path, ':t')
if EXT_RE:match_str(path) then -- valid extensions
@@ -425,6 +393,7 @@ local function extract_sect_and_name_path(path)
return sect, name
end
+---@return boolean
local function find_man()
if vim.bo.filetype == 'man' then
return true
@@ -442,10 +411,11 @@ local function find_man()
return false
end
+---@param pager boolean
local function set_options(pager)
vim.bo.swapfile = false
vim.bo.buftype = 'nofile'
- vim.bo.bufhidden = 'hide'
+ vim.bo.bufhidden = 'unload'
vim.bo.modified = false
vim.bo.readonly = true
vim.bo.modifiable = false
@@ -453,17 +423,20 @@ local function set_options(pager)
vim.bo.filetype = 'man'
end
+---@param path string
+---@param silent boolean?
+---@return string
local function get_page(path, silent)
-- Disable hard-wrap by using a big $MANWIDTH (max 1000 on some systems #9065).
-- Soft-wrap: ftplugin/man.lua sets wrap/breakindent/….
-- Hard-wrap: driven by `man`.
- local manwidth
+ local manwidth ---@type integer|string
if (vim.g.man_hardwrap or 1) ~= 1 then
manwidth = 999
elseif vim.env.MANWIDTH then
manwidth = vim.env.MANWIDTH
else
- manwidth = api.nvim_win_get_width(0)
+ manwidth = api.nvim_win_get_width(0) - vim.o.wrapmargin
end
local cmd = localfile_arg and { 'man', '-l', path } or { 'man', path }
@@ -472,12 +445,20 @@ local function get_page(path, silent)
-- http://comments.gmane.org/gmane.editors.vim.devel/29085
-- Set MAN_KEEP_FORMATTING so Debian man doesn't discard backspaces.
return system(cmd, silent, {
- 'MANPAGER=cat',
- 'MANWIDTH=' .. manwidth,
- 'MAN_KEEP_FORMATTING=1',
+ MANPAGER = 'cat',
+ MANWIDTH = manwidth,
+ MAN_KEEP_FORMATTING = 1,
})
end
+---@param lnum integer
+---@return string
+local function getline(lnum)
+ ---@diagnostic disable-next-line
+ return fn.getline(lnum)
+end
+
+---@param page string
local function put_page(page)
vim.bo.modifiable = true
vim.bo.readonly = false
@@ -485,7 +466,7 @@ local function put_page(page)
api.nvim_buf_set_lines(0, 0, -1, false, vim.split(page, '\n'))
- while fn.getline(1):match('^%s*$') do
+ while getline(1):match('^%s*$') do
api.nvim_buf_set_lines(0, 0, 1, false, {})
end
-- XXX: nroff justifies text by filling it with whitespace. That interacts
@@ -512,13 +493,21 @@ local function format_candidate(path, psect)
return ''
end
+---@generic T
+---@param list T[]
+---@param elem T
+---@return T[]
local function move_elem_to_head(list, elem)
+ ---@diagnostic disable-next-line:no-unknown
local list1 = vim.tbl_filter(function(v)
return v ~= elem
end, list)
return { elem, unpack(list1) }
end
+---@param sect string
+---@param name string
+---@return string[]
local function get_paths(sect, name)
-- Try several sources for getting the list man directories:
-- 1. `man -w` (works on most systems)
@@ -533,10 +522,11 @@ local function get_paths(sect, name)
end
local mandirs = table.concat(vim.split(mandirs_raw, '[:\n]', { trimempty = true }), ',')
- local paths = fn.globpath(mandirs, 'man?/' .. name .. '*.' .. sect .. '*', false, true)
+ ---@type string[]
+ local paths = fn.globpath(mandirs, 'man[^\\/]*/' .. name .. '*.' .. sect .. '*', false, true)
- -- Prioritize the result from verify_exists as it obeys b:man_default_sects.
- local first = verify_exists(sect, name, true)
+ -- Prioritize the result from find_path as it obeys b:man_default_sects.
+ local first = M.find_path(sect, name)
if first then
paths = move_elem_to_head(paths, first)
end
@@ -544,6 +534,10 @@ local function get_paths(sect, name)
return paths
end
+---@param sect string
+---@param psect string
+---@param name string
+---@return string[]
local function complete(sect, psect, name)
local pages = get_paths(sect, name)
-- We remove duplicates in case the same manpage in different languages was found.
@@ -553,6 +547,8 @@ local function complete(sect, psect, name)
end
-- see extract_sect_and_name_ref on why tolower(sect)
+---@param arg_lead string
+---@param cmd_line string
function M.man_complete(arg_lead, cmd_line, _)
local args = vim.split(cmd_line, '%s+', { trimempty = true })
local cmd_offset = fn.index(args, 'Man')
@@ -589,6 +585,7 @@ function M.man_complete(arg_lead, cmd_line, _)
end
if #args == 2 then
+ ---@type string, string
local name, sect
if arg_lead == '' then
-- cursor (|) is at ':Man 1 |'
@@ -618,10 +615,13 @@ function M.man_complete(arg_lead, cmd_line, _)
return complete(sect, sect, name)
end
+---@param pattern string
+---@return {name:string,filename:string,cmd:string}[]
function M.goto_tag(pattern, _, _)
local sect, name = extract_sect_and_name_ref(pattern)
local paths = get_paths(sect, name)
+ ---@type {name:string,title:string}[]
local structured = {}
for _, path in ipairs(paths) do
@@ -634,6 +634,7 @@ function M.goto_tag(pattern, _, _)
end
end
+ ---@param entry {name:string,title:string}
return vim.tbl_map(function(entry)
return {
name = entry.name,
@@ -645,7 +646,7 @@ end
-- Called when Nvim is invoked as $MANPAGER.
function M.init_pager()
- if fn.getline(1):match('^%s*$') then
+ if getline(1):match('^%s*$') then
api.nvim_buf_set_lines(0, 0, 1, false, {})
else
vim.cmd('keepjumps 1')
@@ -653,7 +654,7 @@ function M.init_pager()
highlight_man_page()
-- Guess the ref from the heading (which is usually uppercase, so we cannot
-- know the correct casing, cf. `man glDrawArraysInstanced`).
- local ref = fn.substitute(matchstr(fn.getline(1), [[^[^)]\+)]]) or '', ' ', '_', 'g')
+ local ref = fn.substitute(matchstr(getline(1), [[^[^)]\+)]]) or '', ' ', '_', 'g')
local ok, res = pcall(extract_sect_and_name_ref, ref)
vim.b.man_sect = ok and res or ''
@@ -664,12 +665,10 @@ function M.init_pager()
set_options(true)
end
+---@param count integer
+---@param args string[]
function M.open_page(count, smods, args)
- if #args > 2 then
- man_error('too many arguments')
- end
-
- local ref
+ local ref ---@type string
if #args == 0 then
ref = vim.bo.filetype == 'man' and fn.expand('<cWORD>') or fn.expand('<cword>')
if ref == '' then
@@ -680,9 +679,19 @@ function M.open_page(count, smods, args)
else
-- Combine the name and sect into a manpage reference so that all
-- verification/extraction can be kept in a single function.
- -- If args[2] is a reference as well, that is fine because it is the only
- -- reference that will match.
- ref = ('%s(%s)'):format(args[2], args[1])
+ if args[1]:match('^%d$') or args[1]:match('^%d%a') or args[1]:match('^%a$') then
+ -- NB: Valid sections are not only digits, but also:
+ -- - <digit><word> (see POSIX mans),
+ -- - and even <letter> and <word> (see, for example, by tcl/tk)
+ -- NB2: don't optimize to :match("^%d"), as it will match manpages like
+ -- 441toppm and others whose name starts with digit
+ local sect = args[1]
+ table.remove(args, 1)
+ local name = table.concat(args, ' ')
+ ref = ('%s(%s)'):format(name, sect)
+ else
+ ref = table.concat(args, ' ')
+ end
end
local sect, name = extract_sect_and_name_ref(ref)
@@ -690,9 +699,16 @@ function M.open_page(count, smods, args)
sect = tostring(count)
end
- local path = verify_exists(sect, name)
- sect, name = extract_sect_and_name_path(path)
+ -- Try both spaces and underscores, use the first that exists.
+ local path = M.find_path(sect, name)
+ if path == nil then
+ path = M.find_path(sect, spaces_to_underscores(name))
+ if path == nil then
+ man_error('no manual entry for ' .. name)
+ end
+ end
+ sect, name = extract_sect_and_name_path(path)
local buf = fn.bufnr()
local save_tfu = vim.bo[buf].tagfunc
vim.bo[buf].tagfunc = "v:lua.require'man'.goto_tag"
@@ -723,7 +739,10 @@ end
-- Called when a man:// buffer is opened.
function M.read_page(ref)
local sect, name = extract_sect_and_name_ref(ref)
- local path = verify_exists(sect, name)
+ local path = M.find_path(sect, name)
+ if path == nil then
+ man_error('no manual entry for ' .. name)
+ end
sect = extract_sect_and_name_path(path)
local page = get_page(path)
vim.b.man_sect = sect
@@ -731,24 +750,27 @@ function M.read_page(ref)
end
function M.show_toc()
- local bufname = fn.bufname('%')
+ local bufnr = api.nvim_get_current_buf()
+ local bufname = api.nvim_buf_get_name(bufnr)
local info = fn.getloclist(0, { winid = 1 })
if info ~= '' and vim.w[info.winid].qf_toc == bufname then
vim.cmd.lopen()
return
end
+ ---@type {bufnr:integer, lnum:integer, text:string}[]
local toc = {}
+
local lnum = 2
local last_line = fn.line('$') - 1
local section_title_re = vim.regex([[^\%( \{3\}\)\=\S.*$]])
local flag_title_re = vim.regex([[^\s\+\%(+\|-\)\S\+]])
while lnum and lnum < last_line do
- local text = fn.getline(lnum)
+ local text = getline(lnum)
if section_title_re:match_str(text) then
-- if text is a section title
toc[#toc + 1] = {
- bufnr = fn.bufnr('%'),
+ bufnr = bufnr,
lnum = lnum,
text = text,
}
@@ -756,7 +778,7 @@ function M.show_toc()
-- if text is a flag title. we strip whitespaces and prepend two
-- spaces to have a consistent format in the loclist.
toc[#toc + 1] = {
- bufnr = fn.bufnr('%'),
+ bufnr = bufnr,
lnum = lnum,
text = ' ' .. fn.substitute(text, [[^\s*\(.\{-}\)\s*$]], [[\1]], ''),
}
@@ -772,7 +794,7 @@ end
local function init()
local path = get_path('', 'man', true)
- local page
+ local page ---@type string?
if path ~= nil then
-- Check for -l support.
page = get_page(path, true)