15d7ead3bSBram Moolenaar" Test various aspects of the Vim script language.
2b544f3c8SBram Moolenaar" Most of this was formerly in test49.
3b544f3c8SBram Moolenaar
48c5a278fSBram Moolenaarsource check.vim
593344c2dSBram Moolenaarsource shared.vim
68c5a278fSBram Moolenaar
7b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
8b544f3c8SBram Moolenaar" Test environment							    {{{1
9b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
10b544f3c8SBram Moolenaar
11b544f3c8SBram Moolenaarcom!		   XpathINIT  let g:Xpath = ''
12b544f3c8SBram Moolenaarcom! -nargs=1 -bar Xpath      let g:Xpath = g:Xpath . <args>
13b544f3c8SBram Moolenaar
14b544f3c8SBram Moolenaar" Append a message to the "messages" file
151e115360SBram Moolenaarfunc Xout(text)
16b544f3c8SBram Moolenaar    split messages
17b544f3c8SBram Moolenaar    $put =a:text
18b544f3c8SBram Moolenaar    wq
19b544f3c8SBram Moolenaarendfunc
20b544f3c8SBram Moolenaar
21b544f3c8SBram Moolenaarcom! -nargs=1	     Xout     call Xout(<args>)
22b544f3c8SBram Moolenaar
23b544f3c8SBram Moolenaar" MakeScript() - Make a script file from a function.			    {{{2
24b544f3c8SBram Moolenaar"
25b544f3c8SBram Moolenaar" Create a script that consists of the body of the function a:funcname.
26b544f3c8SBram Moolenaar" Replace any ":return" by a ":finish", any argument variable by a global
27e21c1580SBram Moolenaar" variable, and every ":call" by a ":source" for the next following argument
28b544f3c8SBram Moolenaar" in the variable argument list.  This function is useful if similar tests are
29b544f3c8SBram Moolenaar" to be made for a ":return" from a function call or a ":finish" in a script
30b544f3c8SBram Moolenaar" file.
311e115360SBram Moolenaarfunc MakeScript(funcname, ...)
32b544f3c8SBram Moolenaar    let script = tempname()
33b544f3c8SBram Moolenaar    execute "redir! >" . script
34b544f3c8SBram Moolenaar    execute "function" a:funcname
35b544f3c8SBram Moolenaar    redir END
36b544f3c8SBram Moolenaar    execute "edit" script
37b544f3c8SBram Moolenaar    " Delete the "function" and the "endfunction" lines.  Do not include the
38b544f3c8SBram Moolenaar    " word "function" in the pattern since it might be translated if LANG is
39b544f3c8SBram Moolenaar    " set.  When MakeScript() is being debugged, this deletes also the debugging
40b544f3c8SBram Moolenaar    " output of its line 3 and 4.
41b544f3c8SBram Moolenaar    exec '1,/.*' . a:funcname . '(.*)/d'
42b544f3c8SBram Moolenaar    /^\d*\s*endfunction\>/,$d
43b544f3c8SBram Moolenaar    %s/^\d*//e
44b544f3c8SBram Moolenaar    %s/return/finish/e
45b544f3c8SBram Moolenaar    %s/\<a:\(\h\w*\)/g:\1/ge
46b544f3c8SBram Moolenaar    normal gg0
47b544f3c8SBram Moolenaar    let cnt = 0
48b544f3c8SBram Moolenaar    while search('\<call\s*\%(\u\|s:\)\w*\s*(.*)', 'W') > 0
49b544f3c8SBram Moolenaar	let cnt = cnt + 1
50b544f3c8SBram Moolenaar	s/\<call\s*\%(\u\|s:\)\w*\s*(.*)/\='source ' . a:{cnt}/
51b544f3c8SBram Moolenaar    endwhile
52b544f3c8SBram Moolenaar    g/^\s*$/d
53b544f3c8SBram Moolenaar    write
54b544f3c8SBram Moolenaar    bwipeout
55b544f3c8SBram Moolenaar    return script
561e115360SBram Moolenaarendfunc
57b544f3c8SBram Moolenaar
58b544f3c8SBram Moolenaar" ExecAsScript - Source a temporary script made from a function.	    {{{2
59b544f3c8SBram Moolenaar"
60b544f3c8SBram Moolenaar" Make a temporary script file from the function a:funcname, ":source" it, and
61b544f3c8SBram Moolenaar" delete it afterwards.  However, if an exception is thrown the file may remain,
62b544f3c8SBram Moolenaar" the caller should call DeleteTheScript() afterwards.
63b544f3c8SBram Moolenaarlet s:script_name = ''
64b544f3c8SBram Moolenaarfunction! ExecAsScript(funcname)
65b544f3c8SBram Moolenaar    " Make a script from the function passed as argument.
66b544f3c8SBram Moolenaar    let s:script_name = MakeScript(a:funcname)
67b544f3c8SBram Moolenaar
68b544f3c8SBram Moolenaar    " Source and delete the script.
69b544f3c8SBram Moolenaar    exec "source" s:script_name
70b544f3c8SBram Moolenaar    call delete(s:script_name)
71b544f3c8SBram Moolenaar    let s:script_name = ''
72b544f3c8SBram Moolenaarendfunction
73b544f3c8SBram Moolenaar
74b544f3c8SBram Moolenaarfunction! DeleteTheScript()
75b544f3c8SBram Moolenaar    if s:script_name
76b544f3c8SBram Moolenaar	call delete(s:script_name)
77b544f3c8SBram Moolenaar	let s:script_name = ''
78b544f3c8SBram Moolenaar    endif
79b544f3c8SBram Moolenaarendfunc
80b544f3c8SBram Moolenaar
81b544f3c8SBram Moolenaarcom! -nargs=1 -bar ExecAsScript call ExecAsScript(<f-args>)
82b544f3c8SBram Moolenaar
83b544f3c8SBram Moolenaar
84b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
85b544f3c8SBram Moolenaar" Test 1:   :endwhile in function					    {{{1
86b544f3c8SBram Moolenaar"
87b544f3c8SBram Moolenaar"	    Detect if a broken loop is (incorrectly) reactivated by the
88b544f3c8SBram Moolenaar"	    :endwhile.  Use a :return to prevent an endless loop, and make
89b544f3c8SBram Moolenaar"	    this test first to get a meaningful result on an error before other
90b544f3c8SBram Moolenaar"	    tests will hang.
91b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
92b544f3c8SBram Moolenaar
93b544f3c8SBram Moolenaarfunction! T1_F()
94b544f3c8SBram Moolenaar    Xpath 'a'
95b544f3c8SBram Moolenaar    let first = 1
96b544f3c8SBram Moolenaar    while 1
97b544f3c8SBram Moolenaar	Xpath 'b'
98b544f3c8SBram Moolenaar	if first
99b544f3c8SBram Moolenaar	    Xpath 'c'
100b544f3c8SBram Moolenaar	    let first = 0
101b544f3c8SBram Moolenaar	    break
102b544f3c8SBram Moolenaar	else
103b544f3c8SBram Moolenaar	    Xpath 'd'
104b544f3c8SBram Moolenaar	    return
105b544f3c8SBram Moolenaar	endif
106b544f3c8SBram Moolenaar    endwhile
107b544f3c8SBram Moolenaarendfunction
108b544f3c8SBram Moolenaar
109b544f3c8SBram Moolenaarfunction! T1_G()
110b544f3c8SBram Moolenaar    Xpath 'h'
111b544f3c8SBram Moolenaar    let first = 1
112b544f3c8SBram Moolenaar    while 1
113b544f3c8SBram Moolenaar	Xpath 'i'
114b544f3c8SBram Moolenaar	if first
115b544f3c8SBram Moolenaar	    Xpath 'j'
116b544f3c8SBram Moolenaar	    let first = 0
117b544f3c8SBram Moolenaar	    break
118b544f3c8SBram Moolenaar	else
119b544f3c8SBram Moolenaar	    Xpath 'k'
120b544f3c8SBram Moolenaar	    return
121b544f3c8SBram Moolenaar	endif
122b544f3c8SBram Moolenaar	if 1	" unmatched :if
123b544f3c8SBram Moolenaar    endwhile
124b544f3c8SBram Moolenaarendfunction
125b544f3c8SBram Moolenaar
126b544f3c8SBram Moolenaarfunc Test_endwhile_function()
127b544f3c8SBram Moolenaar  XpathINIT
128b544f3c8SBram Moolenaar  call T1_F()
129b544f3c8SBram Moolenaar  Xpath 'F'
130b544f3c8SBram Moolenaar
131b544f3c8SBram Moolenaar  try
132b544f3c8SBram Moolenaar    call T1_G()
133b544f3c8SBram Moolenaar  catch
134b544f3c8SBram Moolenaar    " Catch missing :endif
135b544f3c8SBram Moolenaar    call assert_true(v:exception =~ 'E171')
136b544f3c8SBram Moolenaar    Xpath 'x'
137b544f3c8SBram Moolenaar  endtry
138b544f3c8SBram Moolenaar  Xpath 'G'
139b544f3c8SBram Moolenaar
140b544f3c8SBram Moolenaar  call assert_equal('abcFhijxG', g:Xpath)
141b544f3c8SBram Moolenaarendfunc
142b544f3c8SBram Moolenaar
143b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
144b544f3c8SBram Moolenaar" Test 2:   :endwhile in script						    {{{1
145b544f3c8SBram Moolenaar"
146b544f3c8SBram Moolenaar"	    Detect if a broken loop is (incorrectly) reactivated by the
147b544f3c8SBram Moolenaar"	    :endwhile.  Use a :finish to prevent an endless loop, and place
148b544f3c8SBram Moolenaar"	    this test before others that might hang to get a meaningful result
149b544f3c8SBram Moolenaar"	    on an error.
150b544f3c8SBram Moolenaar"
151b544f3c8SBram Moolenaar"	    This test executes the bodies of the functions T1_F and T1_G from
152b544f3c8SBram Moolenaar"	    the previous test as script files (:return replaced by :finish).
153b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
154b544f3c8SBram Moolenaar
155b544f3c8SBram Moolenaarfunc Test_endwhile_script()
156b544f3c8SBram Moolenaar  XpathINIT
157b544f3c8SBram Moolenaar  ExecAsScript T1_F
158b544f3c8SBram Moolenaar  Xpath 'F'
159b544f3c8SBram Moolenaar  call DeleteTheScript()
160b544f3c8SBram Moolenaar
161b544f3c8SBram Moolenaar  try
162b544f3c8SBram Moolenaar    ExecAsScript T1_G
163b544f3c8SBram Moolenaar  catch
164b544f3c8SBram Moolenaar    " Catch missing :endif
165b544f3c8SBram Moolenaar    call assert_true(v:exception =~ 'E171')
166b544f3c8SBram Moolenaar    Xpath 'x'
167b544f3c8SBram Moolenaar  endtry
168b544f3c8SBram Moolenaar  Xpath 'G'
169b544f3c8SBram Moolenaar  call DeleteTheScript()
170b544f3c8SBram Moolenaar
171b544f3c8SBram Moolenaar  call assert_equal('abcFhijxG', g:Xpath)
172b544f3c8SBram Moolenaarendfunc
173b544f3c8SBram Moolenaar
174b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
175b544f3c8SBram Moolenaar" Test 3:   :if, :elseif, :while, :continue, :break			    {{{1
176b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
177b544f3c8SBram Moolenaar
178b544f3c8SBram Moolenaarfunction Test_if_while()
179b544f3c8SBram Moolenaar    XpathINIT
180b544f3c8SBram Moolenaar    if 1
181b544f3c8SBram Moolenaar	Xpath 'a'
182b544f3c8SBram Moolenaar	let loops = 3
183b544f3c8SBram Moolenaar	while loops > -1	    " main loop: loops == 3, 2, 1 (which breaks)
184b544f3c8SBram Moolenaar	    if loops <= 0
185b544f3c8SBram Moolenaar		let break_err = 1
186b544f3c8SBram Moolenaar		let loops = -1
187b544f3c8SBram Moolenaar	    else
188b544f3c8SBram Moolenaar		Xpath 'b' . loops
189b544f3c8SBram Moolenaar	    endif
190b544f3c8SBram Moolenaar	    if (loops == 2)
191b544f3c8SBram Moolenaar		while loops == 2 " dummy loop
192b544f3c8SBram Moolenaar		    Xpath 'c' . loops
193b544f3c8SBram Moolenaar		    let loops = loops - 1
194b544f3c8SBram Moolenaar		    continue    " stop dummy loop
195b544f3c8SBram Moolenaar		    Xpath 'd' . loops
196b544f3c8SBram Moolenaar		endwhile
197b544f3c8SBram Moolenaar		continue	    " continue main loop
198b544f3c8SBram Moolenaar		Xpath 'e' . loops
199b544f3c8SBram Moolenaar	    elseif (loops == 1)
200b544f3c8SBram Moolenaar		let p = 1
201b544f3c8SBram Moolenaar		while p	    " dummy loop
202b544f3c8SBram Moolenaar		    Xpath 'f' . loops
203b544f3c8SBram Moolenaar		    let p = 0
204b544f3c8SBram Moolenaar		    break	    " break dummy loop
205b544f3c8SBram Moolenaar		    Xpath 'g' . loops
206b544f3c8SBram Moolenaar		endwhile
207b544f3c8SBram Moolenaar		Xpath 'h' . loops
208b544f3c8SBram Moolenaar		unlet p
209b544f3c8SBram Moolenaar		break	    " break main loop
210b544f3c8SBram Moolenaar		Xpath 'i' . loops
211b544f3c8SBram Moolenaar	    endif
212b544f3c8SBram Moolenaar	    if (loops > 0)
213b544f3c8SBram Moolenaar		Xpath 'j' . loops
214b544f3c8SBram Moolenaar	    endif
215b544f3c8SBram Moolenaar	    while loops == 3    " dummy loop
216b544f3c8SBram Moolenaar		let loops = loops - 1
217b544f3c8SBram Moolenaar	    endwhile	    " end dummy loop
218b544f3c8SBram Moolenaar	endwhile		    " end main loop
219b544f3c8SBram Moolenaar	Xpath 'k'
220b544f3c8SBram Moolenaar    else
221b544f3c8SBram Moolenaar	Xpath 'l'
222b544f3c8SBram Moolenaar    endif
223b544f3c8SBram Moolenaar    Xpath 'm'
224b544f3c8SBram Moolenaar    if exists("break_err")
225b544f3c8SBram Moolenaar	Xpath 'm'
226b544f3c8SBram Moolenaar	unlet break_err
227b544f3c8SBram Moolenaar    endif
228b544f3c8SBram Moolenaar
229b544f3c8SBram Moolenaar    unlet loops
230b544f3c8SBram Moolenaar
231b544f3c8SBram Moolenaar    call assert_equal('ab3j3b2c2b1f1h1km', g:Xpath)
232b544f3c8SBram Moolenaarendfunc
233b544f3c8SBram Moolenaar
234b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
235b544f3c8SBram Moolenaar" Test 4:   :return							    {{{1
236b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
237b544f3c8SBram Moolenaar
238b544f3c8SBram Moolenaarfunction! T4_F()
239b544f3c8SBram Moolenaar    if 1
240b544f3c8SBram Moolenaar	Xpath 'a'
241b544f3c8SBram Moolenaar	let loops = 3
242b544f3c8SBram Moolenaar	while loops > 0				"    3:  2:     1:
243b544f3c8SBram Moolenaar	    Xpath 'b' . loops
244b544f3c8SBram Moolenaar	    if (loops == 2)
245b544f3c8SBram Moolenaar		Xpath 'c' . loops
246b544f3c8SBram Moolenaar		return
247b544f3c8SBram Moolenaar		Xpath 'd' . loops
248b544f3c8SBram Moolenaar	    endif
249b544f3c8SBram Moolenaar	    Xpath 'e' . loops
250b544f3c8SBram Moolenaar	    let loops = loops - 1
251b544f3c8SBram Moolenaar	endwhile
252b544f3c8SBram Moolenaar	Xpath 'f'
253b544f3c8SBram Moolenaar    else
254b544f3c8SBram Moolenaar	Xpath 'g'
255b544f3c8SBram Moolenaar    endif
256b544f3c8SBram Moolenaarendfunction
257b544f3c8SBram Moolenaar
258b544f3c8SBram Moolenaarfunction Test_return()
259b544f3c8SBram Moolenaar    XpathINIT
260b544f3c8SBram Moolenaar    call T4_F()
261b544f3c8SBram Moolenaar    Xpath '4'
262b544f3c8SBram Moolenaar
263b544f3c8SBram Moolenaar    call assert_equal('ab3e3b2c24', g:Xpath)
264b544f3c8SBram Moolenaarendfunction
265b544f3c8SBram Moolenaar
266b544f3c8SBram Moolenaar
267b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
268b544f3c8SBram Moolenaar" Test 5:   :finish							    {{{1
269b544f3c8SBram Moolenaar"
270b544f3c8SBram Moolenaar"	    This test executes the body of the function T4_F from the previous
271b544f3c8SBram Moolenaar"	    test as a script file (:return replaced by :finish).
272b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
273b544f3c8SBram Moolenaar
274b544f3c8SBram Moolenaarfunction Test_finish()
275b544f3c8SBram Moolenaar    XpathINIT
276b544f3c8SBram Moolenaar    ExecAsScript T4_F
277b544f3c8SBram Moolenaar    Xpath '5'
278b544f3c8SBram Moolenaar    call DeleteTheScript()
279b544f3c8SBram Moolenaar
280b544f3c8SBram Moolenaar    call assert_equal('ab3e3b2c25', g:Xpath)
281b544f3c8SBram Moolenaarendfunction
282b544f3c8SBram Moolenaar
283b544f3c8SBram Moolenaar
284b544f3c8SBram Moolenaar
285b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
286b544f3c8SBram Moolenaar" Test 6:   Defining functions in :while loops				    {{{1
287b544f3c8SBram Moolenaar"
288b544f3c8SBram Moolenaar"	     Functions can be defined inside other functions.  An inner function
289b544f3c8SBram Moolenaar"	     gets defined when the outer function is executed.  Functions may
290b544f3c8SBram Moolenaar"	     also be defined inside while loops.  Expressions in braces for
291b544f3c8SBram Moolenaar"	     defining the function name are allowed.
292b544f3c8SBram Moolenaar"
293b544f3c8SBram Moolenaar"	     The functions are defined when sourcing the script, only the
294b544f3c8SBram Moolenaar"	     resulting path is checked in the test function.
295b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
296b544f3c8SBram Moolenaar
297b544f3c8SBram MoolenaarXpathINIT
298b544f3c8SBram Moolenaar
299b544f3c8SBram Moolenaar" The command CALL collects the argument of all its invocations in "calls"
300b544f3c8SBram Moolenaar" when used from a function (that is, when the global variable "calls" needs
301b544f3c8SBram Moolenaar" the "g:" prefix).  This is to check that the function code is skipped when
302b544f3c8SBram Moolenaar" the function is defined.  For inner functions, do so only if the outer
303b544f3c8SBram Moolenaar" function is not being executed.
304b544f3c8SBram Moolenaar"
305b544f3c8SBram Moolenaarlet calls = ""
306b544f3c8SBram Moolenaarcom! -nargs=1 CALL
307b544f3c8SBram Moolenaar	    \ if !exists("calls") && !exists("outer") |
308b544f3c8SBram Moolenaar	    \ let g:calls = g:calls . <args> |
309b544f3c8SBram Moolenaar	    \ endif
310b544f3c8SBram Moolenaar
311b544f3c8SBram Moolenaarlet i = 0
312b544f3c8SBram Moolenaarwhile i < 3
313b544f3c8SBram Moolenaar    let i = i + 1
314b544f3c8SBram Moolenaar    if i == 1
315b544f3c8SBram Moolenaar	Xpath 'a'
316b544f3c8SBram Moolenaar	function! F1(arg)
317b544f3c8SBram Moolenaar	    CALL a:arg
318b544f3c8SBram Moolenaar	    let outer = 1
319b544f3c8SBram Moolenaar
320b544f3c8SBram Moolenaar	    let j = 0
321b544f3c8SBram Moolenaar	    while j < 1
322b544f3c8SBram Moolenaar		Xpath 'b'
323b544f3c8SBram Moolenaar		let j = j + 1
324b544f3c8SBram Moolenaar		function! G1(arg)
325b544f3c8SBram Moolenaar		    CALL a:arg
326b544f3c8SBram Moolenaar		endfunction
327b544f3c8SBram Moolenaar		Xpath 'c'
328b544f3c8SBram Moolenaar	    endwhile
329b544f3c8SBram Moolenaar	endfunction
330b544f3c8SBram Moolenaar	Xpath 'd'
331b544f3c8SBram Moolenaar
332b544f3c8SBram Moolenaar	continue
333b544f3c8SBram Moolenaar    endif
334b544f3c8SBram Moolenaar
335b544f3c8SBram Moolenaar    Xpath 'e' . i
336b544f3c8SBram Moolenaar    function! F{i}(i, arg)
337b544f3c8SBram Moolenaar	CALL a:arg
338b544f3c8SBram Moolenaar	let outer = 1
339b544f3c8SBram Moolenaar
340b544f3c8SBram Moolenaar	if a:i == 3
341b544f3c8SBram Moolenaar	    Xpath 'f'
342b544f3c8SBram Moolenaar	endif
343b544f3c8SBram Moolenaar	let k = 0
344b544f3c8SBram Moolenaar	while k < 3
345b544f3c8SBram Moolenaar	    Xpath 'g' . k
346b544f3c8SBram Moolenaar	    let k = k + 1
347b544f3c8SBram Moolenaar	    function! G{a:i}{k}(arg)
348b544f3c8SBram Moolenaar		CALL a:arg
349b544f3c8SBram Moolenaar	    endfunction
350b544f3c8SBram Moolenaar	    Xpath 'h' . k
351b544f3c8SBram Moolenaar	endwhile
352b544f3c8SBram Moolenaar    endfunction
353b544f3c8SBram Moolenaar    Xpath 'i'
354b544f3c8SBram Moolenaar
355b544f3c8SBram Moolenaarendwhile
356b544f3c8SBram Moolenaar
357b544f3c8SBram Moolenaarif exists("*G1")
358b544f3c8SBram Moolenaar    Xpath 'j'
359b544f3c8SBram Moolenaarendif
360b544f3c8SBram Moolenaarif exists("*F1")
361b544f3c8SBram Moolenaar    call F1("F1")
362b544f3c8SBram Moolenaar    if exists("*G1")
363b544f3c8SBram Moolenaar       call G1("G1")
364b544f3c8SBram Moolenaar    endif
365b544f3c8SBram Moolenaarendif
366b544f3c8SBram Moolenaar
367b544f3c8SBram Moolenaarif exists("G21") || exists("G22") || exists("G23")
368b544f3c8SBram Moolenaar    Xpath 'k'
369b544f3c8SBram Moolenaarendif
370b544f3c8SBram Moolenaarif exists("*F2")
371b544f3c8SBram Moolenaar    call F2(2, "F2")
372b544f3c8SBram Moolenaar    if exists("*G21")
373b544f3c8SBram Moolenaar       call G21("G21")
374b544f3c8SBram Moolenaar    endif
375b544f3c8SBram Moolenaar    if exists("*G22")
376b544f3c8SBram Moolenaar       call G22("G22")
377b544f3c8SBram Moolenaar    endif
378b544f3c8SBram Moolenaar    if exists("*G23")
379b544f3c8SBram Moolenaar       call G23("G23")
380b544f3c8SBram Moolenaar    endif
381b544f3c8SBram Moolenaarendif
382b544f3c8SBram Moolenaar
383b544f3c8SBram Moolenaarif exists("G31") || exists("G32") || exists("G33")
384b544f3c8SBram Moolenaar    Xpath 'l'
385b544f3c8SBram Moolenaarendif
386b544f3c8SBram Moolenaarif exists("*F3")
387b544f3c8SBram Moolenaar    call F3(3, "F3")
388b544f3c8SBram Moolenaar    if exists("*G31")
389b544f3c8SBram Moolenaar       call G31("G31")
390b544f3c8SBram Moolenaar    endif
391b544f3c8SBram Moolenaar    if exists("*G32")
392b544f3c8SBram Moolenaar       call G32("G32")
393b544f3c8SBram Moolenaar    endif
394b544f3c8SBram Moolenaar    if exists("*G33")
395b544f3c8SBram Moolenaar       call G33("G33")
396b544f3c8SBram Moolenaar    endif
397b544f3c8SBram Moolenaarendif
398b544f3c8SBram Moolenaar
399b544f3c8SBram MoolenaarXpath 'm'
400b544f3c8SBram Moolenaar
401b544f3c8SBram Moolenaarlet g:test6_result = g:Xpath
402b544f3c8SBram Moolenaarlet g:test6_calls = calls
403b544f3c8SBram Moolenaar
404b544f3c8SBram Moolenaarunlet calls
405b544f3c8SBram Moolenaardelfunction F1
406b544f3c8SBram Moolenaardelfunction G1
407b544f3c8SBram Moolenaardelfunction F2
408b544f3c8SBram Moolenaardelfunction G21
409b544f3c8SBram Moolenaardelfunction G22
410b544f3c8SBram Moolenaardelfunction G23
411b544f3c8SBram Moolenaardelfunction G31
412b544f3c8SBram Moolenaardelfunction G32
413b544f3c8SBram Moolenaardelfunction G33
414b544f3c8SBram Moolenaar
415b544f3c8SBram Moolenaarfunction Test_defining_functions()
416b544f3c8SBram Moolenaar    call assert_equal('ade2ie3ibcg0h1g1h2g2h3fg0h1g1h2g2h3m', g:test6_result)
417b544f3c8SBram Moolenaar    call assert_equal('F1G1F2G21G22G23F3G31G32G33', g:test6_calls)
418b544f3c8SBram Moolenaarendfunc
419b544f3c8SBram Moolenaar
420b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
421b544f3c8SBram Moolenaar" Test 7:   Continuing on errors outside functions			    {{{1
422b544f3c8SBram Moolenaar"
423b544f3c8SBram Moolenaar"	    On an error outside a function, the script processing continues
424b544f3c8SBram Moolenaar"	    at the line following the outermost :endif or :endwhile.  When not
425b544f3c8SBram Moolenaar"	    inside an :if or :while, the script processing continues at the next
426b544f3c8SBram Moolenaar"	    line.
427b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
428b544f3c8SBram Moolenaar
429b544f3c8SBram MoolenaarXpathINIT
430b544f3c8SBram Moolenaar
431b544f3c8SBram Moolenaarif 1
432b544f3c8SBram Moolenaar    Xpath 'a'
433b544f3c8SBram Moolenaar    while 1
434b544f3c8SBram Moolenaar	Xpath 'b'
435b544f3c8SBram Moolenaar	asdf
436b544f3c8SBram Moolenaar	Xpath 'c'
437b544f3c8SBram Moolenaar	break
438b544f3c8SBram Moolenaar    endwhile | Xpath 'd'
439b544f3c8SBram Moolenaar    Xpath 'e'
440b544f3c8SBram Moolenaarendif | Xpath 'f'
441b544f3c8SBram MoolenaarXpath 'g'
442b544f3c8SBram Moolenaar
443b544f3c8SBram Moolenaarwhile 1
444b544f3c8SBram Moolenaar    Xpath 'h'
445b544f3c8SBram Moolenaar    if 1
446b544f3c8SBram Moolenaar	Xpath 'i'
447b544f3c8SBram Moolenaar	asdf
448b544f3c8SBram Moolenaar	Xpath 'j'
449b544f3c8SBram Moolenaar    endif | Xpath 'k'
450b544f3c8SBram Moolenaar    Xpath 'l'
451b544f3c8SBram Moolenaar    break
452b544f3c8SBram Moolenaarendwhile | Xpath 'm'
453b544f3c8SBram MoolenaarXpath 'n'
454b544f3c8SBram Moolenaar
455b544f3c8SBram Moolenaarasdf
456b544f3c8SBram MoolenaarXpath 'o'
457b544f3c8SBram Moolenaar
458b544f3c8SBram Moolenaarasdf | Xpath 'p'
459b544f3c8SBram MoolenaarXpath 'q'
460b544f3c8SBram Moolenaar
461b544f3c8SBram Moolenaarlet g:test7_result = g:Xpath
462b544f3c8SBram Moolenaar
463b544f3c8SBram Moolenaarfunc Test_error_in_script()
464b544f3c8SBram Moolenaar    call assert_equal('abghinoq', g:test7_result)
465b544f3c8SBram Moolenaarendfunc
466b544f3c8SBram Moolenaar
467b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
468b544f3c8SBram Moolenaar" Test 8:   Aborting and continuing on errors inside functions		    {{{1
469b544f3c8SBram Moolenaar"
470b544f3c8SBram Moolenaar"	    On an error inside a function without the "abort" attribute, the
471b544f3c8SBram Moolenaar"	    script processing continues at the next line (unless the error was
472b544f3c8SBram Moolenaar"	    in a :return command).  On an error inside a function with the
473b544f3c8SBram Moolenaar"	    "abort" attribute, the function is aborted and the script processing
474b544f3c8SBram Moolenaar"	    continues after the function call; the value -1 is returned then.
475b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
476b544f3c8SBram Moolenaar
477b544f3c8SBram MoolenaarXpathINIT
478b544f3c8SBram Moolenaar
479b544f3c8SBram Moolenaarfunction! T8_F()
480b544f3c8SBram Moolenaar    if 1
481b544f3c8SBram Moolenaar	Xpath 'a'
482b544f3c8SBram Moolenaar	while 1
483b544f3c8SBram Moolenaar	    Xpath 'b'
484b544f3c8SBram Moolenaar	    asdf
485b544f3c8SBram Moolenaar	    Xpath 'c'
486b544f3c8SBram Moolenaar	    asdf | Xpath 'd'
487b544f3c8SBram Moolenaar	    Xpath 'e'
488b544f3c8SBram Moolenaar	    break
489b544f3c8SBram Moolenaar	endwhile
490b544f3c8SBram Moolenaar	Xpath 'f'
491b544f3c8SBram Moolenaar    endif | Xpath 'g'
492b544f3c8SBram Moolenaar    Xpath 'h'
493b544f3c8SBram Moolenaar
494b544f3c8SBram Moolenaar    while 1
495b544f3c8SBram Moolenaar	Xpath 'i'
496b544f3c8SBram Moolenaar	if 1
497b544f3c8SBram Moolenaar	    Xpath 'j'
498b544f3c8SBram Moolenaar	    asdf
499b544f3c8SBram Moolenaar	    Xpath 'k'
500b544f3c8SBram Moolenaar	    asdf | Xpath 'l'
501b544f3c8SBram Moolenaar	    Xpath 'm'
502b544f3c8SBram Moolenaar	endif
503b544f3c8SBram Moolenaar	Xpath 'n'
504b544f3c8SBram Moolenaar	break
505b544f3c8SBram Moolenaar    endwhile | Xpath 'o'
506b544f3c8SBram Moolenaar    Xpath 'p'
507b544f3c8SBram Moolenaar
508b544f3c8SBram Moolenaar    return novar		" returns (default return value 0)
509b544f3c8SBram Moolenaar    Xpath 'q'
510b544f3c8SBram Moolenaar    return 1			" not reached
511b544f3c8SBram Moolenaarendfunction
512b544f3c8SBram Moolenaar
513b544f3c8SBram Moolenaarfunction! T8_G() abort
514b544f3c8SBram Moolenaar    if 1
515b544f3c8SBram Moolenaar	Xpath 'r'
516b544f3c8SBram Moolenaar	while 1
517b544f3c8SBram Moolenaar	    Xpath 's'
518b544f3c8SBram Moolenaar	    asdf		" returns -1
519b544f3c8SBram Moolenaar	    Xpath 't'
520b544f3c8SBram Moolenaar	    break
521b544f3c8SBram Moolenaar	endwhile
522b544f3c8SBram Moolenaar	Xpath 'v'
523b544f3c8SBram Moolenaar    endif | Xpath 'w'
524b544f3c8SBram Moolenaar    Xpath 'x'
525b544f3c8SBram Moolenaar
526b544f3c8SBram Moolenaar    return -4			" not reached
527b544f3c8SBram Moolenaarendfunction
528b544f3c8SBram Moolenaar
529b544f3c8SBram Moolenaarfunction! T8_H() abort
530b544f3c8SBram Moolenaar    while 1
531b544f3c8SBram Moolenaar	Xpath 'A'
532b544f3c8SBram Moolenaar	if 1
533b544f3c8SBram Moolenaar	    Xpath 'B'
534b544f3c8SBram Moolenaar	    asdf		" returns -1
535b544f3c8SBram Moolenaar	    Xpath 'C'
536b544f3c8SBram Moolenaar	endif
537b544f3c8SBram Moolenaar	Xpath 'D'
538b544f3c8SBram Moolenaar	break
539b544f3c8SBram Moolenaar    endwhile | Xpath 'E'
540b544f3c8SBram Moolenaar    Xpath 'F'
541b544f3c8SBram Moolenaar
542b544f3c8SBram Moolenaar    return -4			" not reached
543b544f3c8SBram Moolenaarendfunction
544b544f3c8SBram Moolenaar
545b544f3c8SBram Moolenaar" Aborted functions (T8_G and T8_H) return -1.
546b544f3c8SBram Moolenaarlet g:test8_sum = (T8_F() + 1) - 4 * T8_G() - 8 * T8_H()
547b544f3c8SBram MoolenaarXpath 'X'
548b544f3c8SBram Moolenaarlet g:test8_result = g:Xpath
549b544f3c8SBram Moolenaar
550b544f3c8SBram Moolenaarfunc Test_error_in_function()
551b544f3c8SBram Moolenaar    call assert_equal(13, g:test8_sum)
552b544f3c8SBram Moolenaar    call assert_equal('abcefghijkmnoprsABX', g:test8_result)
553b544f3c8SBram Moolenaar
554b544f3c8SBram Moolenaar    delfunction T8_F
555b544f3c8SBram Moolenaar    delfunction T8_G
556b544f3c8SBram Moolenaar    delfunction T8_H
557b544f3c8SBram Moolenaarendfunc
558b544f3c8SBram Moolenaar
559b544f3c8SBram Moolenaar
560b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
561b544f3c8SBram Moolenaar" Test 9:   Continuing after aborted functions				    {{{1
562b544f3c8SBram Moolenaar"
563b544f3c8SBram Moolenaar"	    When a function with the "abort" attribute is aborted due to an
564b544f3c8SBram Moolenaar"	    error, the next function back in the call hierarchy without an
565b544f3c8SBram Moolenaar"	    "abort" attribute continues; the value -1 is returned then.
566b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
567b544f3c8SBram Moolenaar
568b544f3c8SBram MoolenaarXpathINIT
569b544f3c8SBram Moolenaar
570b544f3c8SBram Moolenaarfunction! F() abort
571b544f3c8SBram Moolenaar    Xpath 'a'
572b544f3c8SBram Moolenaar    let result = G()	" not aborted
573b544f3c8SBram Moolenaar    Xpath 'b'
574b544f3c8SBram Moolenaar    if result != 2
575b544f3c8SBram Moolenaar	Xpath 'c'
576b544f3c8SBram Moolenaar    endif
577b544f3c8SBram Moolenaar    return 1
578b544f3c8SBram Moolenaarendfunction
579b544f3c8SBram Moolenaar
580b544f3c8SBram Moolenaarfunction! G()		" no abort attribute
581b544f3c8SBram Moolenaar    Xpath 'd'
582b544f3c8SBram Moolenaar    if H() != -1	" aborted
583b544f3c8SBram Moolenaar	Xpath 'e'
584b544f3c8SBram Moolenaar    endif
585b544f3c8SBram Moolenaar    Xpath 'f'
586b544f3c8SBram Moolenaar    return 2
587b544f3c8SBram Moolenaarendfunction
588b544f3c8SBram Moolenaar
589b544f3c8SBram Moolenaarfunction! H() abort
590b544f3c8SBram Moolenaar    Xpath 'g'
591b544f3c8SBram Moolenaar    call I()		" aborted
592b544f3c8SBram Moolenaar    Xpath 'h'
593b544f3c8SBram Moolenaar    return 4
594b544f3c8SBram Moolenaarendfunction
595b544f3c8SBram Moolenaar
596b544f3c8SBram Moolenaarfunction! I() abort
597b544f3c8SBram Moolenaar    Xpath 'i'
598b544f3c8SBram Moolenaar    asdf		" error
599b544f3c8SBram Moolenaar    Xpath 'j'
600b544f3c8SBram Moolenaar    return 8
601b544f3c8SBram Moolenaarendfunction
602b544f3c8SBram Moolenaar
603b544f3c8SBram Moolenaarif F() != 1
604b544f3c8SBram Moolenaar    Xpath 'k'
605b544f3c8SBram Moolenaarendif
606b544f3c8SBram Moolenaar
607b544f3c8SBram Moolenaarlet g:test9_result = g:Xpath
608b544f3c8SBram Moolenaar
609b544f3c8SBram Moolenaardelfunction F
610b544f3c8SBram Moolenaardelfunction G
611b544f3c8SBram Moolenaardelfunction H
612b544f3c8SBram Moolenaardelfunction I
613b544f3c8SBram Moolenaar
614b544f3c8SBram Moolenaarfunc Test_func_abort()
615b544f3c8SBram Moolenaar    call assert_equal('adgifb', g:test9_result)
616b544f3c8SBram Moolenaarendfunc
617b544f3c8SBram Moolenaar
618b544f3c8SBram Moolenaar
619b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
620b544f3c8SBram Moolenaar" Test 10:  :if, :elseif, :while argument parsing			    {{{1
621b544f3c8SBram Moolenaar"
622b544f3c8SBram Moolenaar"	    A '"' or '|' in an argument expression must not be mixed up with
623b544f3c8SBram Moolenaar"	    a comment or a next command after a bar.  Parsing errors should
624b544f3c8SBram Moolenaar"	    be recognized.
625b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
626b544f3c8SBram Moolenaar
627b544f3c8SBram MoolenaarXpathINIT
628b544f3c8SBram Moolenaar
629b544f3c8SBram Moolenaarfunction! MSG(enr, emsg)
630b544f3c8SBram Moolenaar    let english = v:lang == "C" || v:lang =~ '^[Ee]n'
631b544f3c8SBram Moolenaar    if a:enr == ""
632b544f3c8SBram Moolenaar	Xout "TODO: Add message number for:" a:emsg
633b544f3c8SBram Moolenaar	let v:errmsg = ":" . v:errmsg
634b544f3c8SBram Moolenaar    endif
635b544f3c8SBram Moolenaar    let match = 1
636b544f3c8SBram Moolenaar    if v:errmsg !~ '^'.a:enr.':' || (english && v:errmsg !~ a:emsg)
637b544f3c8SBram Moolenaar	let match = 0
638b544f3c8SBram Moolenaar	if v:errmsg == ""
639b544f3c8SBram Moolenaar	    Xout "Message missing."
640b544f3c8SBram Moolenaar	else
641a4208966SBram Moolenaar	    let v:errmsg = v:errmsg->escape('"')
642b544f3c8SBram Moolenaar	    Xout "Unexpected message:" v:errmsg
643b544f3c8SBram Moolenaar	endif
644b544f3c8SBram Moolenaar    endif
645b544f3c8SBram Moolenaar    return match
6461e115360SBram Moolenaarendfunc
647b544f3c8SBram Moolenaar
648b544f3c8SBram Moolenaarif 1 || strlen("\"") | Xpath 'a'
649b544f3c8SBram Moolenaar    Xpath 'b'
650b544f3c8SBram Moolenaarendif
651b544f3c8SBram MoolenaarXpath 'c'
652b544f3c8SBram Moolenaar
653b544f3c8SBram Moolenaarif 0
654b544f3c8SBram Moolenaarelseif 1 || strlen("\"") | Xpath 'd'
655b544f3c8SBram Moolenaar    Xpath 'e'
656b544f3c8SBram Moolenaarendif
657b544f3c8SBram MoolenaarXpath 'f'
658b544f3c8SBram Moolenaar
659b544f3c8SBram Moolenaarwhile 1 || strlen("\"") | Xpath 'g'
660b544f3c8SBram Moolenaar    Xpath 'h'
661b544f3c8SBram Moolenaar    break
662b544f3c8SBram Moolenaarendwhile
663b544f3c8SBram MoolenaarXpath 'i'
664b544f3c8SBram Moolenaar
665b544f3c8SBram Moolenaarlet v:errmsg = ""
666b544f3c8SBram Moolenaarif 1 ||| strlen("\"") | Xpath 'j'
667b544f3c8SBram Moolenaar    Xpath 'k'
668b544f3c8SBram Moolenaarendif
669b544f3c8SBram MoolenaarXpath 'l'
670b544f3c8SBram Moolenaarif !MSG('E15', "Invalid expression")
671b544f3c8SBram Moolenaar    Xpath 'm'
672b544f3c8SBram Moolenaarendif
673b544f3c8SBram Moolenaar
674b544f3c8SBram Moolenaarlet v:errmsg = ""
675b544f3c8SBram Moolenaarif 0
676b544f3c8SBram Moolenaarelseif 1 ||| strlen("\"") | Xpath 'n'
677b544f3c8SBram Moolenaar    Xpath 'o'
678b544f3c8SBram Moolenaarendif
679b544f3c8SBram MoolenaarXpath 'p'
680b544f3c8SBram Moolenaarif !MSG('E15', "Invalid expression")
681b544f3c8SBram Moolenaar    Xpath 'q'
682b544f3c8SBram Moolenaarendif
683b544f3c8SBram Moolenaar
684b544f3c8SBram Moolenaarlet v:errmsg = ""
685b544f3c8SBram Moolenaarwhile 1 ||| strlen("\"") | Xpath 'r'
686b544f3c8SBram Moolenaar    Xpath 's'
687b544f3c8SBram Moolenaar    break
688b544f3c8SBram Moolenaarendwhile
689b544f3c8SBram MoolenaarXpath 't'
690b544f3c8SBram Moolenaarif !MSG('E15', "Invalid expression")
691b544f3c8SBram Moolenaar    Xpath 'u'
692b544f3c8SBram Moolenaarendif
693b544f3c8SBram Moolenaar
694b544f3c8SBram Moolenaarlet g:test10_result = g:Xpath
695b544f3c8SBram Moolenaardelfunction MSG
696b544f3c8SBram Moolenaar
697b544f3c8SBram Moolenaarfunc Test_expr_parsing()
698b544f3c8SBram Moolenaar    call assert_equal('abcdefghilpt', g:test10_result)
699b544f3c8SBram Moolenaarendfunc
700b544f3c8SBram Moolenaar
701b544f3c8SBram Moolenaar
702b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
703b544f3c8SBram Moolenaar" Test 11:  :if, :elseif, :while argument evaluation after abort	    {{{1
704b544f3c8SBram Moolenaar"
705b544f3c8SBram Moolenaar"	    When code is skipped over due to an error, the boolean argument to
706b544f3c8SBram Moolenaar"	    an :if, :elseif, or :while must not be evaluated.
707b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
708b544f3c8SBram Moolenaar
709b544f3c8SBram MoolenaarXpathINIT
710b544f3c8SBram Moolenaar
711b544f3c8SBram Moolenaarlet calls = 0
712b544f3c8SBram Moolenaar
713b544f3c8SBram Moolenaarfunction! P(num)
714b544f3c8SBram Moolenaar    let g:calls = g:calls + a:num   " side effect on call
715b544f3c8SBram Moolenaar    return 0
716b544f3c8SBram Moolenaarendfunction
717b544f3c8SBram Moolenaar
718b544f3c8SBram Moolenaarif 1
719b544f3c8SBram Moolenaar    Xpath 'a'
720b544f3c8SBram Moolenaar    asdf		" error
721b544f3c8SBram Moolenaar    Xpath 'b'
722b544f3c8SBram Moolenaar    if P(1)		" should not be called
723b544f3c8SBram Moolenaar	Xpath 'c'
724b544f3c8SBram Moolenaar    elseif !P(2)	" should not be called
725b544f3c8SBram Moolenaar	Xpath 'd'
726b544f3c8SBram Moolenaar    else
727b544f3c8SBram Moolenaar	Xpath 'e'
728b544f3c8SBram Moolenaar    endif
729b544f3c8SBram Moolenaar    Xpath 'f'
730b544f3c8SBram Moolenaar    while P(4)		" should not be called
731b544f3c8SBram Moolenaar	Xpath 'g'
732b544f3c8SBram Moolenaar    endwhile
733b544f3c8SBram Moolenaar    Xpath 'h'
734b544f3c8SBram Moolenaarendif
735b544f3c8SBram MoolenaarXpath 'x'
736b544f3c8SBram Moolenaar
737b544f3c8SBram Moolenaarlet g:test11_calls = calls
738b544f3c8SBram Moolenaarlet g:test11_result = g:Xpath
739b544f3c8SBram Moolenaar
740b544f3c8SBram Moolenaarunlet calls
741b544f3c8SBram Moolenaardelfunction P
742b544f3c8SBram Moolenaar
743b544f3c8SBram Moolenaarfunc Test_arg_abort()
744b544f3c8SBram Moolenaar    call assert_equal(0, g:test11_calls)
745b544f3c8SBram Moolenaar    call assert_equal('ax', g:test11_result)
746b544f3c8SBram Moolenaarendfunc
747b544f3c8SBram Moolenaar
748b544f3c8SBram Moolenaar
749b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
750b544f3c8SBram Moolenaar" Test 12:  Expressions in braces in skipped code			    {{{1
751b544f3c8SBram Moolenaar"
752b544f3c8SBram Moolenaar"	    In code skipped over due to an error or inactive conditional,
753b544f3c8SBram Moolenaar"	    an expression in braces as part of a variable or function name
754b544f3c8SBram Moolenaar"	    should not be evaluated.
755b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
756b544f3c8SBram Moolenaar
757b544f3c8SBram MoolenaarXpathINIT
758b544f3c8SBram Moolenaar
759b544f3c8SBram Moolenaarfunction! NULL()
760b544f3c8SBram Moolenaar    Xpath 'a'
761b544f3c8SBram Moolenaar    return 0
762b544f3c8SBram Moolenaarendfunction
763b544f3c8SBram Moolenaar
764b544f3c8SBram Moolenaarfunction! ZERO()
765b544f3c8SBram Moolenaar    Xpath 'b'
766b544f3c8SBram Moolenaar    return 0
767b544f3c8SBram Moolenaarendfunction
768b544f3c8SBram Moolenaar
769b544f3c8SBram Moolenaarfunction! F0()
770b544f3c8SBram Moolenaar    Xpath 'c'
771b544f3c8SBram Moolenaarendfunction
772b544f3c8SBram Moolenaar
773b544f3c8SBram Moolenaarfunction! F1(arg)
774b544f3c8SBram Moolenaar    Xpath 'e'
775b544f3c8SBram Moolenaarendfunction
776b544f3c8SBram Moolenaar
777b544f3c8SBram Moolenaarlet V0 = 1
778b544f3c8SBram Moolenaar
779b544f3c8SBram MoolenaarXpath 'f'
780b544f3c8SBram Moolenaarecho 0 ? F{NULL() + V{ZERO()}}() : 1
781b544f3c8SBram Moolenaar
782b544f3c8SBram MoolenaarXpath 'g'
783b544f3c8SBram Moolenaarif 0
784b544f3c8SBram Moolenaar    Xpath 'h'
785b544f3c8SBram Moolenaar    call F{NULL() + V{ZERO()}}()
786b544f3c8SBram Moolenaarendif
787b544f3c8SBram Moolenaar
788b544f3c8SBram MoolenaarXpath 'i'
789b544f3c8SBram Moolenaarif 1
790b544f3c8SBram Moolenaar    asdf		" error
791b544f3c8SBram Moolenaar    Xpath 'j'
792b544f3c8SBram Moolenaar    call F1(F{NULL() + V{ZERO()}}())
793b544f3c8SBram Moolenaarendif
794b544f3c8SBram Moolenaar
795b544f3c8SBram MoolenaarXpath 'k'
796b544f3c8SBram Moolenaarif 1
797b544f3c8SBram Moolenaar    asdf		" error
798b544f3c8SBram Moolenaar    Xpath 'l'
799b544f3c8SBram Moolenaar    call F{NULL() + V{ZERO()}}()
800b544f3c8SBram Moolenaarendif
801b544f3c8SBram Moolenaar
802b544f3c8SBram Moolenaarlet g:test12_result = g:Xpath
803b544f3c8SBram Moolenaar
804b544f3c8SBram Moolenaarfunc Test_braces_skipped()
805b544f3c8SBram Moolenaar    call assert_equal('fgik', g:test12_result)
806b544f3c8SBram Moolenaarendfunc
807b544f3c8SBram Moolenaar
808b544f3c8SBram Moolenaar
809b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
810b544f3c8SBram Moolenaar" Test 13:  Failure in argument evaluation for :while			    {{{1
811b544f3c8SBram Moolenaar"
812b544f3c8SBram Moolenaar"	    A failure in the expression evaluation for the condition of a :while
813b544f3c8SBram Moolenaar"	    causes the whole :while loop until the matching :endwhile being
814b544f3c8SBram Moolenaar"	    ignored.  Continuation is at the next following line.
815b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
816b544f3c8SBram Moolenaar
817b544f3c8SBram MoolenaarXpathINIT
818b544f3c8SBram Moolenaar
819b544f3c8SBram MoolenaarXpath 'a'
820b544f3c8SBram Moolenaarwhile asdf
821b544f3c8SBram Moolenaar    Xpath 'b'
822b544f3c8SBram Moolenaar    while 1
823b544f3c8SBram Moolenaar	Xpath 'c'
824b544f3c8SBram Moolenaar	break
825b544f3c8SBram Moolenaar    endwhile
826b544f3c8SBram Moolenaar    Xpath 'd'
827b544f3c8SBram Moolenaar    break
828b544f3c8SBram Moolenaarendwhile
829b544f3c8SBram MoolenaarXpath 'e'
830b544f3c8SBram Moolenaar
831b544f3c8SBram Moolenaarwhile asdf | Xpath 'f' | endwhile | Xpath 'g'
832b544f3c8SBram MoolenaarXpath 'h'
833b544f3c8SBram Moolenaarlet g:test13_result = g:Xpath
834b544f3c8SBram Moolenaar
835b544f3c8SBram Moolenaarfunc Test_while_fail()
836b544f3c8SBram Moolenaar    call assert_equal('aeh', g:test13_result)
837b544f3c8SBram Moolenaarendfunc
838b544f3c8SBram Moolenaar
839b544f3c8SBram Moolenaar
840b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
841b544f3c8SBram Moolenaar" Test 14:  Failure in argument evaluation for :if			    {{{1
842b544f3c8SBram Moolenaar"
843b544f3c8SBram Moolenaar"	    A failure in the expression evaluation for the condition of an :if
844b544f3c8SBram Moolenaar"	    does not cause the corresponding :else or :endif being matched to
845b544f3c8SBram Moolenaar"	    a previous :if/:elseif.  Neither of both branches of the failed :if
846b544f3c8SBram Moolenaar"	    are executed.
847b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
848b544f3c8SBram Moolenaar
849b544f3c8SBram MoolenaarXpathINIT
850b544f3c8SBram Moolenaar
851b544f3c8SBram Moolenaarfunction! F()
852b544f3c8SBram Moolenaar    Xpath 'a'
853b544f3c8SBram Moolenaar    let x = 0
854b544f3c8SBram Moolenaar    if x		" false
855b544f3c8SBram Moolenaar	Xpath 'b'
856b544f3c8SBram Moolenaar    elseif !x		" always true
857b544f3c8SBram Moolenaar	Xpath 'c'
858b544f3c8SBram Moolenaar	let x = 1
859b544f3c8SBram Moolenaar	if g:boolvar	" possibly undefined
860b544f3c8SBram Moolenaar	    Xpath 'd'
861b544f3c8SBram Moolenaar	else
862b544f3c8SBram Moolenaar	    Xpath 'e'
863b544f3c8SBram Moolenaar	endif
864b544f3c8SBram Moolenaar	Xpath 'f'
865b544f3c8SBram Moolenaar    elseif x		" never executed
866b544f3c8SBram Moolenaar	Xpath 'g'
867b544f3c8SBram Moolenaar    endif
868b544f3c8SBram Moolenaar    Xpath 'h'
869b544f3c8SBram Moolenaarendfunction
870b544f3c8SBram Moolenaar
871b544f3c8SBram Moolenaarlet boolvar = 1
872b544f3c8SBram Moolenaarcall F()
873b544f3c8SBram MoolenaarXpath '-'
874b544f3c8SBram Moolenaar
875b544f3c8SBram Moolenaarunlet boolvar
876b544f3c8SBram Moolenaarcall F()
877b544f3c8SBram Moolenaarlet g:test14_result = g:Xpath
878b544f3c8SBram Moolenaar
879b544f3c8SBram Moolenaardelfunction F
880b544f3c8SBram Moolenaar
881b544f3c8SBram Moolenaarfunc Test_if_fail()
882b544f3c8SBram Moolenaar    call assert_equal('acdfh-acfh', g:test14_result)
883b544f3c8SBram Moolenaarendfunc
884b544f3c8SBram Moolenaar
885b544f3c8SBram Moolenaar
886b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
887b544f3c8SBram Moolenaar" Test 15:  Failure in argument evaluation for :if (bar)		    {{{1
888b544f3c8SBram Moolenaar"
889b544f3c8SBram Moolenaar"	    Like previous test, except that the failing :if ... | ... | :endif
890b544f3c8SBram Moolenaar"	    is in a single line.
891b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
892b544f3c8SBram Moolenaar
893b544f3c8SBram MoolenaarXpathINIT
894b544f3c8SBram Moolenaar
895b544f3c8SBram Moolenaarfunction! F()
896b544f3c8SBram Moolenaar    Xpath 'a'
897b544f3c8SBram Moolenaar    let x = 0
898b544f3c8SBram Moolenaar    if x		" false
899b544f3c8SBram Moolenaar	Xpath 'b'
900b544f3c8SBram Moolenaar    elseif !x		" always true
901b544f3c8SBram Moolenaar	Xpath 'c'
902b544f3c8SBram Moolenaar	let x = 1
903b544f3c8SBram Moolenaar	if g:boolvar | Xpath 'd' | else | Xpath 'e' | endif
904b544f3c8SBram Moolenaar	Xpath 'f'
905b544f3c8SBram Moolenaar    elseif x		" never executed
906b544f3c8SBram Moolenaar	Xpath 'g'
907b544f3c8SBram Moolenaar    endif
908b544f3c8SBram Moolenaar    Xpath 'h'
909b544f3c8SBram Moolenaarendfunction
910b544f3c8SBram Moolenaar
911b544f3c8SBram Moolenaarlet boolvar = 1
912b544f3c8SBram Moolenaarcall F()
913b544f3c8SBram MoolenaarXpath '-'
914b544f3c8SBram Moolenaar
915b544f3c8SBram Moolenaarunlet boolvar
916b544f3c8SBram Moolenaarcall F()
917b544f3c8SBram Moolenaarlet g:test15_result = g:Xpath
918b544f3c8SBram Moolenaar
919b544f3c8SBram Moolenaardelfunction F
920b544f3c8SBram Moolenaar
921b544f3c8SBram Moolenaarfunc Test_if_bar_fail()
922b544f3c8SBram Moolenaar    call assert_equal('acdfh-acfh', g:test15_result)
923b544f3c8SBram Moolenaarendfunc
924b544f3c8SBram Moolenaar
925b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
9261f068233SBram Moolenaar" Test 16:  Double :else or :elseif after :else				    {{{1
9271f068233SBram Moolenaar"
9281f068233SBram Moolenaar"	    Multiple :elses or an :elseif after an :else are forbidden.
9291f068233SBram Moolenaar"-------------------------------------------------------------------------------
9301f068233SBram Moolenaar
9311f068233SBram Moolenaarfunc T16_F() abort
9321f068233SBram Moolenaar  if 0
9331f068233SBram Moolenaar    Xpath 'a'
9341f068233SBram Moolenaar  else
9351f068233SBram Moolenaar    Xpath 'b'
9361f068233SBram Moolenaar  else		" aborts function
9371f068233SBram Moolenaar    Xpath 'c'
9381f068233SBram Moolenaar  endif
9391f068233SBram Moolenaar  Xpath 'd'
9401f068233SBram Moolenaarendfunc
9411f068233SBram Moolenaar
9421f068233SBram Moolenaarfunc T16_G() abort
9431f068233SBram Moolenaar  if 0
9441f068233SBram Moolenaar    Xpath 'a'
9451f068233SBram Moolenaar  else
9461f068233SBram Moolenaar    Xpath 'b'
9471f068233SBram Moolenaar  elseif 1		" aborts function
9481f068233SBram Moolenaar    Xpath 'c'
9491f068233SBram Moolenaar  else
9501f068233SBram Moolenaar    Xpath 'd'
9511f068233SBram Moolenaar  endif
9521f068233SBram Moolenaar  Xpath 'e'
9531f068233SBram Moolenaarendfunc
9541f068233SBram Moolenaar
9551f068233SBram Moolenaarfunc T16_H() abort
9561f068233SBram Moolenaar  if 0
9571f068233SBram Moolenaar    Xpath 'a'
9581f068233SBram Moolenaar  elseif 0
9591f068233SBram Moolenaar    Xpath 'b'
9601f068233SBram Moolenaar  else
9611f068233SBram Moolenaar    Xpath 'c'
9621f068233SBram Moolenaar  else		" aborts function
9631f068233SBram Moolenaar    Xpath 'd'
9641f068233SBram Moolenaar  endif
9651f068233SBram Moolenaar  Xpath 'e'
9661f068233SBram Moolenaarendfunc
9671f068233SBram Moolenaar
9681f068233SBram Moolenaarfunc T16_I() abort
9691f068233SBram Moolenaar  if 0
9701f068233SBram Moolenaar    Xpath 'a'
9711f068233SBram Moolenaar  elseif 0
9721f068233SBram Moolenaar    Xpath 'b'
9731f068233SBram Moolenaar  else
9741f068233SBram Moolenaar    Xpath 'c'
9751f068233SBram Moolenaar  elseif 1		" aborts function
9761f068233SBram Moolenaar    Xpath 'd'
9771f068233SBram Moolenaar  else
9781f068233SBram Moolenaar    Xpath 'e'
9791f068233SBram Moolenaar  endif
9801f068233SBram Moolenaar  Xpath 'f'
9811f068233SBram Moolenaarendfunc
9821f068233SBram Moolenaar
9831f068233SBram Moolenaarfunc Test_Multi_Else()
9841f068233SBram Moolenaar  XpathINIT
9851f068233SBram Moolenaar  try
9861f068233SBram Moolenaar    call T16_F()
9871f068233SBram Moolenaar  catch /E583:/
9881f068233SBram Moolenaar    Xpath 'e'
9891f068233SBram Moolenaar  endtry
9901f068233SBram Moolenaar  call assert_equal('be', g:Xpath)
9911f068233SBram Moolenaar
9921f068233SBram Moolenaar  XpathINIT
9931f068233SBram Moolenaar  try
9941f068233SBram Moolenaar    call T16_G()
9951f068233SBram Moolenaar  catch /E584:/
9961f068233SBram Moolenaar    Xpath 'f'
9971f068233SBram Moolenaar  endtry
9981f068233SBram Moolenaar  call assert_equal('bf', g:Xpath)
9991f068233SBram Moolenaar
10001f068233SBram Moolenaar  XpathINIT
10011f068233SBram Moolenaar  try
10021f068233SBram Moolenaar    call T16_H()
10031f068233SBram Moolenaar  catch /E583:/
10041f068233SBram Moolenaar    Xpath 'f'
10051f068233SBram Moolenaar  endtry
10061f068233SBram Moolenaar  call assert_equal('cf', g:Xpath)
10071f068233SBram Moolenaar
10081f068233SBram Moolenaar  XpathINIT
10091f068233SBram Moolenaar  try
10101f068233SBram Moolenaar    call T16_I()
10111f068233SBram Moolenaar  catch /E584:/
10121f068233SBram Moolenaar    Xpath 'g'
10131f068233SBram Moolenaar  endtry
10141f068233SBram Moolenaar  call assert_equal('cg', g:Xpath)
10151f068233SBram Moolenaarendfunc
10161f068233SBram Moolenaar
10171f068233SBram Moolenaar"-------------------------------------------------------------------------------
10181f068233SBram Moolenaar" Test 17:  Nesting of unmatched :if or :endif inside a :while		    {{{1
10191f068233SBram Moolenaar"
10201f068233SBram Moolenaar"	    The :while/:endwhile takes precedence in nesting over an unclosed
10211f068233SBram Moolenaar"	    :if or an unopened :endif.
10221f068233SBram Moolenaar"-------------------------------------------------------------------------------
10231f068233SBram Moolenaar
10241f068233SBram Moolenaar" While loops inside a function are continued on error.
10251f068233SBram Moolenaarfunc T17_F()
10261f068233SBram Moolenaar  let loops = 3
10271f068233SBram Moolenaar  while loops > 0
10281f068233SBram Moolenaar    let loops -= 1
10291f068233SBram Moolenaar    Xpath 'a' . loops
10301f068233SBram Moolenaar    if (loops == 1)
10311f068233SBram Moolenaar      Xpath 'b' . loops
10321f068233SBram Moolenaar      continue
10331f068233SBram Moolenaar    elseif (loops == 0)
10341f068233SBram Moolenaar      Xpath 'c' . loops
10351f068233SBram Moolenaar      break
10361f068233SBram Moolenaar    elseif 1
10371f068233SBram Moolenaar      Xpath 'd' . loops
10381f068233SBram Moolenaar    " endif missing!
10391f068233SBram Moolenaar  endwhile	" :endwhile after :if 1
10401f068233SBram Moolenaar  Xpath 'e'
10411f068233SBram Moolenaarendfunc
10421f068233SBram Moolenaar
10431f068233SBram Moolenaarfunc T17_G()
10441f068233SBram Moolenaar  let loops = 2
10451f068233SBram Moolenaar  while loops > 0
10461f068233SBram Moolenaar    let loops -= 1
10471f068233SBram Moolenaar    Xpath 'a' . loops
10481f068233SBram Moolenaar    if 0
10491f068233SBram Moolenaar      Xpath 'b' . loops
10501f068233SBram Moolenaar    " endif missing
10511f068233SBram Moolenaar  endwhile	" :endwhile after :if 0
10521f068233SBram Moolenaarendfunc
10531f068233SBram Moolenaar
10541f068233SBram Moolenaarfunc T17_H()
10551f068233SBram Moolenaar  let loops = 2
10561f068233SBram Moolenaar  while loops > 0
10571f068233SBram Moolenaar    let loops -= 1
10581f068233SBram Moolenaar    Xpath 'a' . loops
10591f068233SBram Moolenaar    " if missing!
10601f068233SBram Moolenaar    endif	" :endif without :if in while
10611f068233SBram Moolenaar    Xpath 'b' . loops
10621f068233SBram Moolenaar  endwhile
10631f068233SBram Moolenaarendfunc
10641f068233SBram Moolenaar
10651f068233SBram Moolenaar" Error continuation outside a function is at the outermost :endwhile or :endif.
10661f068233SBram MoolenaarXpathINIT
10671f068233SBram Moolenaarlet v:errmsg = ''
10681f068233SBram Moolenaarlet loops = 2
10691f068233SBram Moolenaarwhile loops > 0
10701f068233SBram Moolenaar    let loops -= 1
10711f068233SBram Moolenaar    Xpath 'a' . loops
10721f068233SBram Moolenaar    if 0
10731f068233SBram Moolenaar	Xpath 'b' . loops
10741f068233SBram Moolenaar    " endif missing! Following :endwhile fails.
10751f068233SBram Moolenaarendwhile | Xpath 'c'
10761f068233SBram MoolenaarXpath 'd'
10771f068233SBram Moolenaarcall assert_match('E171:', v:errmsg)
10781f068233SBram Moolenaarcall assert_equal('a1d', g:Xpath)
10791f068233SBram Moolenaar
10801f068233SBram Moolenaarfunc Test_unmatched_if_in_while()
10811f068233SBram Moolenaar  XpathINIT
10821f068233SBram Moolenaar  call assert_fails('call T17_F()', 'E171:')
10831f068233SBram Moolenaar  call assert_equal('a2d2a1b1a0c0e', g:Xpath)
10841f068233SBram Moolenaar
10851f068233SBram Moolenaar  XpathINIT
10861f068233SBram Moolenaar  call assert_fails('call T17_G()', 'E171:')
10871f068233SBram Moolenaar  call assert_equal('a1a0', g:Xpath)
10881f068233SBram Moolenaar
10891f068233SBram Moolenaar  XpathINIT
10901f068233SBram Moolenaar  call assert_fails('call T17_H()', 'E580:')
10911f068233SBram Moolenaar  call assert_equal('a1b1a0b0', g:Xpath)
10921f068233SBram Moolenaarendfunc
10931f068233SBram Moolenaar
10941f068233SBram Moolenaar"-------------------------------------------------------------------------------
10951f068233SBram Moolenaar"-------------------------------------------------------------------------------
10961f068233SBram Moolenaar"-------------------------------------------------------------------------------
10971f068233SBram Moolenaar" Test 87   using (expr) ? funcref : funcref				    {{{1
10981f068233SBram Moolenaar"
10991f068233SBram Moolenaar"	    Vim needs to correctly parse the funcref and even when it does
11001f068233SBram Moolenaar"	    not execute the funcref, it needs to consume the trailing ()
11011f068233SBram Moolenaar"-------------------------------------------------------------------------------
11021f068233SBram Moolenaar
11031f068233SBram Moolenaarfunc Add2(x1, x2)
11041f068233SBram Moolenaar  return a:x1 + a:x2
11051f068233SBram Moolenaarendfu
11061f068233SBram Moolenaar
11071f068233SBram Moolenaarfunc GetStr()
11081f068233SBram Moolenaar  return "abcdefghijklmnopqrstuvwxyp"
11091f068233SBram Moolenaarendfu
11101f068233SBram Moolenaar
11111f068233SBram Moolenaarfunc Test_funcref_with_condexpr()
11121f068233SBram Moolenaar  call assert_equal(5, function('Add2')(2,3))
11131f068233SBram Moolenaar
11141f068233SBram Moolenaar  call assert_equal(3, 1 ? function('Add2')(1,2) : function('Add2')(2,3))
11151f068233SBram Moolenaar  call assert_equal(5, 0 ? function('Add2')(1,2) : function('Add2')(2,3))
11161f068233SBram Moolenaar  " Make sure, GetStr() still works.
11171f068233SBram Moolenaar  call assert_equal('abcdefghijk', GetStr()[0:10])
11181f068233SBram Moolenaarendfunc
11191f068233SBram Moolenaar
1120b544f3c8SBram Moolenaar" Test 90:  Recognizing {} in variable name.			    {{{1
1121b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
1122b544f3c8SBram Moolenaar
1123b544f3c8SBram Moolenaarfunc Test_curlies()
1124b544f3c8SBram Moolenaar    let s:var = 66
1125b544f3c8SBram Moolenaar    let ns = 's'
1126b544f3c8SBram Moolenaar    call assert_equal(66, {ns}:var)
1127b544f3c8SBram Moolenaar
1128b544f3c8SBram Moolenaar    let g:a = {}
1129b544f3c8SBram Moolenaar    let g:b = 't'
1130b544f3c8SBram Moolenaar    let g:a[g:b] = 77
1131b544f3c8SBram Moolenaar    call assert_equal(77, g:a['t'])
1132b544f3c8SBram Moolenaarendfunc
1133b544f3c8SBram Moolenaar
1134b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
1135b544f3c8SBram Moolenaar" Test 91:  using type().					    {{{1
1136b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
1137b544f3c8SBram Moolenaar
1138b544f3c8SBram Moolenaarfunc Test_type()
1139b544f3c8SBram Moolenaar    call assert_equal(0, type(0))
1140b544f3c8SBram Moolenaar    call assert_equal(1, type(""))
1141b544f3c8SBram Moolenaar    call assert_equal(2, type(function("tr")))
1142b544f3c8SBram Moolenaar    call assert_equal(2, type(function("tr", [8])))
1143b544f3c8SBram Moolenaar    call assert_equal(3, type([]))
1144b544f3c8SBram Moolenaar    call assert_equal(4, type({}))
11455feabe00SBram Moolenaar    if has('float')
1146b544f3c8SBram Moolenaar      call assert_equal(5, type(0.0))
11475feabe00SBram Moolenaar    endif
1148b544f3c8SBram Moolenaar    call assert_equal(6, type(v:false))
1149b544f3c8SBram Moolenaar    call assert_equal(6, type(v:true))
1150b544f3c8SBram Moolenaar    call assert_equal(7, type(v:none))
1151b544f3c8SBram Moolenaar    call assert_equal(7, type(v:null))
1152b544f3c8SBram Moolenaar    call assert_equal(8, v:t_job)
1153b544f3c8SBram Moolenaar    call assert_equal(9, v:t_channel)
1154b544f3c8SBram Moolenaar    call assert_equal(v:t_number, type(0))
1155b544f3c8SBram Moolenaar    call assert_equal(v:t_string, type(""))
1156b544f3c8SBram Moolenaar    call assert_equal(v:t_func, type(function("tr")))
1157b544f3c8SBram Moolenaar    call assert_equal(v:t_func, type(function("tr", [8])))
1158b544f3c8SBram Moolenaar    call assert_equal(v:t_list, type([]))
1159b544f3c8SBram Moolenaar    call assert_equal(v:t_dict, type({}))
11605feabe00SBram Moolenaar    if has('float')
1161b544f3c8SBram Moolenaar      call assert_equal(v:t_float, type(0.0))
11625feabe00SBram Moolenaar    endif
1163b544f3c8SBram Moolenaar    call assert_equal(v:t_bool, type(v:false))
1164b544f3c8SBram Moolenaar    call assert_equal(v:t_bool, type(v:true))
1165b544f3c8SBram Moolenaar    call assert_equal(v:t_none, type(v:none))
1166b544f3c8SBram Moolenaar    call assert_equal(v:t_none, type(v:null))
1167b544f3c8SBram Moolenaar
11687c215c58SBram Moolenaar    call assert_fails("call type(test_void())", 'E685:')
11697c215c58SBram Moolenaar    call assert_fails("call type(test_unknown())", 'E685:')
1170b544f3c8SBram Moolenaar
1171b544f3c8SBram Moolenaar    call assert_equal(0, 0 + v:false)
1172b544f3c8SBram Moolenaar    call assert_equal(1, 0 + v:true)
1173b544f3c8SBram Moolenaar    call assert_equal(0, 0 + v:none)
1174b544f3c8SBram Moolenaar    call assert_equal(0, 0 + v:null)
1175b544f3c8SBram Moolenaar
1176b544f3c8SBram Moolenaar    call assert_equal('v:false', '' . v:false)
1177b544f3c8SBram Moolenaar    call assert_equal('v:true', '' . v:true)
1178b544f3c8SBram Moolenaar    call assert_equal('v:none', '' . v:none)
1179b544f3c8SBram Moolenaar    call assert_equal('v:null', '' . v:null)
1180b544f3c8SBram Moolenaar
1181b544f3c8SBram Moolenaar    call assert_true(v:false == 0)
1182b544f3c8SBram Moolenaar    call assert_false(v:false != 0)
1183b544f3c8SBram Moolenaar    call assert_true(v:true == 1)
1184b544f3c8SBram Moolenaar    call assert_false(v:true != 1)
1185b544f3c8SBram Moolenaar    call assert_false(v:true == v:false)
1186b544f3c8SBram Moolenaar    call assert_true(v:true != v:false)
1187b544f3c8SBram Moolenaar
1188b544f3c8SBram Moolenaar    call assert_true(v:null == 0)
1189b544f3c8SBram Moolenaar    call assert_false(v:null != 0)
1190b544f3c8SBram Moolenaar    call assert_true(v:none == 0)
1191b544f3c8SBram Moolenaar    call assert_false(v:none != 0)
1192b544f3c8SBram Moolenaar
1193b544f3c8SBram Moolenaar    call assert_true(v:false is v:false)
1194b544f3c8SBram Moolenaar    call assert_true(v:true is v:true)
1195b544f3c8SBram Moolenaar    call assert_true(v:none is v:none)
1196b544f3c8SBram Moolenaar    call assert_true(v:null is v:null)
1197b544f3c8SBram Moolenaar
1198b544f3c8SBram Moolenaar    call assert_false(v:false isnot v:false)
1199b544f3c8SBram Moolenaar    call assert_false(v:true isnot v:true)
1200b544f3c8SBram Moolenaar    call assert_false(v:none isnot v:none)
1201b544f3c8SBram Moolenaar    call assert_false(v:null isnot v:null)
1202b544f3c8SBram Moolenaar
1203b544f3c8SBram Moolenaar    call assert_false(v:false is 0)
1204b544f3c8SBram Moolenaar    call assert_false(v:true is 1)
1205b544f3c8SBram Moolenaar    call assert_false(v:true is v:false)
1206b544f3c8SBram Moolenaar    call assert_false(v:none is 0)
1207b544f3c8SBram Moolenaar    call assert_false(v:null is 0)
1208b544f3c8SBram Moolenaar    call assert_false(v:null is v:none)
1209b544f3c8SBram Moolenaar
1210b544f3c8SBram Moolenaar    call assert_true(v:false isnot 0)
1211b544f3c8SBram Moolenaar    call assert_true(v:true isnot 1)
1212b544f3c8SBram Moolenaar    call assert_true(v:true isnot v:false)
1213b544f3c8SBram Moolenaar    call assert_true(v:none isnot 0)
1214b544f3c8SBram Moolenaar    call assert_true(v:null isnot 0)
1215b544f3c8SBram Moolenaar    call assert_true(v:null isnot v:none)
1216b544f3c8SBram Moolenaar
1217b544f3c8SBram Moolenaar    call assert_equal(v:false, eval(string(v:false)))
1218b544f3c8SBram Moolenaar    call assert_equal(v:true, eval(string(v:true)))
1219b544f3c8SBram Moolenaar    call assert_equal(v:none, eval(string(v:none)))
1220b544f3c8SBram Moolenaar    call assert_equal(v:null, eval(string(v:null)))
1221b544f3c8SBram Moolenaar
1222b544f3c8SBram Moolenaar    call assert_equal(v:false, copy(v:false))
1223b544f3c8SBram Moolenaar    call assert_equal(v:true, copy(v:true))
1224b544f3c8SBram Moolenaar    call assert_equal(v:none, copy(v:none))
1225b544f3c8SBram Moolenaar    call assert_equal(v:null, copy(v:null))
1226b544f3c8SBram Moolenaar
1227b544f3c8SBram Moolenaar    call assert_equal([v:false], deepcopy([v:false]))
1228b544f3c8SBram Moolenaar    call assert_equal([v:true], deepcopy([v:true]))
1229b544f3c8SBram Moolenaar    call assert_equal([v:none], deepcopy([v:none]))
1230b544f3c8SBram Moolenaar    call assert_equal([v:null], deepcopy([v:null]))
1231b544f3c8SBram Moolenaar
1232b544f3c8SBram Moolenaar    call assert_true(empty(v:false))
1233b544f3c8SBram Moolenaar    call assert_false(empty(v:true))
1234b544f3c8SBram Moolenaar    call assert_true(empty(v:null))
1235b544f3c8SBram Moolenaar    call assert_true(empty(v:none))
1236b544f3c8SBram Moolenaar
1237b544f3c8SBram Moolenaar    func ChangeYourMind()
1238b544f3c8SBram Moolenaar	try
1239b544f3c8SBram Moolenaar	    return v:true
1240b544f3c8SBram Moolenaar	finally
1241b544f3c8SBram Moolenaar	    return 'something else'
1242b544f3c8SBram Moolenaar	endtry
1243b544f3c8SBram Moolenaar    endfunc
1244b544f3c8SBram Moolenaar
1245b544f3c8SBram Moolenaar    call ChangeYourMind()
1246b544f3c8SBram Moolenaarendfunc
1247b544f3c8SBram Moolenaar
1248b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
1249b544f3c8SBram Moolenaar" Test 92:  skipping code					    {{{1
1250b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
1251b544f3c8SBram Moolenaar
1252b544f3c8SBram Moolenaarfunc Test_skip()
1253b544f3c8SBram Moolenaar    let Fn = function('Test_type')
1254b544f3c8SBram Moolenaar    call assert_false(0 && Fn[1])
1255b544f3c8SBram Moolenaar    call assert_false(0 && string(Fn))
1256b544f3c8SBram Moolenaar    call assert_false(0 && len(Fn))
1257b544f3c8SBram Moolenaar    let l = []
1258b544f3c8SBram Moolenaar    call assert_false(0 && l[1])
1259b544f3c8SBram Moolenaar    call assert_false(0 && string(l))
1260b544f3c8SBram Moolenaar    call assert_false(0 && len(l))
1261b544f3c8SBram Moolenaar    let f = 1.0
1262b544f3c8SBram Moolenaar    call assert_false(0 && f[1])
1263b544f3c8SBram Moolenaar    call assert_false(0 && string(f))
1264b544f3c8SBram Moolenaar    call assert_false(0 && len(f))
1265b544f3c8SBram Moolenaar    let sp = v:null
1266b544f3c8SBram Moolenaar    call assert_false(0 && sp[1])
1267b544f3c8SBram Moolenaar    call assert_false(0 && string(sp))
1268b544f3c8SBram Moolenaar    call assert_false(0 && len(sp))
1269b544f3c8SBram Moolenaar
1270b544f3c8SBram Moolenaarendfunc
1271b544f3c8SBram Moolenaar
1272b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
1273b544f3c8SBram Moolenaar" Test 93:  :echo and string()					    {{{1
1274b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
1275b544f3c8SBram Moolenaar
1276b544f3c8SBram Moolenaarfunc Test_echo_and_string()
1277b544f3c8SBram Moolenaar    " String
1278b544f3c8SBram Moolenaar    let a = 'foo bar'
1279b544f3c8SBram Moolenaar    redir => result
1280b544f3c8SBram Moolenaar    echo a
1281b544f3c8SBram Moolenaar    echo string(a)
1282b544f3c8SBram Moolenaar    redir END
1283b544f3c8SBram Moolenaar    let l = split(result, "\n")
1284b544f3c8SBram Moolenaar    call assert_equal(["foo bar",
1285b544f3c8SBram Moolenaar		     \ "'foo bar'"], l)
1286b544f3c8SBram Moolenaar
1287b544f3c8SBram Moolenaar    " Float
1288b544f3c8SBram Moolenaar    if has('float')
1289b544f3c8SBram Moolenaar	let a = -1.2e0
1290b544f3c8SBram Moolenaar	redir => result
1291b544f3c8SBram Moolenaar	echo a
1292b544f3c8SBram Moolenaar	echo string(a)
1293b544f3c8SBram Moolenaar	redir END
1294b544f3c8SBram Moolenaar	let l = split(result, "\n")
1295b544f3c8SBram Moolenaar	call assert_equal(["-1.2",
1296b544f3c8SBram Moolenaar			 \ "-1.2"], l)
1297b544f3c8SBram Moolenaar    endif
1298b544f3c8SBram Moolenaar
1299b544f3c8SBram Moolenaar    " Funcref
1300b544f3c8SBram Moolenaar    redir => result
1301b544f3c8SBram Moolenaar    echo function('string')
1302b544f3c8SBram Moolenaar    echo string(function('string'))
1303b544f3c8SBram Moolenaar    redir END
1304b544f3c8SBram Moolenaar    let l = split(result, "\n")
1305b544f3c8SBram Moolenaar    call assert_equal(["string",
1306b544f3c8SBram Moolenaar		     \ "function('string')"], l)
1307b544f3c8SBram Moolenaar
1308b544f3c8SBram Moolenaar    " Recursive dictionary
1309b544f3c8SBram Moolenaar    let a = {}
1310b544f3c8SBram Moolenaar    let a["a"] = a
1311b544f3c8SBram Moolenaar    redir => result
1312b544f3c8SBram Moolenaar    echo a
1313b544f3c8SBram Moolenaar    echo string(a)
1314b544f3c8SBram Moolenaar    redir END
1315b544f3c8SBram Moolenaar    let l = split(result, "\n")
1316b544f3c8SBram Moolenaar    call assert_equal(["{'a': {...}}",
1317b544f3c8SBram Moolenaar		     \ "{'a': {...}}"], l)
1318b544f3c8SBram Moolenaar
1319b544f3c8SBram Moolenaar    " Recursive list
1320b544f3c8SBram Moolenaar    let a = [0]
1321b544f3c8SBram Moolenaar    let a[0] = a
1322b544f3c8SBram Moolenaar    redir => result
1323b544f3c8SBram Moolenaar    echo a
1324b544f3c8SBram Moolenaar    echo string(a)
1325b544f3c8SBram Moolenaar    redir END
1326b544f3c8SBram Moolenaar    let l = split(result, "\n")
1327b544f3c8SBram Moolenaar    call assert_equal(["[[...]]",
1328b544f3c8SBram Moolenaar		     \ "[[...]]"], l)
1329b544f3c8SBram Moolenaar
1330b544f3c8SBram Moolenaar    " Empty dictionaries in a list
1331b544f3c8SBram Moolenaar    let a = {}
1332b544f3c8SBram Moolenaar    redir => result
1333b544f3c8SBram Moolenaar    echo [a, a, a]
1334b544f3c8SBram Moolenaar    echo string([a, a, a])
1335b544f3c8SBram Moolenaar    redir END
1336b544f3c8SBram Moolenaar    let l = split(result, "\n")
1337b544f3c8SBram Moolenaar    call assert_equal(["[{}, {}, {}]",
1338b544f3c8SBram Moolenaar		     \ "[{}, {}, {}]"], l)
1339b544f3c8SBram Moolenaar
1340b544f3c8SBram Moolenaar    " Empty dictionaries in a dictionary
1341b544f3c8SBram Moolenaar    let a = {}
1342b544f3c8SBram Moolenaar    let b = {"a": a, "b": a}
1343b544f3c8SBram Moolenaar    redir => result
1344b544f3c8SBram Moolenaar    echo b
1345b544f3c8SBram Moolenaar    echo string(b)
1346b544f3c8SBram Moolenaar    redir END
1347b544f3c8SBram Moolenaar    let l = split(result, "\n")
1348b544f3c8SBram Moolenaar    call assert_equal(["{'a': {}, 'b': {}}",
1349b544f3c8SBram Moolenaar		     \ "{'a': {}, 'b': {}}"], l)
1350b544f3c8SBram Moolenaar
1351b544f3c8SBram Moolenaar    " Empty lists in a list
1352b544f3c8SBram Moolenaar    let a = []
1353b544f3c8SBram Moolenaar    redir => result
1354b544f3c8SBram Moolenaar    echo [a, a, a]
1355b544f3c8SBram Moolenaar    echo string([a, a, a])
1356b544f3c8SBram Moolenaar    redir END
1357b544f3c8SBram Moolenaar    let l = split(result, "\n")
1358b544f3c8SBram Moolenaar    call assert_equal(["[[], [], []]",
1359b544f3c8SBram Moolenaar		     \ "[[], [], []]"], l)
1360b544f3c8SBram Moolenaar
1361b544f3c8SBram Moolenaar    " Empty lists in a dictionary
1362b544f3c8SBram Moolenaar    let a = []
1363b544f3c8SBram Moolenaar    let b = {"a": a, "b": a}
1364b544f3c8SBram Moolenaar    redir => result
1365b544f3c8SBram Moolenaar    echo b
1366b544f3c8SBram Moolenaar    echo string(b)
1367b544f3c8SBram Moolenaar    redir END
1368b544f3c8SBram Moolenaar    let l = split(result, "\n")
1369b544f3c8SBram Moolenaar    call assert_equal(["{'a': [], 'b': []}",
1370b544f3c8SBram Moolenaar		     \ "{'a': [], 'b': []}"], l)
1371b544f3c8SBram Moolenaar
1372b544f3c8SBram Moolenaar    " Dictionaries in a list
1373b544f3c8SBram Moolenaar    let a = {"one": "yes", "two": "yes", "three": "yes"}
1374b544f3c8SBram Moolenaar    redir => result
1375b544f3c8SBram Moolenaar    echo [a, a, a]
1376b544f3c8SBram Moolenaar    echo string([a, a, a])
1377b544f3c8SBram Moolenaar    redir END
1378b544f3c8SBram Moolenaar    let l = split(result, "\n")
1379b544f3c8SBram Moolenaar    call assert_equal(["[{'one': 'yes', 'two': 'yes', 'three': 'yes'}, {...}, {...}]",
1380b544f3c8SBram Moolenaar		     \ "[{'one': 'yes', 'two': 'yes', 'three': 'yes'}, {'one': 'yes', 'two': 'yes', 'three': 'yes'}, {'one': 'yes', 'two': 'yes', 'three': 'yes'}]"], l)
1381b544f3c8SBram Moolenaar
1382b544f3c8SBram Moolenaar    " Dictionaries in a dictionary
1383b544f3c8SBram Moolenaar    let a = {"one": "yes", "two": "yes", "three": "yes"}
1384b544f3c8SBram Moolenaar    let b = {"a": a, "b": a}
1385b544f3c8SBram Moolenaar    redir => result
1386b544f3c8SBram Moolenaar    echo b
1387b544f3c8SBram Moolenaar    echo string(b)
1388b544f3c8SBram Moolenaar    redir END
1389b544f3c8SBram Moolenaar    let l = split(result, "\n")
1390b544f3c8SBram Moolenaar    call assert_equal(["{'a': {'one': 'yes', 'two': 'yes', 'three': 'yes'}, 'b': {...}}",
1391b544f3c8SBram Moolenaar		     \ "{'a': {'one': 'yes', 'two': 'yes', 'three': 'yes'}, 'b': {'one': 'yes', 'two': 'yes', 'three': 'yes'}}"], l)
1392b544f3c8SBram Moolenaar
1393b544f3c8SBram Moolenaar    " Lists in a list
1394b544f3c8SBram Moolenaar    let a = [1, 2, 3]
1395b544f3c8SBram Moolenaar    redir => result
1396b544f3c8SBram Moolenaar    echo [a, a, a]
1397b544f3c8SBram Moolenaar    echo string([a, a, a])
1398b544f3c8SBram Moolenaar    redir END
1399b544f3c8SBram Moolenaar    let l = split(result, "\n")
1400b544f3c8SBram Moolenaar    call assert_equal(["[[1, 2, 3], [...], [...]]",
1401b544f3c8SBram Moolenaar		     \ "[[1, 2, 3], [1, 2, 3], [1, 2, 3]]"], l)
1402b544f3c8SBram Moolenaar
1403b544f3c8SBram Moolenaar    " Lists in a dictionary
1404b544f3c8SBram Moolenaar    let a = [1, 2, 3]
1405b544f3c8SBram Moolenaar    let b = {"a": a, "b": a}
1406b544f3c8SBram Moolenaar    redir => result
1407b544f3c8SBram Moolenaar    echo b
1408b544f3c8SBram Moolenaar    echo string(b)
1409b544f3c8SBram Moolenaar    redir END
1410b544f3c8SBram Moolenaar    let l = split(result, "\n")
1411b544f3c8SBram Moolenaar    call assert_equal(["{'a': [1, 2, 3], 'b': [...]}",
1412b544f3c8SBram Moolenaar		     \ "{'a': [1, 2, 3], 'b': [1, 2, 3]}"], l)
1413b544f3c8SBram Moolenaar
1414b544f3c8SBram Moolenaarendfunc
1415b544f3c8SBram Moolenaar
1416b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
1417b544f3c8SBram Moolenaar" Test 94:  64-bit Numbers					    {{{1
1418b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
1419b544f3c8SBram Moolenaar
1420b544f3c8SBram Moolenaarfunc Test_num64()
1421b544f3c8SBram Moolenaar    call assert_notequal( 4294967296, 0)
1422b544f3c8SBram Moolenaar    call assert_notequal(-4294967296, 0)
1423b544f3c8SBram Moolenaar    call assert_equal( 4294967296,  0xFFFFffff + 1)
1424b544f3c8SBram Moolenaar    call assert_equal(-4294967296, -0xFFFFffff - 1)
1425b544f3c8SBram Moolenaar
1426b544f3c8SBram Moolenaar    call assert_equal( 9223372036854775807,  1 / 0)
1427b544f3c8SBram Moolenaar    call assert_equal(-9223372036854775807, -1 / 0)
1428b544f3c8SBram Moolenaar    call assert_equal(-9223372036854775807 - 1,  0 / 0)
1429b544f3c8SBram Moolenaar
14305feabe00SBram Moolenaar    if has('float')
1431b544f3c8SBram Moolenaar      call assert_equal( 0x7FFFffffFFFFffff, float2nr( 1.0e150))
1432b544f3c8SBram Moolenaar      call assert_equal(-0x7FFFffffFFFFffff, float2nr(-1.0e150))
14335feabe00SBram Moolenaar    endif
1434b544f3c8SBram Moolenaar
1435b544f3c8SBram Moolenaar    let rng = range(0xFFFFffff, 0x100000001)
1436b544f3c8SBram Moolenaar    call assert_equal([0xFFFFffff, 0x100000000, 0x100000001], rng)
1437b544f3c8SBram Moolenaar    call assert_equal(0x100000001, max(rng))
1438b544f3c8SBram Moolenaar    call assert_equal(0xFFFFffff, min(rng))
1439b544f3c8SBram Moolenaar    call assert_equal(rng, sort(range(0x100000001, 0xFFFFffff, -1), 'N'))
1440b544f3c8SBram Moolenaarendfunc
1441b544f3c8SBram Moolenaar
1442b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
1443b544f3c8SBram Moolenaar" Test 95:  lines of :append, :change, :insert			    {{{1
1444b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
1445b544f3c8SBram Moolenaar
1446b544f3c8SBram Moolenaarfunction! DefineFunction(name, body)
1447b544f3c8SBram Moolenaar    let func = join(['function! ' . a:name . '()'] + a:body + ['endfunction'], "\n")
1448b544f3c8SBram Moolenaar    exec func
1449b544f3c8SBram Moolenaarendfunction
1450b544f3c8SBram Moolenaar
1451b544f3c8SBram Moolenaarfunc Test_script_lines()
1452b544f3c8SBram Moolenaar    " :append
1453b544f3c8SBram Moolenaar    try
1454b544f3c8SBram Moolenaar	call DefineFunction('T_Append', [
1455b544f3c8SBram Moolenaar		    \ 'append',
1456b544f3c8SBram Moolenaar		    \ 'py <<EOS',
1457b544f3c8SBram Moolenaar		    \ '.',
1458b544f3c8SBram Moolenaar		    \ ])
1459b544f3c8SBram Moolenaar    catch
146037175409SBram Moolenaar	call assert_report("Can't define function")
1461b544f3c8SBram Moolenaar    endtry
1462b544f3c8SBram Moolenaar    try
1463b544f3c8SBram Moolenaar	call DefineFunction('T_Append', [
1464b544f3c8SBram Moolenaar		    \ 'append',
1465b544f3c8SBram Moolenaar		    \ 'abc',
1466b544f3c8SBram Moolenaar		    \ ])
146737175409SBram Moolenaar	call assert_report("Shouldn't be able to define function")
1468b544f3c8SBram Moolenaar    catch
1469b544f3c8SBram Moolenaar	call assert_exception('Vim(function):E126: Missing :endfunction')
1470b544f3c8SBram Moolenaar    endtry
1471b544f3c8SBram Moolenaar
1472b544f3c8SBram Moolenaar    " :change
1473b544f3c8SBram Moolenaar    try
1474b544f3c8SBram Moolenaar	call DefineFunction('T_Change', [
1475b544f3c8SBram Moolenaar		    \ 'change',
1476b544f3c8SBram Moolenaar		    \ 'py <<EOS',
1477b544f3c8SBram Moolenaar		    \ '.',
1478b544f3c8SBram Moolenaar		    \ ])
1479b544f3c8SBram Moolenaar    catch
148037175409SBram Moolenaar	call assert_report("Can't define function")
1481b544f3c8SBram Moolenaar    endtry
1482b544f3c8SBram Moolenaar    try
1483b544f3c8SBram Moolenaar	call DefineFunction('T_Change', [
1484b544f3c8SBram Moolenaar		    \ 'change',
1485b544f3c8SBram Moolenaar		    \ 'abc',
1486b544f3c8SBram Moolenaar		    \ ])
148737175409SBram Moolenaar	call assert_report("Shouldn't be able to define function")
1488b544f3c8SBram Moolenaar    catch
1489b544f3c8SBram Moolenaar	call assert_exception('Vim(function):E126: Missing :endfunction')
1490b544f3c8SBram Moolenaar    endtry
1491b544f3c8SBram Moolenaar
1492b544f3c8SBram Moolenaar    " :insert
1493b544f3c8SBram Moolenaar    try
1494b544f3c8SBram Moolenaar	call DefineFunction('T_Insert', [
1495b544f3c8SBram Moolenaar		    \ 'insert',
1496b544f3c8SBram Moolenaar		    \ 'py <<EOS',
1497b544f3c8SBram Moolenaar		    \ '.',
1498b544f3c8SBram Moolenaar		    \ ])
1499b544f3c8SBram Moolenaar    catch
150037175409SBram Moolenaar	call assert_report("Can't define function")
1501b544f3c8SBram Moolenaar    endtry
1502b544f3c8SBram Moolenaar    try
1503b544f3c8SBram Moolenaar	call DefineFunction('T_Insert', [
1504b544f3c8SBram Moolenaar		    \ 'insert',
1505b544f3c8SBram Moolenaar		    \ 'abc',
1506b544f3c8SBram Moolenaar		    \ ])
150737175409SBram Moolenaar	call assert_report("Shouldn't be able to define function")
1508b544f3c8SBram Moolenaar    catch
1509b544f3c8SBram Moolenaar	call assert_exception('Vim(function):E126: Missing :endfunction')
1510b544f3c8SBram Moolenaar    endtry
1511b544f3c8SBram Moolenaarendfunc
1512b544f3c8SBram Moolenaar
1513b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
1514478af67dSBram Moolenaar" Test 96:  line continuation						    {{{1
1515478af67dSBram Moolenaar"
1516478af67dSBram Moolenaar"	    Undefined behavior was detected by ubsan with line continuation
1517478af67dSBram Moolenaar"	    after an empty line.
1518478af67dSBram Moolenaar"-------------------------------------------------------------------------------
1519478af67dSBram Moolenaarfunc Test_script_emty_line_continuation()
1520478af67dSBram Moolenaar
1521478af67dSBram Moolenaar    \
1522478af67dSBram Moolenaarendfunc
1523478af67dSBram Moolenaar
1524478af67dSBram Moolenaar"-------------------------------------------------------------------------------
1525863e80b4SBram Moolenaar" Test 97:  bitwise functions						    {{{1
1526863e80b4SBram Moolenaar"-------------------------------------------------------------------------------
1527863e80b4SBram Moolenaarfunc Test_bitwise_functions()
1528863e80b4SBram Moolenaar    " and
1529863e80b4SBram Moolenaar    call assert_equal(127, and(127, 127))
1530863e80b4SBram Moolenaar    call assert_equal(16, and(127, 16))
1531073e4b92SBram Moolenaar    eval 127->and(16)->assert_equal(16)
1532863e80b4SBram Moolenaar    call assert_equal(0, and(127, 128))
1533863e80b4SBram Moolenaar    call assert_fails("call and([], 1)", 'E745:')
1534863e80b4SBram Moolenaar    call assert_fails("call and({}, 1)", 'E728:')
15355feabe00SBram Moolenaar    if has('float')
15365feabe00SBram Moolenaar      call assert_fails("call and(1.0, 1)", 'E805:')
1537863e80b4SBram Moolenaar      call assert_fails("call and(1, 1.0)", 'E805:')
15385feabe00SBram Moolenaar    endif
1539863e80b4SBram Moolenaar    call assert_fails("call and(1, [])", 'E745:')
1540863e80b4SBram Moolenaar    call assert_fails("call and(1, {})", 'E728:')
1541863e80b4SBram Moolenaar    " or
1542863e80b4SBram Moolenaar    call assert_equal(23, or(16, 7))
1543863e80b4SBram Moolenaar    call assert_equal(15, or(8, 7))
1544073e4b92SBram Moolenaar    eval 8->or(7)->assert_equal(15)
1545863e80b4SBram Moolenaar    call assert_equal(123, or(0, 123))
1546863e80b4SBram Moolenaar    call assert_fails("call or([], 1)", 'E745:')
1547863e80b4SBram Moolenaar    call assert_fails("call or({}, 1)", 'E728:')
15485feabe00SBram Moolenaar    if has('float')
15495feabe00SBram Moolenaar      call assert_fails("call or(1.0, 1)", 'E805:')
1550863e80b4SBram Moolenaar      call assert_fails("call or(1, 1.0)", 'E805:')
15515feabe00SBram Moolenaar    endif
1552863e80b4SBram Moolenaar    call assert_fails("call or(1, [])", 'E745:')
1553863e80b4SBram Moolenaar    call assert_fails("call or(1, {})", 'E728:')
1554863e80b4SBram Moolenaar    " xor
1555863e80b4SBram Moolenaar    call assert_equal(0, xor(127, 127))
1556863e80b4SBram Moolenaar    call assert_equal(111, xor(127, 16))
1557073e4b92SBram Moolenaar    eval 127->xor(16)->assert_equal(111)
1558863e80b4SBram Moolenaar    call assert_equal(255, xor(127, 128))
15595feabe00SBram Moolenaar    if has('float')
1560863e80b4SBram Moolenaar      call assert_fails("call xor(1.0, 1)", 'E805:')
15615feabe00SBram Moolenaar      call assert_fails("call xor(1, 1.0)", 'E805:')
15625feabe00SBram Moolenaar    endif
1563863e80b4SBram Moolenaar    call assert_fails("call xor([], 1)", 'E745:')
1564863e80b4SBram Moolenaar    call assert_fails("call xor({}, 1)", 'E728:')
1565863e80b4SBram Moolenaar    call assert_fails("call xor(1, [])", 'E745:')
1566863e80b4SBram Moolenaar    call assert_fails("call xor(1, {})", 'E728:')
1567863e80b4SBram Moolenaar    " invert
1568863e80b4SBram Moolenaar    call assert_equal(65408, and(invert(127), 65535))
1569073e4b92SBram Moolenaar    eval 127->invert()->and(65535)->assert_equal(65408)
1570863e80b4SBram Moolenaar    call assert_equal(65519, and(invert(16), 65535))
1571863e80b4SBram Moolenaar    call assert_equal(65407, and(invert(128), 65535))
15725feabe00SBram Moolenaar    if has('float')
1573863e80b4SBram Moolenaar      call assert_fails("call invert(1.0)", 'E805:')
15745feabe00SBram Moolenaar    endif
1575863e80b4SBram Moolenaar    call assert_fails("call invert([])", 'E745:')
1576863e80b4SBram Moolenaar    call assert_fails("call invert({})", 'E728:')
1577863e80b4SBram Moolenaarendfunc
1578863e80b4SBram Moolenaar
1579663bb233SBram Moolenaar" Test trailing text after :endfunction				    {{{1
1580663bb233SBram Moolenaarfunc Test_endfunction_trailing()
1581663bb233SBram Moolenaar    call assert_false(exists('*Xtest'))
1582663bb233SBram Moolenaar
1583663bb233SBram Moolenaar    exe "func Xtest()\necho 'hello'\nendfunc\nlet done = 'yes'"
1584663bb233SBram Moolenaar    call assert_true(exists('*Xtest'))
1585663bb233SBram Moolenaar    call assert_equal('yes', done)
1586663bb233SBram Moolenaar    delfunc Xtest
1587663bb233SBram Moolenaar    unlet done
1588663bb233SBram Moolenaar
1589663bb233SBram Moolenaar    exe "func Xtest()\necho 'hello'\nendfunc|let done = 'yes'"
1590663bb233SBram Moolenaar    call assert_true(exists('*Xtest'))
1591663bb233SBram Moolenaar    call assert_equal('yes', done)
1592663bb233SBram Moolenaar    delfunc Xtest
1593663bb233SBram Moolenaar    unlet done
1594663bb233SBram Moolenaar
159553564f7cSBram Moolenaar    " trailing line break
159653564f7cSBram Moolenaar    exe "func Xtest()\necho 'hello'\nendfunc\n"
159753564f7cSBram Moolenaar    call assert_true(exists('*Xtest'))
159853564f7cSBram Moolenaar    delfunc Xtest
159953564f7cSBram Moolenaar
1600663bb233SBram Moolenaar    set verbose=1
1601663bb233SBram Moolenaar    exe "func Xtest()\necho 'hello'\nendfunc \" garbage"
1602f8be461dSBram Moolenaar    call assert_notmatch('W22:', split(execute('1messages'), "\n")[0])
1603663bb233SBram Moolenaar    call assert_true(exists('*Xtest'))
1604663bb233SBram Moolenaar    delfunc Xtest
1605663bb233SBram Moolenaar
1606f8be461dSBram Moolenaar    exe "func Xtest()\necho 'hello'\nendfunc garbage"
1607f8be461dSBram Moolenaar    call assert_match('W22:', split(execute('1messages'), "\n")[0])
1608663bb233SBram Moolenaar    call assert_true(exists('*Xtest'))
1609663bb233SBram Moolenaar    delfunc Xtest
1610663bb233SBram Moolenaar    set verbose=0
161153564f7cSBram Moolenaar
161253564f7cSBram Moolenaar    function Foo()
161353564f7cSBram Moolenaar	echo 'hello'
161453564f7cSBram Moolenaar    endfunction | echo 'xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx'
161553564f7cSBram Moolenaar    delfunc Foo
1616663bb233SBram Moolenaarendfunc
1617663bb233SBram Moolenaar
1618d6abcd15SBram Moolenaarfunc Test_delfunction_force()
1619d6abcd15SBram Moolenaar    delfunc! Xtest
1620d6abcd15SBram Moolenaar    delfunc! Xtest
1621d6abcd15SBram Moolenaar    func Xtest()
1622d6abcd15SBram Moolenaar	echo 'nothing'
1623d6abcd15SBram Moolenaar    endfunc
1624d6abcd15SBram Moolenaar    delfunc! Xtest
1625d6abcd15SBram Moolenaar    delfunc! Xtest
16268dfcce3aSBram Moolenaar
16278dfcce3aSBram Moolenaar    " Try deleting the current function
16288dfcce3aSBram Moolenaar    call assert_fails('delfunc Test_delfunction_force', 'E131:')
1629d6abcd15SBram Moolenaarendfunc
1630d6abcd15SBram Moolenaar
16316f9a476bSBram Moolenaar" Test using bang after user command				    {{{1
16326f9a476bSBram Moolenaarfunc Test_user_command_with_bang()
16336f9a476bSBram Moolenaar    command -bang Nieuw let nieuw = 1
16346f9a476bSBram Moolenaar    Ni!
16356f9a476bSBram Moolenaar    call assert_equal(1, nieuw)
16366f9a476bSBram Moolenaar    unlet nieuw
16376f9a476bSBram Moolenaar    delcommand Nieuw
16386f9a476bSBram Moolenaarendfunc
16396f9a476bSBram Moolenaar
1640bb3e6416SBram Moolenaar" Test for script-local function
1641bb3e6416SBram Moolenaarfunc <SID>DoLast()
1642bb3e6416SBram Moolenaar  call append(line('$'), "last line")
1643bb3e6416SBram Moolenaarendfunc
1644bb3e6416SBram Moolenaar
1645bb3e6416SBram Moolenaarfunc s:DoNothing()
1646bb3e6416SBram Moolenaar  call append(line('$'), "nothing line")
1647bb3e6416SBram Moolenaarendfunc
1648bb3e6416SBram Moolenaar
1649bb3e6416SBram Moolenaarfunc Test_script_local_func()
165034059e7bSBram Moolenaar  set nocp nomore viminfo+=nviminfo
1651bb3e6416SBram Moolenaar  new
1652bb3e6416SBram Moolenaar  nnoremap <buffer> _x	:call <SID>DoNothing()<bar>call <SID>DoLast()<bar>delfunc <SID>DoNothing<bar>delfunc <SID>DoLast<cr>
1653bb3e6416SBram Moolenaar
1654bb3e6416SBram Moolenaar  normal _x
1655bb3e6416SBram Moolenaar  call assert_equal('nothing line', getline(2))
1656bb3e6416SBram Moolenaar  call assert_equal('last line', getline(3))
1657bb3e6416SBram Moolenaar  enew! | close
1658bb3e6416SBram Moolenaarendfunc
1659bb3e6416SBram Moolenaar
1660b9adef79SBram Moolenaarfunc Test_script_expand_sfile()
1661b9adef79SBram Moolenaar  let lines =<< trim END
1662b9adef79SBram Moolenaar    func s:snr()
1663b9adef79SBram Moolenaar      return expand('<sfile>')
1664b9adef79SBram Moolenaar    endfunc
1665b9adef79SBram Moolenaar    let g:result = s:snr()
1666b9adef79SBram Moolenaar  END
1667b9adef79SBram Moolenaar  call writefile(lines, 'Xexpand')
1668b9adef79SBram Moolenaar  source Xexpand
1669b9adef79SBram Moolenaar  call assert_match('<SNR>\d\+_snr', g:result)
1670b9adef79SBram Moolenaar  source Xexpand
1671b9adef79SBram Moolenaar  call assert_match('<SNR>\d\+_snr', g:result)
1672b9adef79SBram Moolenaar
1673b9adef79SBram Moolenaar  call delete('Xexpand')
1674b9adef79SBram Moolenaar  unlet g:result
1675b9adef79SBram Moolenaarendfunc
1676b9adef79SBram Moolenaar
1677ff697e6cSBram Moolenaarfunc Test_compound_assignment_operators()
1678ff697e6cSBram Moolenaar    " Test for number
1679ff697e6cSBram Moolenaar    let x = 1
1680ff697e6cSBram Moolenaar    let x += 10
1681ff697e6cSBram Moolenaar    call assert_equal(11, x)
1682ff697e6cSBram Moolenaar    let x -= 5
1683ff697e6cSBram Moolenaar    call assert_equal(6, x)
1684ff697e6cSBram Moolenaar    let x *= 4
1685ff697e6cSBram Moolenaar    call assert_equal(24, x)
1686ff697e6cSBram Moolenaar    let x /= 3
1687ff697e6cSBram Moolenaar    call assert_equal(8, x)
1688ff697e6cSBram Moolenaar    let x %= 3
1689ff697e6cSBram Moolenaar    call assert_equal(2, x)
1690ff697e6cSBram Moolenaar    let x .= 'n'
1691ff697e6cSBram Moolenaar    call assert_equal('2n', x)
1692ff697e6cSBram Moolenaar
1693e21c1580SBram Moolenaar    " Test special cases: division or modulus with 0.
1694e21c1580SBram Moolenaar    let x = 1
1695e21c1580SBram Moolenaar    let x /= 0
1696e21c1580SBram Moolenaar    call assert_equal(0x7FFFFFFFFFFFFFFF, x)
1697e21c1580SBram Moolenaar
1698e21c1580SBram Moolenaar    let x = -1
1699e21c1580SBram Moolenaar    let x /= 0
1700e21c1580SBram Moolenaar    call assert_equal(-0x7FFFFFFFFFFFFFFF, x)
1701e21c1580SBram Moolenaar
1702e21c1580SBram Moolenaar    let x = 0
1703e21c1580SBram Moolenaar    let x /= 0
1704e21c1580SBram Moolenaar    call assert_equal(-0x7FFFFFFFFFFFFFFF - 1, x)
1705e21c1580SBram Moolenaar
1706e21c1580SBram Moolenaar    let x = 1
1707e21c1580SBram Moolenaar    let x %= 0
1708e21c1580SBram Moolenaar    call assert_equal(0, x)
1709e21c1580SBram Moolenaar
1710e21c1580SBram Moolenaar    let x = -1
1711e21c1580SBram Moolenaar    let x %= 0
1712e21c1580SBram Moolenaar    call assert_equal(0, x)
1713e21c1580SBram Moolenaar
1714e21c1580SBram Moolenaar    let x = 0
1715e21c1580SBram Moolenaar    let x %= 0
1716e21c1580SBram Moolenaar    call assert_equal(0, x)
1717e21c1580SBram Moolenaar
1718ff697e6cSBram Moolenaar    " Test for string
1719ff697e6cSBram Moolenaar    let x = 'str'
1720ff697e6cSBram Moolenaar    let x .= 'ing'
1721ff697e6cSBram Moolenaar    call assert_equal('string', x)
1722ff697e6cSBram Moolenaar    let x += 1
1723ff697e6cSBram Moolenaar    call assert_equal(1, x)
1724ff697e6cSBram Moolenaar
1725ff697e6cSBram Moolenaar    if has('float')
1726ff697e6cSBram Moolenaar      " Test for float
17275feabe00SBram Moolenaar      let x -= 1.5
17285feabe00SBram Moolenaar      call assert_equal(-0.5, x)
1729ff697e6cSBram Moolenaar      let x = 0.5
1730ff697e6cSBram Moolenaar      let x += 4.5
1731ff697e6cSBram Moolenaar      call assert_equal(5.0, x)
1732ff697e6cSBram Moolenaar      let x -= 1.5
1733ff697e6cSBram Moolenaar      call assert_equal(3.5, x)
1734ff697e6cSBram Moolenaar      let x *= 3.0
1735ff697e6cSBram Moolenaar      call assert_equal(10.5, x)
1736ff697e6cSBram Moolenaar      let x /= 2.5
1737ff697e6cSBram Moolenaar      call assert_equal(4.2, x)
1738ff697e6cSBram Moolenaar      call assert_fails('let x %= 0.5', 'E734')
1739ff697e6cSBram Moolenaar      call assert_fails('let x .= "f"', 'E734')
17408b633135SBram Moolenaar      let x = !3.14
17418b633135SBram Moolenaar      call assert_equal(0.0, x)
1742ff697e6cSBram Moolenaar    endif
1743ff697e6cSBram Moolenaar
1744ff697e6cSBram Moolenaar    " Test for environment variable
1745ff697e6cSBram Moolenaar    let $FOO = 1
1746ff697e6cSBram Moolenaar    call assert_fails('let $FOO += 1', 'E734')
1747ff697e6cSBram Moolenaar    call assert_fails('let $FOO -= 1', 'E734')
1748ff697e6cSBram Moolenaar    call assert_fails('let $FOO *= 1', 'E734')
1749ff697e6cSBram Moolenaar    call assert_fails('let $FOO /= 1', 'E734')
1750ff697e6cSBram Moolenaar    call assert_fails('let $FOO %= 1', 'E734')
1751ff697e6cSBram Moolenaar    let $FOO .= 's'
1752ff697e6cSBram Moolenaar    call assert_equal('1s', $FOO)
1753ff697e6cSBram Moolenaar    unlet $FOO
1754ff697e6cSBram Moolenaar
1755ff697e6cSBram Moolenaar    " Test for option variable (type: number)
1756ff697e6cSBram Moolenaar    let &scrolljump = 1
1757ff697e6cSBram Moolenaar    let &scrolljump += 5
1758ff697e6cSBram Moolenaar    call assert_equal(6, &scrolljump)
1759ff697e6cSBram Moolenaar    let &scrolljump -= 2
1760ff697e6cSBram Moolenaar    call assert_equal(4, &scrolljump)
1761ff697e6cSBram Moolenaar    let &scrolljump *= 3
1762ff697e6cSBram Moolenaar    call assert_equal(12, &scrolljump)
1763ff697e6cSBram Moolenaar    let &scrolljump /= 2
1764ff697e6cSBram Moolenaar    call assert_equal(6, &scrolljump)
1765ff697e6cSBram Moolenaar    let &scrolljump %= 5
1766ff697e6cSBram Moolenaar    call assert_equal(1, &scrolljump)
1767ff697e6cSBram Moolenaar    call assert_fails('let &scrolljump .= "j"', 'E734')
1768ff697e6cSBram Moolenaar    set scrolljump&vim
1769ff697e6cSBram Moolenaar
1770ff697e6cSBram Moolenaar    " Test for register
1771ff697e6cSBram Moolenaar    let @/ = 1
1772ff697e6cSBram Moolenaar    call assert_fails('let @/ += 1', 'E734')
1773ff697e6cSBram Moolenaar    call assert_fails('let @/ -= 1', 'E734')
1774ff697e6cSBram Moolenaar    call assert_fails('let @/ *= 1', 'E734')
1775ff697e6cSBram Moolenaar    call assert_fails('let @/ /= 1', 'E734')
1776ff697e6cSBram Moolenaar    call assert_fails('let @/ %= 1', 'E734')
1777ff697e6cSBram Moolenaar    let @/ .= 's'
1778ff697e6cSBram Moolenaar    call assert_equal('1s', @/)
1779ff697e6cSBram Moolenaar    let @/ = ''
1780ff697e6cSBram Moolenaarendfunc
1781ff697e6cSBram Moolenaar
1782c3e92c16SBram Moolenaarfunc Test_refcount()
1783c3e92c16SBram Moolenaar    " Immediate values
1784c3e92c16SBram Moolenaar    call assert_equal(-1, test_refcount(1))
1785c3e92c16SBram Moolenaar    call assert_equal(-1, test_refcount('s'))
1786c3e92c16SBram Moolenaar    call assert_equal(-1, test_refcount(v:true))
1787c3e92c16SBram Moolenaar    call assert_equal(0, test_refcount([]))
1788c3e92c16SBram Moolenaar    call assert_equal(0, test_refcount({}))
1789c3e92c16SBram Moolenaar    call assert_equal(0, test_refcount(0zff))
1790c3e92c16SBram Moolenaar    call assert_equal(0, test_refcount({-> line('.')}))
1791c3e92c16SBram Moolenaar    if has('float')
1792c3e92c16SBram Moolenaar        call assert_equal(-1, test_refcount(0.1))
1793c3e92c16SBram Moolenaar    endif
1794c3e92c16SBram Moolenaar    if has('job')
1795c3e92c16SBram Moolenaar        call assert_equal(0, test_refcount(job_start([&shell, &shellcmdflag, 'echo .'])))
1796c3e92c16SBram Moolenaar    endif
1797c3e92c16SBram Moolenaar
1798c3e92c16SBram Moolenaar    " No refcount types
1799c3e92c16SBram Moolenaar    let x = 1
1800c3e92c16SBram Moolenaar    call assert_equal(-1, test_refcount(x))
1801c3e92c16SBram Moolenaar    let x = 's'
1802c3e92c16SBram Moolenaar    call assert_equal(-1, test_refcount(x))
1803c3e92c16SBram Moolenaar    let x = v:true
1804c3e92c16SBram Moolenaar    call assert_equal(-1, test_refcount(x))
1805c3e92c16SBram Moolenaar    if has('float')
1806c3e92c16SBram Moolenaar        let x = 0.1
1807c3e92c16SBram Moolenaar        call assert_equal(-1, test_refcount(x))
1808c3e92c16SBram Moolenaar    endif
1809c3e92c16SBram Moolenaar
1810c3e92c16SBram Moolenaar    " Check refcount
1811c3e92c16SBram Moolenaar    let x = []
1812c3e92c16SBram Moolenaar    call assert_equal(1, test_refcount(x))
1813c3e92c16SBram Moolenaar
1814c3e92c16SBram Moolenaar    let x = {}
1815ce90e36fSBram Moolenaar    call assert_equal(1, x->test_refcount())
1816c3e92c16SBram Moolenaar
1817c3e92c16SBram Moolenaar    let x = 0zff
1818c3e92c16SBram Moolenaar    call assert_equal(1, test_refcount(x))
1819c3e92c16SBram Moolenaar
1820c3e92c16SBram Moolenaar    let X = {-> line('.')}
1821c3e92c16SBram Moolenaar    call assert_equal(1, test_refcount(X))
1822c3e92c16SBram Moolenaar    let Y = X
1823c3e92c16SBram Moolenaar    call assert_equal(2, test_refcount(X))
1824c3e92c16SBram Moolenaar
1825c3e92c16SBram Moolenaar    if has('job')
1826c3e92c16SBram Moolenaar        let job = job_start([&shell, &shellcmdflag, 'echo .'])
1827c3e92c16SBram Moolenaar        call assert_equal(1, test_refcount(job))
1828c3e92c16SBram Moolenaar        call assert_equal(1, test_refcount(job_getchannel(job)))
1829c3e92c16SBram Moolenaar        call assert_equal(1, test_refcount(job))
1830c3e92c16SBram Moolenaar    endif
1831c3e92c16SBram Moolenaar
1832c3e92c16SBram Moolenaar    " Function arguments, copying and unassigning
1833c3e92c16SBram Moolenaar    func ExprCheck(x, i)
1834c3e92c16SBram Moolenaar        let i = a:i + 1
1835c3e92c16SBram Moolenaar        call assert_equal(i, test_refcount(a:x))
1836c3e92c16SBram Moolenaar        let Y = a:x
1837c3e92c16SBram Moolenaar        call assert_equal(i + 1, test_refcount(a:x))
1838c3e92c16SBram Moolenaar        call assert_equal(test_refcount(a:x), test_refcount(Y))
1839c3e92c16SBram Moolenaar        let Y = 0
1840c3e92c16SBram Moolenaar        call assert_equal(i, test_refcount(a:x))
1841c3e92c16SBram Moolenaar    endfunc
1842c3e92c16SBram Moolenaar    call ExprCheck([], 0)
1843c3e92c16SBram Moolenaar    call ExprCheck({}, 0)
1844c3e92c16SBram Moolenaar    call ExprCheck(0zff, 0)
1845c3e92c16SBram Moolenaar    call ExprCheck({-> line('.')}, 0)
1846c3e92c16SBram Moolenaar    if has('job')
1847c3e92c16SBram Moolenaar	call ExprCheck(job, 1)
1848c3e92c16SBram Moolenaar	call ExprCheck(job_getchannel(job), 1)
1849c3e92c16SBram Moolenaar	call job_stop(job)
1850c3e92c16SBram Moolenaar    endif
1851c3e92c16SBram Moolenaar    delfunc ExprCheck
1852c3e92c16SBram Moolenaar
1853c3e92c16SBram Moolenaar    " Regarding function
1854c3e92c16SBram Moolenaar    func Func(x) abort
1855c3e92c16SBram Moolenaar        call assert_equal(2, test_refcount(function('Func')))
1856c3e92c16SBram Moolenaar        call assert_equal(0, test_refcount(funcref('Func')))
1857c3e92c16SBram Moolenaar    endfunc
1858c3e92c16SBram Moolenaar    call assert_equal(1, test_refcount(function('Func')))
1859c3e92c16SBram Moolenaar    call assert_equal(0, test_refcount(function('Func', [1])))
1860c3e92c16SBram Moolenaar    call assert_equal(0, test_refcount(funcref('Func')))
1861c3e92c16SBram Moolenaar    call assert_equal(0, test_refcount(funcref('Func', [1])))
1862c3e92c16SBram Moolenaar    let X = function('Func')
1863c3e92c16SBram Moolenaar    let Y = X
1864c3e92c16SBram Moolenaar    call assert_equal(1, test_refcount(X))
1865c3e92c16SBram Moolenaar    let X = function('Func', [1])
1866c3e92c16SBram Moolenaar    let Y = X
1867c3e92c16SBram Moolenaar    call assert_equal(2, test_refcount(X))
1868c3e92c16SBram Moolenaar    let X = funcref('Func')
1869c3e92c16SBram Moolenaar    let Y = X
1870c3e92c16SBram Moolenaar    call assert_equal(2, test_refcount(X))
1871c3e92c16SBram Moolenaar    let X = funcref('Func', [1])
1872c3e92c16SBram Moolenaar    let Y = X
1873c3e92c16SBram Moolenaar    call assert_equal(2, test_refcount(X))
1874c3e92c16SBram Moolenaar    unlet X
1875c3e92c16SBram Moolenaar    unlet Y
1876c3e92c16SBram Moolenaar    call Func(1)
1877c3e92c16SBram Moolenaar    delfunc Func
1878c3e92c16SBram Moolenaar
1879c3e92c16SBram Moolenaar    " Function with dict
1880c3e92c16SBram Moolenaar    func DictFunc() dict
1881c3e92c16SBram Moolenaar        call assert_equal(3, test_refcount(self))
1882c3e92c16SBram Moolenaar    endfunc
1883c3e92c16SBram Moolenaar    let d = {'Func': function('DictFunc')}
1884c3e92c16SBram Moolenaar    call assert_equal(1, test_refcount(d))
1885c3e92c16SBram Moolenaar    call assert_equal(0, test_refcount(d.Func))
1886c3e92c16SBram Moolenaar    call d.Func()
1887c3e92c16SBram Moolenaar    unlet d
1888c3e92c16SBram Moolenaar    delfunc DictFunc
1889c3e92c16SBram Moolenaarendfunc
1890c3e92c16SBram Moolenaar
18917d491c42SBram Moolenaarfunc Test_funccall_garbage_collect()
18926e5000d4SBram Moolenaar    func Func(x, ...)
18936e5000d4SBram Moolenaar        call add(a:x, a:000)
18946e5000d4SBram Moolenaar    endfunc
18956e5000d4SBram Moolenaar    call Func([], [])
18966e5000d4SBram Moolenaar    " Must not crash cause by invalid freeing
18976e5000d4SBram Moolenaar    call test_garbagecollect_now()
18986e5000d4SBram Moolenaar    call assert_true(v:true)
18996e5000d4SBram Moolenaar    delfunc Func
19006e5000d4SBram Moolenaarendfunc
19016e5000d4SBram Moolenaar
1902bc2cfe46SBram Moolenaarfunc Test_function_defined_line()
19038c5a278fSBram Moolenaar    CheckNotGui
1904bc2cfe46SBram Moolenaar
1905bc2cfe46SBram Moolenaar    let lines =<< trim [CODE]
1906bc2cfe46SBram Moolenaar    " F1
1907bc2cfe46SBram Moolenaar    func F1()
1908bc2cfe46SBram Moolenaar        " F2
1909bc2cfe46SBram Moolenaar        func F2()
1910bc2cfe46SBram Moolenaar            "
1911bc2cfe46SBram Moolenaar            "
1912bc2cfe46SBram Moolenaar            "
1913bc2cfe46SBram Moolenaar            return
1914bc2cfe46SBram Moolenaar        endfunc
1915bc2cfe46SBram Moolenaar        " F3
1916bc2cfe46SBram Moolenaar        execute "func F3()\n\n\n\nreturn\nendfunc"
1917bc2cfe46SBram Moolenaar        " F4
1918bc2cfe46SBram Moolenaar        execute "func F4()\n
1919bc2cfe46SBram Moolenaar                    \\n
1920bc2cfe46SBram Moolenaar                    \\n
1921bc2cfe46SBram Moolenaar                    \\n
1922bc2cfe46SBram Moolenaar                    \return\n
1923bc2cfe46SBram Moolenaar                    \endfunc"
1924bc2cfe46SBram Moolenaar    endfunc
1925bc2cfe46SBram Moolenaar    " F5
1926bc2cfe46SBram Moolenaar    execute "func F5()\n\n\n\nreturn\nendfunc"
1927bc2cfe46SBram Moolenaar    " F6
1928bc2cfe46SBram Moolenaar    execute "func F6()\n
1929bc2cfe46SBram Moolenaar                \\n
1930bc2cfe46SBram Moolenaar                \\n
1931bc2cfe46SBram Moolenaar                \\n
1932bc2cfe46SBram Moolenaar                \return\n
1933bc2cfe46SBram Moolenaar                \endfunc"
1934bc2cfe46SBram Moolenaar    call F1()
1935bc2cfe46SBram Moolenaar    verbose func F1
1936bc2cfe46SBram Moolenaar    verbose func F2
1937bc2cfe46SBram Moolenaar    verbose func F3
1938bc2cfe46SBram Moolenaar    verbose func F4
1939bc2cfe46SBram Moolenaar    verbose func F5
1940bc2cfe46SBram Moolenaar    verbose func F6
1941bc2cfe46SBram Moolenaar    qall!
1942bc2cfe46SBram Moolenaar    [CODE]
1943bc2cfe46SBram Moolenaar
1944bc2cfe46SBram Moolenaar    call writefile(lines, 'Xtest.vim')
194593344c2dSBram Moolenaar    let res = system(GetVimCommandClean() .. ' -es -X -S Xtest.vim')
1946bc2cfe46SBram Moolenaar    call assert_equal(0, v:shell_error)
1947bc2cfe46SBram Moolenaar
1948bc2cfe46SBram Moolenaar    let m = matchstr(res, 'function F1()[^[:print:]]*[[:print:]]*')
1949bc2cfe46SBram Moolenaar    call assert_match(' line 2$', m)
1950bc2cfe46SBram Moolenaar
1951bc2cfe46SBram Moolenaar    let m = matchstr(res, 'function F2()[^[:print:]]*[[:print:]]*')
1952bc2cfe46SBram Moolenaar    call assert_match(' line 4$', m)
1953bc2cfe46SBram Moolenaar
1954bc2cfe46SBram Moolenaar    let m = matchstr(res, 'function F3()[^[:print:]]*[[:print:]]*')
1955bc2cfe46SBram Moolenaar    call assert_match(' line 11$', m)
1956bc2cfe46SBram Moolenaar
1957bc2cfe46SBram Moolenaar    let m = matchstr(res, 'function F4()[^[:print:]]*[[:print:]]*')
1958bc2cfe46SBram Moolenaar    call assert_match(' line 13$', m)
1959bc2cfe46SBram Moolenaar
1960bc2cfe46SBram Moolenaar    let m = matchstr(res, 'function F5()[^[:print:]]*[[:print:]]*')
1961bc2cfe46SBram Moolenaar    call assert_match(' line 21$', m)
1962bc2cfe46SBram Moolenaar
1963bc2cfe46SBram Moolenaar    let m = matchstr(res, 'function F6()[^[:print:]]*[[:print:]]*')
1964bc2cfe46SBram Moolenaar    call assert_match(' line 23$', m)
1965bc2cfe46SBram Moolenaar
1966bc2cfe46SBram Moolenaar    call delete('Xtest.vim')
1967bc2cfe46SBram Moolenaarendfunc
1968bc2cfe46SBram Moolenaar
19699f6277bdSBram Moolenaar" Test for missing :endif, :endfor, :endwhile and :endtry           {{{1
19709f6277bdSBram Moolenaarfunc Test_missing_end()
19719f6277bdSBram Moolenaar  call writefile(['if 2 > 1', 'echo ">"'], 'Xscript')
19729f6277bdSBram Moolenaar  call assert_fails('source Xscript', 'E171:')
19739f6277bdSBram Moolenaar  call writefile(['for i in range(5)', 'echo i'], 'Xscript')
19749f6277bdSBram Moolenaar  call assert_fails('source Xscript', 'E170:')
19759f6277bdSBram Moolenaar  call writefile(['while v:true', 'echo "."'], 'Xscript')
19769f6277bdSBram Moolenaar  call assert_fails('source Xscript', 'E170:')
19779f6277bdSBram Moolenaar  call writefile(['try', 'echo "."'], 'Xscript')
19789f6277bdSBram Moolenaar  call assert_fails('source Xscript', 'E600:')
19799f6277bdSBram Moolenaar  call delete('Xscript')
1980818fc9adSBram Moolenaar
1981818fc9adSBram Moolenaar  " Using endfor with :while
1982818fc9adSBram Moolenaar  let caught_e732 = 0
1983818fc9adSBram Moolenaar  try
1984818fc9adSBram Moolenaar    while v:true
1985818fc9adSBram Moolenaar    endfor
1986818fc9adSBram Moolenaar  catch /E732:/
1987818fc9adSBram Moolenaar    let caught_e732 = 1
1988818fc9adSBram Moolenaar  endtry
1989818fc9adSBram Moolenaar  call assert_equal(1, caught_e732)
1990818fc9adSBram Moolenaar
1991818fc9adSBram Moolenaar  " Using endwhile with :for
1992818fc9adSBram Moolenaar  let caught_e733 = 0
1993818fc9adSBram Moolenaar  try
1994818fc9adSBram Moolenaar    for i in range(1)
1995818fc9adSBram Moolenaar    endwhile
1996818fc9adSBram Moolenaar  catch /E733:/
1997818fc9adSBram Moolenaar    let caught_e733 = 1
1998818fc9adSBram Moolenaar  endtry
1999818fc9adSBram Moolenaar  call assert_equal(1, caught_e733)
2000818fc9adSBram Moolenaar
2001*ee4e0c1eSBram Moolenaar  " Using endfunc with :if
2002*ee4e0c1eSBram Moolenaar  call assert_fails('exe "if 1 | endfunc | endif"', 'E193:')
2003*ee4e0c1eSBram Moolenaar
2004818fc9adSBram Moolenaar  " Missing 'in' in a :for statement
2005818fc9adSBram Moolenaar  call assert_fails('for i range(1) | endfor', 'E690:')
20069f6277bdSBram Moolenaarendfunc
20079f6277bdSBram Moolenaar
20089f6277bdSBram Moolenaar" Test for deep nesting of if/for/while/try statements              {{{1
20099f6277bdSBram Moolenaarfunc Test_deep_nest()
20109f6277bdSBram Moolenaar  if !CanRunVimInTerminal()
20119f6277bdSBram Moolenaar    throw 'Skipped: cannot run vim in terminal'
20129f6277bdSBram Moolenaar  endif
20139f6277bdSBram Moolenaar
20149f6277bdSBram Moolenaar  let lines =<< trim [SCRIPT]
20159f6277bdSBram Moolenaar    " Deep nesting of if ... endif
20169f6277bdSBram Moolenaar    func Test1()
20179f6277bdSBram Moolenaar      let @a = join(repeat(['if v:true'], 51), "\n")
20189f6277bdSBram Moolenaar      let @a ..= "\n"
20199f6277bdSBram Moolenaar      let @a ..= join(repeat(['endif'], 51), "\n")
20209f6277bdSBram Moolenaar      @a
20219f6277bdSBram Moolenaar      let @a = ''
20229f6277bdSBram Moolenaar    endfunc
20239f6277bdSBram Moolenaar
20249f6277bdSBram Moolenaar    " Deep nesting of for ... endfor
20259f6277bdSBram Moolenaar    func Test2()
20269f6277bdSBram Moolenaar      let @a = join(repeat(['for i in [1]'], 51), "\n")
20279f6277bdSBram Moolenaar      let @a ..= "\n"
20289f6277bdSBram Moolenaar      let @a ..= join(repeat(['endfor'], 51), "\n")
20299f6277bdSBram Moolenaar      @a
20309f6277bdSBram Moolenaar      let @a = ''
20319f6277bdSBram Moolenaar    endfunc
20329f6277bdSBram Moolenaar
20339f6277bdSBram Moolenaar    " Deep nesting of while ... endwhile
20349f6277bdSBram Moolenaar    func Test3()
20359f6277bdSBram Moolenaar      let @a = join(repeat(['while v:true'], 51), "\n")
20369f6277bdSBram Moolenaar      let @a ..= "\n"
20379f6277bdSBram Moolenaar      let @a ..= join(repeat(['endwhile'], 51), "\n")
20389f6277bdSBram Moolenaar      @a
20399f6277bdSBram Moolenaar      let @a = ''
20409f6277bdSBram Moolenaar    endfunc
20419f6277bdSBram Moolenaar
20429f6277bdSBram Moolenaar    " Deep nesting of try ... endtry
20439f6277bdSBram Moolenaar    func Test4()
20449f6277bdSBram Moolenaar      let @a = join(repeat(['try'], 51), "\n")
20459f6277bdSBram Moolenaar      let @a ..= "\necho v:true\n"
20469f6277bdSBram Moolenaar      let @a ..= join(repeat(['endtry'], 51), "\n")
20479f6277bdSBram Moolenaar      @a
20489f6277bdSBram Moolenaar      let @a = ''
20499f6277bdSBram Moolenaar    endfunc
2050*ee4e0c1eSBram Moolenaar
2051*ee4e0c1eSBram Moolenaar    " Deep nesting of function ... endfunction
2052*ee4e0c1eSBram Moolenaar    func Test5()
2053*ee4e0c1eSBram Moolenaar      let @a = join(repeat(['function X()'], 51), "\n")
2054*ee4e0c1eSBram Moolenaar      let @a ..= "\necho v:true\n"
2055*ee4e0c1eSBram Moolenaar      let @a ..= join(repeat(['endfunction'], 51), "\n")
2056*ee4e0c1eSBram Moolenaar      @a
2057*ee4e0c1eSBram Moolenaar      let @a = ''
2058*ee4e0c1eSBram Moolenaar    endfunc
20599f6277bdSBram Moolenaar  [SCRIPT]
20609f6277bdSBram Moolenaar  call writefile(lines, 'Xscript')
20619f6277bdSBram Moolenaar
20629f6277bdSBram Moolenaar  let buf = RunVimInTerminal('-S Xscript', {'rows': 6})
20639f6277bdSBram Moolenaar
20649f6277bdSBram Moolenaar  " Deep nesting of if ... endif
20659f6277bdSBram Moolenaar  call term_sendkeys(buf, ":call Test1()\n")
2066*ee4e0c1eSBram Moolenaar  call term_wait(buf)
20679f6277bdSBram Moolenaar  call WaitForAssert({-> assert_match('^E579:', term_getline(buf, 5))})
20689f6277bdSBram Moolenaar
20699f6277bdSBram Moolenaar  " Deep nesting of for ... endfor
20709f6277bdSBram Moolenaar  call term_sendkeys(buf, ":call Test2()\n")
2071*ee4e0c1eSBram Moolenaar  call term_wait(buf)
20729f6277bdSBram Moolenaar  call WaitForAssert({-> assert_match('^E585:', term_getline(buf, 5))})
20739f6277bdSBram Moolenaar
20749f6277bdSBram Moolenaar  " Deep nesting of while ... endwhile
20759f6277bdSBram Moolenaar  call term_sendkeys(buf, ":call Test3()\n")
2076*ee4e0c1eSBram Moolenaar  call term_wait(buf)
20779f6277bdSBram Moolenaar  call WaitForAssert({-> assert_match('^E585:', term_getline(buf, 5))})
20789f6277bdSBram Moolenaar
20799f6277bdSBram Moolenaar  " Deep nesting of try ... endtry
20809f6277bdSBram Moolenaar  call term_sendkeys(buf, ":call Test4()\n")
2081*ee4e0c1eSBram Moolenaar  call term_wait(buf)
20829f6277bdSBram Moolenaar  call WaitForAssert({-> assert_match('^E601:', term_getline(buf, 5))})
20839f6277bdSBram Moolenaar
2084*ee4e0c1eSBram Moolenaar  " Deep nesting of function ... endfunction
2085*ee4e0c1eSBram Moolenaar  call term_sendkeys(buf, ":call Test5()\n")
2086*ee4e0c1eSBram Moolenaar  call term_wait(buf)
2087*ee4e0c1eSBram Moolenaar  call WaitForAssert({-> assert_match('^E1058:', term_getline(buf, 4))})
2088*ee4e0c1eSBram Moolenaar  call term_sendkeys(buf, "\<C-C>\n")
2089*ee4e0c1eSBram Moolenaar  call term_wait(buf)
2090*ee4e0c1eSBram Moolenaar
20919f6277bdSBram Moolenaar  "let l = ''
20929f6277bdSBram Moolenaar  "for i in range(1, 6)
20939f6277bdSBram Moolenaar  "  let l ..= term_getline(buf, i) . "\n"
20949f6277bdSBram Moolenaar  "endfor
20959f6277bdSBram Moolenaar  "call assert_report(l)
20969f6277bdSBram Moolenaar
20979f6277bdSBram Moolenaar  call StopVimInTerminal(buf)
20989f6277bdSBram Moolenaar  call delete('Xscript')
20999f6277bdSBram Moolenaarendfunc
21009f6277bdSBram Moolenaar
2101bc2b71d4SBram Moolenaar" Test for <sfile>, <slnum> in a function                           {{{1
2102bc2b71d4SBram Moolenaarfunc Test_sfile_in_function()
2103bc2b71d4SBram Moolenaar  func Xfunc()
2104bc2b71d4SBram Moolenaar    call assert_match('..Test_sfile_in_function\[5]..Xfunc', expand('<sfile>'))
2105bc2b71d4SBram Moolenaar    call assert_equal('2', expand('<slnum>'))
2106bc2b71d4SBram Moolenaar  endfunc
2107bc2b71d4SBram Moolenaar  call Xfunc()
2108bc2b71d4SBram Moolenaar  delfunc Xfunc
2109bc2b71d4SBram Moolenaarendfunc
2110bc2b71d4SBram Moolenaar
21118b633135SBram Moolenaar" Test for errors in converting to float from various types         {{{1
21128b633135SBram Moolenaarfunc Test_float_conversion_errors()
21138b633135SBram Moolenaar  if has('float')
21148b633135SBram Moolenaar    call assert_fails('let x = 4.0 % 2.0', 'E804')
21158b633135SBram Moolenaar    call assert_fails('echo 1.1[0]', 'E806')
21168b633135SBram Moolenaar    call assert_fails('echo sort([function("min"), 1], "f")', 'E891:')
21178b633135SBram Moolenaar    call assert_fails('echo 3.2 == "vim"', 'E892:')
21188b633135SBram Moolenaar    call assert_fails('echo sort([[], 1], "f")', 'E893:')
21198b633135SBram Moolenaar    call assert_fails('echo sort([{}, 1], "f")', 'E894:')
21208b633135SBram Moolenaar    call assert_fails('echo 3.2 == v:true', 'E362:')
21218b633135SBram Moolenaar    call assert_fails('echo 3.2 == v:none', 'E907:')
21228b633135SBram Moolenaar  endif
21238b633135SBram Moolenaarendfunc
21248b633135SBram Moolenaar
2125863e80b4SBram Moolenaar"-------------------------------------------------------------------------------
2126b544f3c8SBram Moolenaar" Modelines								    {{{1
21271f068233SBram Moolenaar" vim: ts=8 sw=2 sts=2 expandtab tw=80 fdm=marker
2128b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
2129