0
0
mirror of https://github.com/vim/vim.git synced 2025-07-26 11:04:33 -04:00

patch 9.0.0303: it is not easy to get information about a script

Problem:    It is not easy to get information about a script.
Solution:   Make getscriptinf() return the version.  When selecting a specific
            script return functions and variables. (Yegappan Lakshmanan,
            closes #10991)
This commit is contained in:
Yegappan Lakshmanan 2022-08-28 18:52:10 +01:00 committed by Bram Moolenaar
parent 75e9a6693e
commit 2f892d8663
7 changed files with 187 additions and 20 deletions

View File

@ -4099,24 +4099,42 @@ getscriptinfo([{opts}) *getscriptinfo()*
scripts in the order they were sourced, like what scripts in the order they were sourced, like what
`:scriptnames` shows. `:scriptnames` shows.
The optional Dict argument {opts} supports the following
optional items:
name Script name match pattern. If specified,
and "sid" is not specified, information about
scripts with name that match the pattern
"name" are returned.
sid Script ID |<SID>|. If specified, only
information about the script with ID "sid" is
returned and "name" is ignored.
Each item in the returned List is a |Dict| with the following Each item in the returned List is a |Dict| with the following
items: items:
autoload set to TRUE for a script that was used with autoload Set to TRUE for a script that was used with
`import autoload` but was not actually sourced `import autoload` but was not actually sourced
yet (see |import-autoload|). yet (see |import-autoload|).
name vim script file name. functions List of script-local function names defined in
sid script ID |<SID>|. the script. Present only when a particular
sourced script ID of the actually sourced script that script is specified using the "sid" item in
{opts}.
name Vim script file name.
sid Script ID |<SID>|.
sourced Script ID of the actually sourced script that
this script name links to, if any, otherwise this script name links to, if any, otherwise
zero zero
version vimscript version (|scriptversion|) variables A dictionary with the script-local variables.
Present only when the a particular script is
The optional Dict argument {opts} supports the following specified using the "sid" item in {opts}.
items: Note that this is a copy, the value of
name script name match pattern. If specified, script-local variables cannot be changed using
information about scripts with name this dictionary.
that match the pattern "name" are returned. version Vimscript version (|scriptversion|)
Examples: >
:echo getscriptinfo({'name': 'myscript'})
:echo getscriptinfo({'sid': 15}).variables
<
gettabinfo([{tabnr}]) *gettabinfo()* gettabinfo([{tabnr}]) *gettabinfo()*
If {tabnr} is not specified, then information about all the If {tabnr} is not specified, then information about all the
tab pages is returned as a |List|. Each List item is a tab pages is returned as a |List|. Each List item is a

View File

@ -1946,6 +1946,53 @@ get_sourced_lnum(
: SOURCING_LNUM; : SOURCING_LNUM;
} }
/*
* Return a List of script-local functions defined in the script with id
* 'sid'.
*/
static list_T *
get_script_local_funcs(scid_T sid)
{
hashtab_T *functbl;
hashitem_T *hi;
long_u todo;
list_T *l;
l = list_alloc();
if (l == NULL)
return NULL;
// Iterate through all the functions in the global function hash table
// looking for functions with script ID 'sid'.
functbl = func_tbl_get();
todo = functbl->ht_used;
for (hi = functbl->ht_array; todo > 0; ++hi)
{
ufunc_T *fp;
if (HASHITEM_EMPTY(hi))
continue;
--todo;
fp = HI2UF(hi);
// Add active functions with script id == 'sid'
if (!(fp->uf_flags & FC_DEAD) && (fp->uf_script_ctx.sc_sid == sid))
{
char_u *name;
if (fp->uf_name_exp != NULL)
name = fp->uf_name_exp;
else
name = fp->uf_name;
list_append_string(l, name, -1);
}
}
return l;
}
/* /*
* getscriptinfo() function * getscriptinfo() function
*/ */
@ -1956,6 +2003,8 @@ f_getscriptinfo(typval_T *argvars, typval_T *rettv)
list_T *l; list_T *l;
char_u *pat = NULL; char_u *pat = NULL;
regmatch_T regmatch; regmatch_T regmatch;
int filterpat = FALSE;
scid_T sid = -1;
if (rettv_list_alloc(rettv) == FAIL) if (rettv_list_alloc(rettv) == FAIL)
return; return;
@ -1970,9 +2019,15 @@ f_getscriptinfo(typval_T *argvars, typval_T *rettv)
if (argvars[0].v_type == VAR_DICT) if (argvars[0].v_type == VAR_DICT)
{ {
pat = dict_get_string(argvars[0].vval.v_dict, "name", TRUE); sid = dict_get_number_def(argvars[0].vval.v_dict, "sid", -1);
if (pat != NULL) if (sid == -1)
regmatch.regprog = vim_regcomp(pat, RE_MAGIC + RE_STRING); {
pat = dict_get_string(argvars[0].vval.v_dict, "name", TRUE);
if (pat != NULL)
regmatch.regprog = vim_regcomp(pat, RE_MAGIC + RE_STRING);
if (regmatch.regprog != NULL)
filterpat = TRUE;
}
} }
for (i = 1; i <= script_items.ga_len; ++i) for (i = 1; i <= script_items.ga_len; ++i)
@ -1983,8 +2038,10 @@ f_getscriptinfo(typval_T *argvars, typval_T *rettv)
if (si->sn_name == NULL) if (si->sn_name == NULL)
continue; continue;
if (pat != NULL && regmatch.regprog != NULL if (filterpat && !vim_regexec(&regmatch, si->sn_name, (colnr_T)0))
&& !vim_regexec(&regmatch, si->sn_name, (colnr_T)0)) continue;
if (sid != -1 && sid != i)
continue; continue;
if ((d = dict_alloc()) == NULL if ((d = dict_alloc()) == NULL
@ -1996,6 +2053,22 @@ f_getscriptinfo(typval_T *argvars, typval_T *rettv)
|| dict_add_bool(d, "autoload", || dict_add_bool(d, "autoload",
si->sn_state == SN_STATE_NOT_LOADED) == FAIL) si->sn_state == SN_STATE_NOT_LOADED) == FAIL)
return; return;
// When a filter pattern is specified to return information about only
// specific script(s), also add the script-local variables and
// functions.
if (sid != -1)
{
dict_T *var_dict;
var_dict = dict_copy(&si->sn_vars->sv_dict, TRUE, TRUE,
get_copyID());
if (var_dict == NULL
|| dict_add_dict(d, "variables", var_dict) == FAIL
|| dict_add_list(d, "functions",
get_script_local_funcs(sid)) == FAIL)
return;
}
} }
vim_regfree(regmatch.regprog); vim_regfree(regmatch.regprog);

View File

@ -32,20 +32,53 @@ endfunc
" Test for the getscriptinfo() function " Test for the getscriptinfo() function
func Test_getscriptinfo() func Test_getscriptinfo()
let lines =<< trim END let lines =<< trim END
scriptversion 3
let g:loaded_script_id = expand("<SID>") let g:loaded_script_id = expand("<SID>")
let s:XscriptVar = [1, #{v: 2}] let s:XscriptVar = [1, #{v: 2}]
func s:XscriptFunc() func s:XgetScriptVar()
return s:XscriptVar
endfunc endfunc
func s:Xscript_legacy_func1()
endfunc
def s:Xscript_def_func1()
enddef
func Xscript_legacy_func2()
endfunc
def Xscript_def_func2()
enddef
END END
call writefile(lines, 'X22script91') call writefile(lines, 'X22script91')
source X22script91 source X22script91
let l = getscriptinfo() let l = getscriptinfo()
call assert_match('X22script91$', l[-1].name) call assert_match('X22script91$', l[-1].name)
call assert_equal(g:loaded_script_id, $"<SNR>{l[-1].sid}_") call assert_equal(g:loaded_script_id, $"<SNR>{l[-1].sid}_")
call assert_equal(3, l[-1].version)
call assert_equal(0, has_key(l[-1], 'variables'))
call assert_equal(0, has_key(l[-1], 'functions'))
let l = getscriptinfo({'name': '22script91'}) " Get script information using script name
let l = getscriptinfo(#{name: '22script91'})
call assert_equal(1, len(l)) call assert_equal(1, len(l))
call assert_match('22script91$', l[0].name) call assert_match('22script91$', l[0].name)
let sid = l[0].sid
" Get script information using script-ID
let l = getscriptinfo({'sid': sid})
call assert_equal(#{XscriptVar: [1, {'v': 2}]}, l[0].variables)
let funcs = ['Xscript_legacy_func2',
\ $"<SNR>{sid}_Xscript_legacy_func1",
\ $"<SNR>{sid}_Xscript_def_func1",
\ 'Xscript_def_func2',
\ $"<SNR>{sid}_XgetScriptVar"]
for f in funcs
call assert_true(index(l[0].functions, f) != -1)
endfor
" Verify that a script-local variable cannot be modified using the dict
" returned by getscriptinfo()
let l[0].variables.XscriptVar = ['n']
let funcname = $"<SNR>{sid}_XgetScriptVar"
call assert_equal([1, {'v': 2}], call(funcname, []))
let l = getscriptinfo({'name': 'foobar'}) let l = getscriptinfo({'name': 'foobar'})
call assert_equal(0, len(l)) call assert_equal(0, len(l))
@ -58,6 +91,8 @@ func Test_getscriptinfo()
call assert_true(len(l) > 1) call assert_true(len(l) > 1)
call assert_fails("echo getscriptinfo('foobar')", 'E1206:') call assert_fails("echo getscriptinfo('foobar')", 'E1206:')
call assert_fails("echo getscriptinfo({'sid': []})", 'E745:')
call delete('X22script91') call delete('X22script91')
endfunc endfunc

View File

@ -1898,6 +1898,46 @@ enddef
def Test_getscriptinfo() def Test_getscriptinfo()
v9.CheckDefAndScriptFailure(['getscriptinfo("x")'], ['E1013: Argument 1: type mismatch, expected dict<any> but got string', 'E1206: Dictionary required for argument 1']) v9.CheckDefAndScriptFailure(['getscriptinfo("x")'], ['E1013: Argument 1: type mismatch, expected dict<any> but got string', 'E1206: Dictionary required for argument 1'])
var lines1 =<< trim END
vim9script
g:loaded_script_id = expand("<SID>")
var XscriptVar = [1, {v: 2}]
func XgetScriptVar()
return XscriptVar
endfunc
func Xscript_legacy_func1()
endfunc
def Xscript_def_func1()
enddef
func g:Xscript_legacy_func2()
endfunc
def g:Xscript_def_func2()
enddef
END
writefile(lines1, 'X22script92')
var lines2 =<< trim END
source X22script92
var sid = matchstr(g:loaded_script_id, '<SNR>\zs\d\+\ze_')->str2nr()
var l = getscriptinfo({sid: sid, name: 'ignored'})
assert_match('X22script92$', l[0].name)
assert_equal(g:loaded_script_id, $"<SNR>{l[0].sid}_")
assert_equal(999999, l[0].version)
assert_equal(0, l[0].sourced)
assert_equal({XscriptVar: [1, {v: 2}]}, l[0].variables)
var funcs = ['Xscript_legacy_func2',
$"<SNR>{sid}_Xscript_legacy_func1",
$"<SNR>{sid}_Xscript_def_func1",
'Xscript_def_func2',
$"<SNR>{sid}_XgetScriptVar"]
for f in funcs
assert_true(index(l[0].functions, f) != -1)
endfor
END
v9.CheckDefAndScriptSuccess(lines2)
delete('X22script92')
enddef enddef
def Test_gettabinfo() def Test_gettabinfo()

View File

@ -741,6 +741,7 @@ def Test_use_relative_autoload_import_in_mapping()
assert_true(len(l) == 1) assert_true(len(l) == 1)
assert_match('XrelautoloadExport.vim$', l[0].name) assert_match('XrelautoloadExport.vim$', l[0].name)
assert_false(l[0].autoload) assert_false(l[0].autoload)
assert_equal(999999, l[0].version)
unlet g:result unlet g:result
delete('XrelautoloadExport.vim') delete('XrelautoloadExport.vim')

View File

@ -40,7 +40,6 @@ func_init()
hash_init(&func_hashtab); hash_init(&func_hashtab);
} }
#if defined(FEAT_PROFILE) || defined(PROTO)
/* /*
* Return the function hash table * Return the function hash table
*/ */
@ -49,7 +48,6 @@ func_tbl_get(void)
{ {
return &func_hashtab; return &func_hashtab;
} }
#endif
/* /*
* Get one function argument. * Get one function argument.

View File

@ -707,6 +707,8 @@ static char *(features[]) =
static int included_patches[] = static int included_patches[] =
{ /* Add new patch number below this line */ { /* Add new patch number below this line */
/**/
303,
/**/ /**/
302, 302,
/**/ /**/