-- Functions shared by Nvim and its test-suite. -- -- The singular purpose of this module is to share code with the Nvim -- test-suite. If, in the future, Nvim itself is used to run the test-suite -- instead of "vanilla Lua", these functions could move to src/nvim/lua/vim.lua local vim = {} --- Returns a deep copy of the given object. Non-table objects are copied as --- in a typical Lua assignment, whereas table objects are copied recursively. --- --@param orig Table to copy --@returns New table of copied keys and (nested) values. function vim.deepcopy(orig) end -- luacheck: no unused vim.deepcopy = (function() local function _id(v) return v end local deepcopy_funcs = { table = function(orig) local copy = {} for k, v in pairs(orig) do copy[vim.deepcopy(k)] = vim.deepcopy(v) end return copy end, number = _id, string = _id, ['nil'] = _id, boolean = _id, } return function(orig) return deepcopy_funcs[type(orig)](orig) end end)() --- Splits a string at each instance of a separator. --- --@see |vim.split()| --@see https://www.lua.org/pil/20.2.html --@see http://lua-users.org/wiki/StringLibraryTutorial --- --@param s String to split --@param sep Separator string or pattern --@param plain If `true` use `sep` literally (passed to String.find) --@returns Iterator over the split components function vim.gsplit(s, sep, plain) assert(type(s) == "string", string.format("Expected string, got %s", type(s))) assert(type(sep) == "string", string.format("Expected string, got %s", type(sep))) assert(type(plain) == "boolean" or type(plain) == "nil", string.format("Expected boolean or nil, got %s", type(plain))) local start = 1 local done = false local function _pass(i, j, ...) if i then assert(j+1 > start, "Infinite loop detected") local seg = s:sub(start, i - 1) start = j + 1 return seg, ... else done = true return s:sub(start) end end return function() if done then return end if sep == '' then if start == #s then done = true end return _pass(start+1, start) end return _pass(s:find(sep, start, plain)) end end --- Splits a string at each instance of a separator. --- --- Examples: ---
---  split(":aa::b:", ":")     --> {'','aa','','bb',''}
---  split("axaby", "ab?")     --> {'','x','y'}
---  split(x*yz*o, "*", true)  --> {'x','yz','o'}
--- 
-- --@see |vim.gsplit()| --- --@param s String to split --@param sep Separator string or pattern --@param plain If `true` use `sep` literally (passed to String.find) --@returns List-like table of the split components. function vim.split(s,sep,plain) local t={} for c in vim.gsplit(s, sep, plain) do table.insert(t,c) end return t end --- Checks if a list-like (vector) table contains `value`. --- --@param t Table to check --@param value Value to compare --@returns true if `t` contains `value` function vim.tbl_contains(t, value) assert(type(t) == 'table', string.format("Expected table, got %s", type(t))) for _,v in ipairs(t) do if v == value then return true end end return false end --- Merges two or more map-like tables. --- --@see |extend()| --- --@param behavior Decides what to do if a key is found in more than one map: --- - "error": raise an error --- - "keep": use value from the leftmost map --- - "force": use value from the rightmost map --@param ... Two or more map-like tables. function vim.tbl_extend(behavior, ...) if (behavior ~= 'error' and behavior ~= 'keep' and behavior ~= 'force') then error('invalid "behavior": '..tostring(behavior)) end local ret = {} for i = 1, select('#', ...) do local tbl = select(i, ...) if tbl then for k, v in pairs(tbl) do if behavior ~= 'force' and ret[k] ~= nil then if behavior == 'error' then error('key found in more than one map: '..k) end -- Else behavior is "keep". else ret[k] = v end end end end return ret end --- Creates a copy of a list-like table such that any nested tables are --- "unrolled" and appended to the result. --- --@param t List-like table --@returns Flattened copy of the given list-like table. function vim.tbl_flatten(t) -- From https://github.com/premake/premake-core/blob/master/src/base/table.lua local result = {} local function _tbl_flatten(_t) local n = #_t for i = 1, n do local v = _t[i] if type(v) == "table" then _tbl_flatten(v) elseif v then table.insert(result, v) end end end _tbl_flatten(t) return result end --- Trim whitespace (Lua pattern "%s") from both sides of a string. --- --@see https://www.lua.org/pil/20.2.html --@param s String to trim --@returns String with whitespace removed from its beginning and end function vim.trim(s) assert(type(s) == 'string', string.format("Expected string, got %s", type(s))) return s:match('^%s*(.*%S)') or '' end --- Escapes magic chars in a Lua pattern string. --- --@see https://github.com/rxi/lume --@param s String to escape --@returns %-escaped pattern string function vim.pesc(s) assert(type(s) == 'string', string.format("Expected string, got %s", type(s))) return s:gsub('[%(%)%.%%%+%-%*%?%[%]%^%$]', '%%%1') end --- Type checking validation function --- --- Examples: ---
---  validate({ arg={ { 'foo' }, 'table' }})                                     --> Nop
---  validate({ arg={ 1, 'table' } })                                            --> error("arg: expected table, got number")
---  validate({ arg1={ { 'foo' }, 'table' }, arg2={ 1, 'string' } })             --> error("arg2: expected string, got number")
---  validate({ arg={ 3, function(a) return (a % 2) == 0  end, 'even number' }}) --> error("arg: expected even number, got 3")
--- 
--- ---@param ... Table or list of table. That table is "argument_name = { validation_target, type_name (, whether nil is allowed) }" --- or "argument_name = { validation_target, validation function, expected_description }". --- The following can be used as type_names: --- - table or t --- - string or s --- - number or n --- - boolean or b --- - function or f --- - nil --- - thread --- - userdata function vim.validate(opt) local function _type_name(t) if t == 't' or t == 'table' then return 'table' end if t == 's' or t == 'string' then return 'string' end if t == 'n' or t == 'number' then return 'number' end if t == 'b' or t == 'boolean' then return 'boolean' end if t == 'f' or t == 'function' then return 'function' end if t == 'c' then return 'callable' end if t == 'nil' then return 'nil' end if t == 'thread' or t == 'thread' then return 'thread' end if t == 'userdata' then return 'userdata' end if vim.is_callable(t) then return end error(string.format("Invalid type name '%s'. See \":help validate\" for more info.", t)) end local function _check_type(target, expected_type) if expected_type == 'callable' then return vim.is_callable(target) else return type(target) == expected_type end end for arg, v in pairs(opt) do assert(type(arg) == 'string',string.format('Expected string, got %s', type(arg))) assert(type(v) == 'table', string.format('Expected table, got %s', type(v))) local actual_arg_type = type(v[1]) local expected_type = _type_name(v[2]) if expected_type then if v[3] == true then assert(_check_type(v[1], expected_type) or actual_arg_type == 'nil', string.format("%s: expected %s, got %s", arg, expected_type, actual_arg_type)) else assert(_check_type(v[1], expected_type), string.format("%s: expected %s, got %s", arg, expected_type, actual_arg_type)) end else assert(v[2](v[1]), string.format("%s: expected %s, got %s", arg, v[3], v[1])) end end end --- Return whether an object can call be used as a function. --- --@param f Any type of variable --@return Boolean function vim.is_callable(f) if type(f) == 'function' then return true end local m = getmetatable(f) if m == nil then return false end return type(m.__call) == 'function' end return vim