patch 9.1.0547: No way to get the arity of a Vim function
Problem:  No way to get the arity of a Vim function
          (Austin Ziegler)
Solution: Enhance get() Vim script function to return the function
          argument info using get(func, "arity") (LemonBoy)
fixes: #15097
closes: #15109
Signed-off-by: LemonBoy <thatlemon@gmail.com>
Signed-off-by: Christian Brabandt <cb@256bit.org>
			
			
This commit is contained in:
		
				
					committed by
					
						 Christian Brabandt
						Christian Brabandt
					
				
			
			
				
	
			
			
			
						parent
						
							03acd4761b
						
					
				
				
					commit
					48b7d05a4f
				
			| @ -1,4 +1,4 @@ | |||||||
| *builtin.txt*	For Vim version 9.1.  Last change: 2024 Jun 23 | *builtin.txt*	For Vim version 9.1.  Last change: 2024 Jul 09 | ||||||
|  |  | ||||||
|  |  | ||||||
| 		  VIM REFERENCE MANUAL	  by Bram Moolenaar | 		  VIM REFERENCE MANUAL	  by Bram Moolenaar | ||||||
| @ -3574,7 +3574,7 @@ garbagecollect([{atexit}])				*garbagecollect()* | |||||||
| 		Return type: |String| | 		Return type: |String| | ||||||
|  |  | ||||||
|  |  | ||||||
| get({list}, {idx} [, {default}])			*get()* | get({list}, {idx} [, {default}])			*get()* *get()-list* | ||||||
| 		Get item {idx} from |List| {list}.  When this item is not | 		Get item {idx} from |List| {list}.  When this item is not | ||||||
| 		available return {default}.  Return zero when {default} is | 		available return {default}.  Return zero when {default} is | ||||||
| 		omitted. | 		omitted. | ||||||
| @ -3583,7 +3583,7 @@ get({list}, {idx} [, {default}])			*get()* | |||||||
| < | < | ||||||
| 		Return type: any, depending on {list} | 		Return type: any, depending on {list} | ||||||
|  |  | ||||||
| get({blob}, {idx} [, {default}]) | get({blob}, {idx} [, {default}])			*get()-blob* | ||||||
| 		Get byte {idx} from |Blob| {blob}.  When this byte is not | 		Get byte {idx} from |Blob| {blob}.  When this byte is not | ||||||
| 		available return {default}.  Return -1 when {default} is | 		available return {default}.  Return -1 when {default} is | ||||||
| 		omitted. | 		omitted. | ||||||
| @ -3592,7 +3592,7 @@ get({blob}, {idx} [, {default}]) | |||||||
| < | < | ||||||
| 		Return type: |Number| | 		Return type: |Number| | ||||||
|  |  | ||||||
| get({dict}, {key} [, {default}]) | get({dict}, {key} [, {default}])			*get()-dict* | ||||||
| 		Get item with key {key} from |Dictionary| {dict}.  When this | 		Get item with key {key} from |Dictionary| {dict}.  When this | ||||||
| 		item is not available return {default}.  Return zero when | 		item is not available return {default}.  Return zero when | ||||||
| 		{default} is omitted.  Useful example: > | 		{default} is omitted.  Useful example: > | ||||||
| @ -3604,18 +3604,32 @@ get({dict}, {key} [, {default}]) | |||||||
| < | < | ||||||
| 		Return type: any, depending on {dict} | 		Return type: any, depending on {dict} | ||||||
|  |  | ||||||
| get({func}, {what}) | get({func}, {what})					*get()-func* | ||||||
| 		Get item {what} from Funcref {func}.  Possible values for | 		Get item {what} from |Funcref| {func}.  Possible values for | ||||||
| 		{what} are: | 		{what} are: | ||||||
| 		  "name"    The function name | 		  "name"    The function name | ||||||
| 		  "func"    The function | 		  "func"    The function | ||||||
| 		  "dict"    The dictionary | 		  "dict"    The dictionary | ||||||
| 		  "args"    The list with arguments | 		  "args"    The list with arguments | ||||||
|  | 		  "arity"   A dictionary with information about the number of | ||||||
|  | 			    arguments accepted by the function (minus the | ||||||
|  | 			    {arglist}) with the following fields: | ||||||
|  | 				required    the number of positional arguments | ||||||
|  | 				optional    the number of optional arguments, | ||||||
|  | 					    in addition to the required ones | ||||||
|  | 				varargs     |TRUE| if the function accepts a | ||||||
|  | 					    variable number of arguments |...| | ||||||
|  |  | ||||||
|  | 				Note: There is no error, if the {arglist} of | ||||||
|  | 				the Funcref contains more arguments than the | ||||||
|  | 				Funcref expects, it's not validated. | ||||||
|  |  | ||||||
| 		Returns zero on error. | 		Returns zero on error. | ||||||
|  |  | ||||||
| 		Preferably used as a |method|: > | 		Preferably used as a |method|: > | ||||||
| 			myfunc->get(what) | 			myfunc->get(what) | ||||||
| < | < | ||||||
| 		Return type: any, depending on {func} | 		Return type: any, depending on {func} and {what} | ||||||
|  |  | ||||||
| 							*getbufinfo()* | 							*getbufinfo()* | ||||||
| getbufinfo([{buf}]) | getbufinfo([{buf}]) | ||||||
|  | |||||||
| @ -7783,6 +7783,10 @@ gdb-version	terminal.txt	/*gdb-version* | |||||||
| ge	motion.txt	/*ge* | ge	motion.txt	/*ge* | ||||||
| gender-neutral	helphelp.txt	/*gender-neutral* | gender-neutral	helphelp.txt	/*gender-neutral* | ||||||
| get()	builtin.txt	/*get()* | get()	builtin.txt	/*get()* | ||||||
|  | get()-blob	builtin.txt	/*get()-blob* | ||||||
|  | get()-dict	builtin.txt	/*get()-dict* | ||||||
|  | get()-func	builtin.txt	/*get()-func* | ||||||
|  | get()-list	builtin.txt	/*get()-list* | ||||||
| get-ms-debuggers	debug.txt	/*get-ms-debuggers* | get-ms-debuggers	debug.txt	/*get-ms-debuggers* | ||||||
| getbufinfo()	builtin.txt	/*getbufinfo()* | getbufinfo()	builtin.txt	/*getbufinfo()* | ||||||
| getbufline()	builtin.txt	/*getbufline()* | getbufline()	builtin.txt	/*getbufline()* | ||||||
|  | |||||||
| @ -1,4 +1,4 @@ | |||||||
| *version9.txt*  For Vim version 9.1.  Last change: 2024 Jul 06 | *version9.txt*  For Vim version 9.1.  Last change: 2024 Jul 08 | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| 		  VIM REFERENCE MANUAL    by Bram Moolenaar | 		  VIM REFERENCE MANUAL    by Bram Moolenaar | ||||||
| @ -41577,6 +41577,8 @@ Changed~ | |||||||
| - 'nrformat' accepts the new "blank" suboption, to determine a signed or | - 'nrformat' accepts the new "blank" suboption, to determine a signed or | ||||||
|   unsigned number based on whitespace in front of a minus sign. |   unsigned number based on whitespace in front of a minus sign. | ||||||
| - allow to specify a priority when defining a new sign |:sign-define| | - allow to specify a priority when defining a new sign |:sign-define| | ||||||
|  | - provide information about function arguments using the get(func, "arity") | ||||||
|  |   function |get()-func| | ||||||
| 
 | 
 | ||||||
| 							*added-9.2* | 							*added-9.2* | ||||||
| Added ~ | Added ~ | ||||||
|  | |||||||
| @ -5134,6 +5134,36 @@ f_get(typval_T *argvars, typval_T *rettv) | |||||||
| 			list_append_tv(rettv->vval.v_list, &pt->pt_argv[i]); | 			list_append_tv(rettv->vval.v_list, &pt->pt_argv[i]); | ||||||
| 		} | 		} | ||||||
| 	    } | 	    } | ||||||
|  | 	    else if (STRCMP(what, "arity") == 0) | ||||||
|  | 	    { | ||||||
|  | 		int required = 0, optional = 0, varargs = FALSE; | ||||||
|  | 		char_u *name = partial_name(pt); | ||||||
|  |  | ||||||
|  | 		get_func_arity(name, &required, &optional, &varargs); | ||||||
|  |  | ||||||
|  | 		rettv->v_type = VAR_DICT; | ||||||
|  | 		if (rettv_dict_alloc(rettv) == OK) | ||||||
|  | 		{ | ||||||
|  | 		    dict_T *dict = rettv->vval.v_dict; | ||||||
|  |  | ||||||
|  | 		    // Take into account the arguments of the partial, if any. | ||||||
|  | 		    // Note that it is possible to supply more arguments than the function | ||||||
|  | 		    // accepts. | ||||||
|  | 		    if (pt->pt_argc >= required + optional) | ||||||
|  | 			required = optional = 0; | ||||||
|  | 		    else if (pt->pt_argc > required) | ||||||
|  | 		    { | ||||||
|  | 			optional -= pt->pt_argc - required; | ||||||
|  | 			required = 0; | ||||||
|  | 		    } | ||||||
|  | 		    else | ||||||
|  | 			required -= pt->pt_argc; | ||||||
|  |  | ||||||
|  | 		    dict_add_number(dict, "required", required); | ||||||
|  | 		    dict_add_number(dict, "optional", optional); | ||||||
|  | 		    dict_add_bool(dict, "varargs", varargs); | ||||||
|  | 		} | ||||||
|  | 	    } | ||||||
| 	    else | 	    else | ||||||
| 		semsg(_(e_invalid_argument_str), what); | 		semsg(_(e_invalid_argument_str), what); | ||||||
|  |  | ||||||
|  | |||||||
| @ -95,4 +95,5 @@ int set_ref_in_call_stack(int copyID); | |||||||
| int set_ref_in_functions(int copyID); | int set_ref_in_functions(int copyID); | ||||||
| int set_ref_in_func_args(int copyID); | int set_ref_in_func_args(int copyID); | ||||||
| int set_ref_in_func(char_u *name, ufunc_T *fp_in, int copyID); | int set_ref_in_func(char_u *name, ufunc_T *fp_in, int copyID); | ||||||
|  | int get_func_arity(char_u *name, int *required, int *optional, int *varargs); | ||||||
| /* vim: set ft=c : */ | /* vim: set ft=c : */ | ||||||
|  | |||||||
| @ -142,20 +142,28 @@ func Test_get_func() | |||||||
|   let l:F = function('tr') |   let l:F = function('tr') | ||||||
|   call assert_equal('tr', get(l:F, 'name')) |   call assert_equal('tr', get(l:F, 'name')) | ||||||
|   call assert_equal(l:F, get(l:F, 'func')) |   call assert_equal(l:F, get(l:F, 'func')) | ||||||
|  |   call assert_equal({'required': 3, 'optional': 0, 'varargs': v:false}, | ||||||
|  |       \ get(l:F, 'arity')) | ||||||
|  |  | ||||||
|   let Fb_func = function('s:FooBar') |   let Fb_func = function('s:FooBar') | ||||||
|   call assert_match('<SNR>\d\+_FooBar', get(Fb_func, 'name')) |   call assert_match('<SNR>\d\+_FooBar', get(Fb_func, 'name')) | ||||||
|  |   call assert_equal({'required': 0, 'optional': 0, 'varargs': v:false}, | ||||||
|  |       \ get(Fb_func, 'arity')) | ||||||
|   let Fb_ref = funcref('s:FooBar') |   let Fb_ref = funcref('s:FooBar') | ||||||
|   call assert_match('<SNR>\d\+_FooBar', get(Fb_ref, 'name')) |   call assert_match('<SNR>\d\+_FooBar', get(Fb_ref, 'name')) | ||||||
|  |   call assert_equal({'required': 0, 'optional': 0, 'varargs': v:false}, | ||||||
|  |       \ get(Fb_ref, 'arity')) | ||||||
|  |  | ||||||
|   call assert_equal({'func has': 'no dict'}, get(l:F, 'dict', {'func has': 'no dict'})) |   call assert_equal({'func has': 'no dict'}, get(l:F, 'dict', {'func has': 'no dict'})) | ||||||
|   call assert_equal(0, get(l:F, 'dict')) |   call assert_equal(0, get(l:F, 'dict')) | ||||||
|   call assert_equal([], get(l:F, 'args')) |   call assert_equal([], get(l:F, 'args')) | ||||||
|  |  | ||||||
|   let NF = test_null_function() |   let NF = test_null_function() | ||||||
|   call assert_equal('', get(NF, 'name')) |   call assert_equal('', get(NF, 'name')) | ||||||
|   call assert_equal(NF, get(NF, 'func')) |   call assert_equal(NF, get(NF, 'func')) | ||||||
|   call assert_equal(0, get(NF, 'dict')) |   call assert_equal(0, get(NF, 'dict')) | ||||||
|   call assert_equal([], get(NF, 'args')) |   call assert_equal([], get(NF, 'args')) | ||||||
|  |   call assert_equal({'required': 0, 'optional': 0, 'varargs': v:false}, get(NF, 'arity')) | ||||||
| endfunc | endfunc | ||||||
|  |  | ||||||
| " get({partial}, {what} [, {default}]) - in test_partial.vim | " get({partial}, {what} [, {default}]) - in test_partial.vim | ||||||
|  | |||||||
| @ -311,6 +311,11 @@ func Test_auto_partial_rebind() | |||||||
| endfunc | endfunc | ||||||
|  |  | ||||||
| func Test_get_partial_items() | func Test_get_partial_items() | ||||||
|  |   func s:Qux(x, y, z=3, w=1, ...) | ||||||
|  |   endfunc | ||||||
|  |   func s:Qux1(x, y) | ||||||
|  |   endfunc | ||||||
|  |  | ||||||
|   let dict = {'name': 'hello'} |   let dict = {'name': 'hello'} | ||||||
|   let args = ["foo", "bar"] |   let args = ["foo", "bar"] | ||||||
|   let Func = function('MyDictFunc') |   let Func = function('MyDictFunc') | ||||||
| @ -331,6 +336,23 @@ func Test_get_partial_items() | |||||||
|   let dict = {'partial has': 'no dict'} |   let dict = {'partial has': 'no dict'} | ||||||
|   call assert_equal(dict, get(P, 'dict', dict)) |   call assert_equal(dict, get(P, 'dict', dict)) | ||||||
|   call assert_equal(0, get(l:P, 'dict')) |   call assert_equal(0, get(l:P, 'dict')) | ||||||
|  |  | ||||||
|  |   call assert_equal({'required': 2, 'optional': 2, 'varargs': v:true}, | ||||||
|  |       \ get(funcref('s:Qux', []), 'arity')) | ||||||
|  |   call assert_equal({'required': 1, 'optional': 2, 'varargs': v:true}, | ||||||
|  |       \ get(funcref('s:Qux', [1]), 'arity')) | ||||||
|  |   call assert_equal({'required': 0, 'optional': 2, 'varargs': v:true}, | ||||||
|  |       \ get(funcref('s:Qux', [1, 2]), 'arity')) | ||||||
|  |   call assert_equal({'required': 0, 'optional': 1, 'varargs': v:true}, | ||||||
|  |       \ get(funcref('s:Qux', [1, 2, 3]), 'arity')) | ||||||
|  |   call assert_equal({'required': 0, 'optional': 0, 'varargs': v:true}, | ||||||
|  |       \ get(funcref('s:Qux', [1, 2, 3, 4]), 'arity')) | ||||||
|  |   " More args than expected is not an error | ||||||
|  |   call assert_equal({'required': 0, 'optional': 0, 'varargs': v:false}, | ||||||
|  |       \ get(funcref('s:Qux1', [1, 2, 3, 4]), 'arity')) | ||||||
|  |  | ||||||
|  |   delfunc s:Qux | ||||||
|  |   delfunc s:Qux1 | ||||||
| endfunc | endfunc | ||||||
|  |  | ||||||
| func Test_compare_partials() | func Test_compare_partials() | ||||||
|  | |||||||
| @ -5503,6 +5503,47 @@ ex_function(exarg_T *eap) | |||||||
|     ga_clear_strings(&lines_to_free); |     ga_clear_strings(&lines_to_free); | ||||||
| } | } | ||||||
|  |  | ||||||
|  |     int | ||||||
|  | get_func_arity(char_u *name, int *required, int *optional, int *varargs) | ||||||
|  | { | ||||||
|  |     ufunc_T	*ufunc = NULL; | ||||||
|  |     int		argcount = 0; | ||||||
|  |     int		min_argcount = 0; | ||||||
|  |     int		idx; | ||||||
|  |  | ||||||
|  |     idx = find_internal_func(name); | ||||||
|  |     if (idx >= 0) | ||||||
|  |     { | ||||||
|  | 	internal_func_get_argcount(idx, &argcount, &min_argcount); | ||||||
|  | 	*varargs = FALSE; | ||||||
|  |     } | ||||||
|  |     else | ||||||
|  |     { | ||||||
|  | 	char_u	fname_buf[FLEN_FIXED + 1]; | ||||||
|  | 	char_u	*tofree = NULL; | ||||||
|  | 	funcerror_T error = FCERR_NONE; | ||||||
|  | 	char_u	*fname; | ||||||
|  |  | ||||||
|  | 	// May need to translate <SNR>123_ to K_SNR. | ||||||
|  | 	fname = fname_trans_sid(name, fname_buf, &tofree, &error); | ||||||
|  | 	if (error == FCERR_NONE) | ||||||
|  | 	    ufunc = find_func(fname, FALSE); | ||||||
|  | 	vim_free(tofree); | ||||||
|  |  | ||||||
|  | 	if (ufunc == NULL) | ||||||
|  | 	    return FAIL; | ||||||
|  |  | ||||||
|  | 	argcount = ufunc->uf_args.ga_len; | ||||||
|  | 	min_argcount = ufunc->uf_args.ga_len - ufunc->uf_def_args.ga_len; | ||||||
|  | 	*varargs = has_varargs(ufunc); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     *required = min_argcount; | ||||||
|  |     *optional = argcount - min_argcount; | ||||||
|  |  | ||||||
|  |     return OK; | ||||||
|  | } | ||||||
|  |  | ||||||
| /* | /* | ||||||
|  * Find a function by name, including "<lambda>123". |  * Find a function by name, including "<lambda>123". | ||||||
|  * Check for "profile" and "debug" arguments and set"compile_type". |  * Check for "profile" and "debug" arguments and set"compile_type". | ||||||
|  | |||||||
| @ -704,6 +704,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 */ | ||||||
|  | /**/ | ||||||
|  |     547, | ||||||
| /**/ | /**/ | ||||||
|     546, |     546, | ||||||
| /**/ | /**/ | ||||||
|  | |||||||
		Reference in New Issue
	
	Block a user