mirror of
https://github.com/neovim/neovim.git
synced 2025-10-03 00:18:33 +00:00
API: nvim_call_dict_function #3032
This commit is contained in:

committed by
Justin M. Keyes

parent
f46f138fb6
commit
124275dd58
@@ -297,6 +297,18 @@ Object nvim_eval(String expr, Error *err)
|
|||||||
/// @return Result of the function call
|
/// @return Result of the function call
|
||||||
Object nvim_call_function(String fname, Array args, Error *err)
|
Object nvim_call_function(String fname, Array args, Error *err)
|
||||||
FUNC_API_SINCE(1)
|
FUNC_API_SINCE(1)
|
||||||
|
{
|
||||||
|
return call_function(fname, args, NULL, err);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Call an internal or user defined function.
|
||||||
|
///
|
||||||
|
/// @param fname Function name
|
||||||
|
/// @param args Function arguments
|
||||||
|
/// @param self `self` dict (only required for dict functions)
|
||||||
|
/// @param[out] err Details of an error that may have occurred
|
||||||
|
/// @return Result of the function call
|
||||||
|
static Object call_function(String fname, Array args, dict_T *self, Error *err)
|
||||||
{
|
{
|
||||||
Object rv = OBJECT_INIT;
|
Object rv = OBJECT_INIT;
|
||||||
if (args.size > MAX_FUNC_ARGS) {
|
if (args.size > MAX_FUNC_ARGS) {
|
||||||
@@ -321,7 +333,7 @@ Object nvim_call_function(String fname, Array args, Error *err)
|
|||||||
int r = call_func((char_u *)fname.data, (int)fname.size,
|
int r = call_func((char_u *)fname.data, (int)fname.size,
|
||||||
&rettv, (int)args.size, vim_args, NULL,
|
&rettv, (int)args.size, vim_args, NULL,
|
||||||
curwin->w_cursor.lnum, curwin->w_cursor.lnum, &dummy,
|
curwin->w_cursor.lnum, curwin->w_cursor.lnum, &dummy,
|
||||||
true, NULL, NULL);
|
true, NULL, self);
|
||||||
if (r == FAIL) {
|
if (r == FAIL) {
|
||||||
api_set_error(err, kErrorTypeException, "Error calling function.");
|
api_set_error(err, kErrorTypeException, "Error calling function.");
|
||||||
}
|
}
|
||||||
@@ -356,6 +368,98 @@ Object nvim_execute_lua(String code, Array args, Error *err)
|
|||||||
return executor_exec_lua_api(code, args, err);
|
return executor_exec_lua_api(code, args, err);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Call the given dict function with the given arguments stored in an array.
|
||||||
|
///
|
||||||
|
/// @param self |self| dict or string expression evaluating to a dict
|
||||||
|
/// @param internal true if the function is stored in the self-dict
|
||||||
|
/// @param fnname Function to call
|
||||||
|
/// @param args Functions arguments packed in an Array
|
||||||
|
/// @param[out] err Details of an error that may have occurred
|
||||||
|
/// @return Result of the function call
|
||||||
|
Object nvim_call_dict_function(Object self, Boolean internal, String fnname,
|
||||||
|
Array args, Error *err)
|
||||||
|
FUNC_API_SINCE(4)
|
||||||
|
{
|
||||||
|
Object rv = OBJECT_INIT;
|
||||||
|
|
||||||
|
typval_T rettv;
|
||||||
|
bool mustfree = false;
|
||||||
|
switch (self.type) {
|
||||||
|
case kObjectTypeString: {
|
||||||
|
try_start();
|
||||||
|
if (eval0((char_u *)self.data.string.data, &rettv, NULL, true) == FAIL) {
|
||||||
|
api_set_error(err, kErrorTypeException,
|
||||||
|
"Failed to evaluate self expression");
|
||||||
|
}
|
||||||
|
if (try_end(err)) {
|
||||||
|
return rv;
|
||||||
|
}
|
||||||
|
// Evaluation of the string arg created a new dict or increased the
|
||||||
|
// refcount of a dict. Not necessary for a dict inside a RPC Object.
|
||||||
|
mustfree = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case kObjectTypeDictionary: {
|
||||||
|
if (internal) {
|
||||||
|
api_set_error(err, kErrorTypeValidation,
|
||||||
|
"Funcrefs are not supported for RPC dicts");
|
||||||
|
return rv;
|
||||||
|
} else if (!object_to_vim(self, &rettv, err)) {
|
||||||
|
tv_clear(&rettv);
|
||||||
|
return rv;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
default: {
|
||||||
|
api_set_error(err, kErrorTypeValidation,
|
||||||
|
"self argument must be String or Dictionary");
|
||||||
|
return rv;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
dict_T *self_dict = rettv.vval.v_dict;
|
||||||
|
if (rettv.v_type != VAR_DICT || !self_dict) {
|
||||||
|
api_set_error(err, kErrorTypeValidation,
|
||||||
|
"Referenced self-dict does not exist");
|
||||||
|
goto end;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Set the function to call
|
||||||
|
String func = STRING_INIT;
|
||||||
|
if (internal /* && self.type == kObjectTypeString */) {
|
||||||
|
dictitem_T *const di = tv_dict_find(self_dict, fnname.data,
|
||||||
|
(ptrdiff_t)fnname.size);
|
||||||
|
if (di == NULL) {
|
||||||
|
api_set_error(err, kErrorTypeValidation,
|
||||||
|
"Function not found in self-dict");
|
||||||
|
goto end;
|
||||||
|
}
|
||||||
|
if (di->di_tv.v_type != VAR_STRING) {
|
||||||
|
api_set_error(err, kErrorTypeValidation,
|
||||||
|
"Value inside self-dict is not a valid function name");
|
||||||
|
goto end;
|
||||||
|
}
|
||||||
|
func.data = (char *)di->di_tv.vval.v_string;
|
||||||
|
func.size = strlen(func.data);
|
||||||
|
} else {
|
||||||
|
func.data = fnname.data;
|
||||||
|
func.size = fnname.size;
|
||||||
|
}
|
||||||
|
if (!func.data || func.size < 1) {
|
||||||
|
api_set_error(err, kErrorTypeValidation,
|
||||||
|
"Invalid (empty) function name");
|
||||||
|
goto end;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Finally try to call the function
|
||||||
|
rv = call_function(func, args, self_dict, err);
|
||||||
|
end:
|
||||||
|
if (mustfree) {
|
||||||
|
tv_clear(&rettv);
|
||||||
|
}
|
||||||
|
|
||||||
|
return rv;
|
||||||
|
}
|
||||||
|
|
||||||
/// Calculates the number of display cells occupied by `text`.
|
/// Calculates the number of display cells occupied by `text`.
|
||||||
/// <Tab> counts as one cell.
|
/// <Tab> counts as one cell.
|
||||||
///
|
///
|
||||||
|
@@ -4,17 +4,19 @@ local global_helpers = require('test.helpers')
|
|||||||
|
|
||||||
local NIL = helpers.NIL
|
local NIL = helpers.NIL
|
||||||
local clear, nvim, eq, neq = helpers.clear, helpers.nvim, helpers.eq, helpers.neq
|
local clear, nvim, eq, neq = helpers.clear, helpers.nvim, helpers.eq, helpers.neq
|
||||||
|
local command = helpers.command
|
||||||
|
local funcs = helpers.funcs
|
||||||
|
local iswin = helpers.iswin
|
||||||
|
local meth_pcall = helpers.meth_pcall
|
||||||
|
local meths = helpers.meths
|
||||||
local ok, nvim_async, feed = helpers.ok, helpers.nvim_async, helpers.feed
|
local ok, nvim_async, feed = helpers.ok, helpers.nvim_async, helpers.feed
|
||||||
local os_name = helpers.os_name
|
local os_name = helpers.os_name
|
||||||
local meths = helpers.meths
|
|
||||||
local funcs = helpers.funcs
|
|
||||||
local request = helpers.request
|
local request = helpers.request
|
||||||
local meth_pcall = helpers.meth_pcall
|
local source = helpers.source
|
||||||
local command = helpers.command
|
|
||||||
local iswin = helpers.iswin
|
|
||||||
|
|
||||||
local intchar2lua = global_helpers.intchar2lua
|
local expect_err = global_helpers.expect_err
|
||||||
local format_string = global_helpers.format_string
|
local format_string = global_helpers.format_string
|
||||||
|
local intchar2lua = global_helpers.intchar2lua
|
||||||
local mergedicts_copy = global_helpers.mergedicts_copy
|
local mergedicts_copy = global_helpers.mergedicts_copy
|
||||||
|
|
||||||
describe('api', function()
|
describe('api', function()
|
||||||
@@ -177,6 +179,30 @@ describe('api', function()
|
|||||||
end)
|
end)
|
||||||
end)
|
end)
|
||||||
|
|
||||||
|
describe('nvim_call_dict_function', function()
|
||||||
|
it('invokes VimL dict', function()
|
||||||
|
source('function! F(name) dict\n return self.greeting . ", " . a:name . "!"\nendfunction')
|
||||||
|
nvim('set_var', 'dict_function_dict', { greeting = 'Hello', F = 'function("F")' })
|
||||||
|
eq('Hello, World!', nvim('call_dict_function', 'g:dict_function_dict', false, 'F', {'World'}))
|
||||||
|
eq({ greeting = 'Hello', F = 'function("F")' }, nvim('get_var', 'dict_function_dict'))
|
||||||
|
nvim('set_var', 'dict_function_dict_i', { greeting = 'Hi', F = "F" })
|
||||||
|
eq('Hi, Moon!', nvim('call_dict_function', 'g:dict_function_dict_i', true, 'F', {'Moon'}))
|
||||||
|
eq({ greeting = 'Hi', F = "F" }, nvim('get_var', 'dict_function_dict_i'))
|
||||||
|
end)
|
||||||
|
it('invokes RPC dict', function()
|
||||||
|
source('function! G() dict\n return self.result\nendfunction')
|
||||||
|
eq('self', nvim('call_dict_function', { result = 'self', G = 'G'}, false, 'G', {}))
|
||||||
|
end)
|
||||||
|
it('fails for a RPC dictionary and internal set to true', function()
|
||||||
|
expect_err('Funcrefs are not supported for RPC dicts', request,
|
||||||
|
'nvim_call_dict_function', { f = '' }, true, 'f', {1,2})
|
||||||
|
end)
|
||||||
|
it('fails with empty function name', function()
|
||||||
|
expect_err('Invalid %(empty%) function name', request,
|
||||||
|
'nvim_call_dict_function', "{ 'f': '' }", true, 'f', {1,2})
|
||||||
|
end)
|
||||||
|
end)
|
||||||
|
|
||||||
describe('nvim_execute_lua', function()
|
describe('nvim_execute_lua', function()
|
||||||
it('works', function()
|
it('works', function()
|
||||||
meths.execute_lua('vim.api.nvim_set_var("test", 3)', {})
|
meths.execute_lua('vim.api.nvim_set_var("test", 3)', {})
|
||||||
|
Reference in New Issue
Block a user