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"-------------------------------------------------------------------------------
926b544f3c8SBram Moolenaar" Test 90:  Recognizing {} in variable name.			    {{{1
927b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
928b544f3c8SBram Moolenaar
929b544f3c8SBram Moolenaarfunc Test_curlies()
930b544f3c8SBram Moolenaar    let s:var = 66
931b544f3c8SBram Moolenaar    let ns = 's'
932b544f3c8SBram Moolenaar    call assert_equal(66, {ns}:var)
933b544f3c8SBram Moolenaar
934b544f3c8SBram Moolenaar    let g:a = {}
935b544f3c8SBram Moolenaar    let g:b = 't'
936b544f3c8SBram Moolenaar    let g:a[g:b] = 77
937b544f3c8SBram Moolenaar    call assert_equal(77, g:a['t'])
938b544f3c8SBram Moolenaarendfunc
939b544f3c8SBram Moolenaar
940b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
941b544f3c8SBram Moolenaar" Test 91:  using type().					    {{{1
942b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
943b544f3c8SBram Moolenaar
944b544f3c8SBram Moolenaarfunc Test_type()
945b544f3c8SBram Moolenaar    call assert_equal(0, type(0))
946b544f3c8SBram Moolenaar    call assert_equal(1, type(""))
947b544f3c8SBram Moolenaar    call assert_equal(2, type(function("tr")))
948b544f3c8SBram Moolenaar    call assert_equal(2, type(function("tr", [8])))
949b544f3c8SBram Moolenaar    call assert_equal(3, type([]))
950b544f3c8SBram Moolenaar    call assert_equal(4, type({}))
951b544f3c8SBram Moolenaar    call assert_equal(5, type(0.0))
952b544f3c8SBram Moolenaar    call assert_equal(6, type(v:false))
953b544f3c8SBram Moolenaar    call assert_equal(6, type(v:true))
954b544f3c8SBram Moolenaar    call assert_equal(7, type(v:none))
955b544f3c8SBram Moolenaar    call assert_equal(7, type(v:null))
956b544f3c8SBram Moolenaar    call assert_equal(8, v:t_job)
957b544f3c8SBram Moolenaar    call assert_equal(9, v:t_channel)
958b544f3c8SBram Moolenaar    call assert_equal(v:t_number, type(0))
959b544f3c8SBram Moolenaar    call assert_equal(v:t_string, type(""))
960b544f3c8SBram Moolenaar    call assert_equal(v:t_func, type(function("tr")))
961b544f3c8SBram Moolenaar    call assert_equal(v:t_func, type(function("tr", [8])))
962b544f3c8SBram Moolenaar    call assert_equal(v:t_list, type([]))
963b544f3c8SBram Moolenaar    call assert_equal(v:t_dict, type({}))
964b544f3c8SBram Moolenaar    call assert_equal(v:t_float, type(0.0))
965b544f3c8SBram Moolenaar    call assert_equal(v:t_bool, type(v:false))
966b544f3c8SBram Moolenaar    call assert_equal(v:t_bool, type(v:true))
967b544f3c8SBram Moolenaar    call assert_equal(v:t_none, type(v:none))
968b544f3c8SBram Moolenaar    call assert_equal(v:t_none, type(v:null))
969b544f3c8SBram Moolenaar
970b544f3c8SBram Moolenaar
971b544f3c8SBram Moolenaar    call assert_equal(0, 0 + v:false)
972b544f3c8SBram Moolenaar    call assert_equal(1, 0 + v:true)
973b544f3c8SBram Moolenaar    call assert_equal(0, 0 + v:none)
974b544f3c8SBram Moolenaar    call assert_equal(0, 0 + v:null)
975b544f3c8SBram Moolenaar
976b544f3c8SBram Moolenaar    call assert_equal('v:false', '' . v:false)
977b544f3c8SBram Moolenaar    call assert_equal('v:true', '' . v:true)
978b544f3c8SBram Moolenaar    call assert_equal('v:none', '' . v:none)
979b544f3c8SBram Moolenaar    call assert_equal('v:null', '' . v:null)
980b544f3c8SBram Moolenaar
981b544f3c8SBram Moolenaar    call assert_true(v:false == 0)
982b544f3c8SBram Moolenaar    call assert_false(v:false != 0)
983b544f3c8SBram Moolenaar    call assert_true(v:true == 1)
984b544f3c8SBram Moolenaar    call assert_false(v:true != 1)
985b544f3c8SBram Moolenaar    call assert_false(v:true == v:false)
986b544f3c8SBram Moolenaar    call assert_true(v:true != v:false)
987b544f3c8SBram Moolenaar
988b544f3c8SBram Moolenaar    call assert_true(v:null == 0)
989b544f3c8SBram Moolenaar    call assert_false(v:null != 0)
990b544f3c8SBram Moolenaar    call assert_true(v:none == 0)
991b544f3c8SBram Moolenaar    call assert_false(v:none != 0)
992b544f3c8SBram Moolenaar
993b544f3c8SBram Moolenaar    call assert_true(v:false is v:false)
994b544f3c8SBram Moolenaar    call assert_true(v:true is v:true)
995b544f3c8SBram Moolenaar    call assert_true(v:none is v:none)
996b544f3c8SBram Moolenaar    call assert_true(v:null is v:null)
997b544f3c8SBram Moolenaar
998b544f3c8SBram Moolenaar    call assert_false(v:false isnot v:false)
999b544f3c8SBram Moolenaar    call assert_false(v:true isnot v:true)
1000b544f3c8SBram Moolenaar    call assert_false(v:none isnot v:none)
1001b544f3c8SBram Moolenaar    call assert_false(v:null isnot v:null)
1002b544f3c8SBram Moolenaar
1003b544f3c8SBram Moolenaar    call assert_false(v:false is 0)
1004b544f3c8SBram Moolenaar    call assert_false(v:true is 1)
1005b544f3c8SBram Moolenaar    call assert_false(v:true is v:false)
1006b544f3c8SBram Moolenaar    call assert_false(v:none is 0)
1007b544f3c8SBram Moolenaar    call assert_false(v:null is 0)
1008b544f3c8SBram Moolenaar    call assert_false(v:null is v:none)
1009b544f3c8SBram Moolenaar
1010b544f3c8SBram Moolenaar    call assert_true(v:false isnot 0)
1011b544f3c8SBram Moolenaar    call assert_true(v:true isnot 1)
1012b544f3c8SBram Moolenaar    call assert_true(v:true isnot v:false)
1013b544f3c8SBram Moolenaar    call assert_true(v:none isnot 0)
1014b544f3c8SBram Moolenaar    call assert_true(v:null isnot 0)
1015b544f3c8SBram Moolenaar    call assert_true(v:null isnot v:none)
1016b544f3c8SBram Moolenaar
1017b544f3c8SBram Moolenaar    call assert_equal(v:false, eval(string(v:false)))
1018b544f3c8SBram Moolenaar    call assert_equal(v:true, eval(string(v:true)))
1019b544f3c8SBram Moolenaar    call assert_equal(v:none, eval(string(v:none)))
1020b544f3c8SBram Moolenaar    call assert_equal(v:null, eval(string(v:null)))
1021b544f3c8SBram Moolenaar
1022b544f3c8SBram Moolenaar    call assert_equal(v:false, copy(v:false))
1023b544f3c8SBram Moolenaar    call assert_equal(v:true, copy(v:true))
1024b544f3c8SBram Moolenaar    call assert_equal(v:none, copy(v:none))
1025b544f3c8SBram Moolenaar    call assert_equal(v:null, copy(v:null))
1026b544f3c8SBram Moolenaar
1027b544f3c8SBram Moolenaar    call assert_equal([v:false], deepcopy([v:false]))
1028b544f3c8SBram Moolenaar    call assert_equal([v:true], deepcopy([v:true]))
1029b544f3c8SBram Moolenaar    call assert_equal([v:none], deepcopy([v:none]))
1030b544f3c8SBram Moolenaar    call assert_equal([v:null], deepcopy([v:null]))
1031b544f3c8SBram Moolenaar
1032b544f3c8SBram Moolenaar    call assert_true(empty(v:false))
1033b544f3c8SBram Moolenaar    call assert_false(empty(v:true))
1034b544f3c8SBram Moolenaar    call assert_true(empty(v:null))
1035b544f3c8SBram Moolenaar    call assert_true(empty(v:none))
1036b544f3c8SBram Moolenaar
1037b544f3c8SBram Moolenaar    func ChangeYourMind()
1038b544f3c8SBram Moolenaar	try
1039b544f3c8SBram Moolenaar	    return v:true
1040b544f3c8SBram Moolenaar	finally
1041b544f3c8SBram Moolenaar	    return 'something else'
1042b544f3c8SBram Moolenaar	endtry
1043b544f3c8SBram Moolenaar    endfunc
1044b544f3c8SBram Moolenaar
1045b544f3c8SBram Moolenaar    call ChangeYourMind()
1046b544f3c8SBram Moolenaarendfunc
1047b544f3c8SBram Moolenaar
1048b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
1049b544f3c8SBram Moolenaar" Test 92:  skipping code					    {{{1
1050b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
1051b544f3c8SBram Moolenaar
1052b544f3c8SBram Moolenaarfunc Test_skip()
1053b544f3c8SBram Moolenaar    let Fn = function('Test_type')
1054b544f3c8SBram Moolenaar    call assert_false(0 && Fn[1])
1055b544f3c8SBram Moolenaar    call assert_false(0 && string(Fn))
1056b544f3c8SBram Moolenaar    call assert_false(0 && len(Fn))
1057b544f3c8SBram Moolenaar    let l = []
1058b544f3c8SBram Moolenaar    call assert_false(0 && l[1])
1059b544f3c8SBram Moolenaar    call assert_false(0 && string(l))
1060b544f3c8SBram Moolenaar    call assert_false(0 && len(l))
1061b544f3c8SBram Moolenaar    let f = 1.0
1062b544f3c8SBram Moolenaar    call assert_false(0 && f[1])
1063b544f3c8SBram Moolenaar    call assert_false(0 && string(f))
1064b544f3c8SBram Moolenaar    call assert_false(0 && len(f))
1065b544f3c8SBram Moolenaar    let sp = v:null
1066b544f3c8SBram Moolenaar    call assert_false(0 && sp[1])
1067b544f3c8SBram Moolenaar    call assert_false(0 && string(sp))
1068b544f3c8SBram Moolenaar    call assert_false(0 && len(sp))
1069b544f3c8SBram Moolenaar
1070b544f3c8SBram Moolenaarendfunc
1071b544f3c8SBram Moolenaar
1072b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
1073b544f3c8SBram Moolenaar" Test 93:  :echo and string()					    {{{1
1074b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
1075b544f3c8SBram Moolenaar
1076b544f3c8SBram Moolenaarfunc Test_echo_and_string()
1077b544f3c8SBram Moolenaar    " String
1078b544f3c8SBram Moolenaar    let a = 'foo bar'
1079b544f3c8SBram Moolenaar    redir => result
1080b544f3c8SBram Moolenaar    echo a
1081b544f3c8SBram Moolenaar    echo string(a)
1082b544f3c8SBram Moolenaar    redir END
1083b544f3c8SBram Moolenaar    let l = split(result, "\n")
1084b544f3c8SBram Moolenaar    call assert_equal(["foo bar",
1085b544f3c8SBram Moolenaar		     \ "'foo bar'"], l)
1086b544f3c8SBram Moolenaar
1087b544f3c8SBram Moolenaar    " Float
1088b544f3c8SBram Moolenaar    if has('float')
1089b544f3c8SBram Moolenaar	let a = -1.2e0
1090b544f3c8SBram Moolenaar	redir => result
1091b544f3c8SBram Moolenaar	echo a
1092b544f3c8SBram Moolenaar	echo string(a)
1093b544f3c8SBram Moolenaar	redir END
1094b544f3c8SBram Moolenaar	let l = split(result, "\n")
1095b544f3c8SBram Moolenaar	call assert_equal(["-1.2",
1096b544f3c8SBram Moolenaar			 \ "-1.2"], l)
1097b544f3c8SBram Moolenaar    endif
1098b544f3c8SBram Moolenaar
1099b544f3c8SBram Moolenaar    " Funcref
1100b544f3c8SBram Moolenaar    redir => result
1101b544f3c8SBram Moolenaar    echo function('string')
1102b544f3c8SBram Moolenaar    echo string(function('string'))
1103b544f3c8SBram Moolenaar    redir END
1104b544f3c8SBram Moolenaar    let l = split(result, "\n")
1105b544f3c8SBram Moolenaar    call assert_equal(["string",
1106b544f3c8SBram Moolenaar		     \ "function('string')"], l)
1107b544f3c8SBram Moolenaar
1108b544f3c8SBram Moolenaar    " Recursive dictionary
1109b544f3c8SBram Moolenaar    let a = {}
1110b544f3c8SBram Moolenaar    let a["a"] = a
1111b544f3c8SBram Moolenaar    redir => result
1112b544f3c8SBram Moolenaar    echo a
1113b544f3c8SBram Moolenaar    echo string(a)
1114b544f3c8SBram Moolenaar    redir END
1115b544f3c8SBram Moolenaar    let l = split(result, "\n")
1116b544f3c8SBram Moolenaar    call assert_equal(["{'a': {...}}",
1117b544f3c8SBram Moolenaar		     \ "{'a': {...}}"], l)
1118b544f3c8SBram Moolenaar
1119b544f3c8SBram Moolenaar    " Recursive list
1120b544f3c8SBram Moolenaar    let a = [0]
1121b544f3c8SBram Moolenaar    let a[0] = a
1122b544f3c8SBram Moolenaar    redir => result
1123b544f3c8SBram Moolenaar    echo a
1124b544f3c8SBram Moolenaar    echo string(a)
1125b544f3c8SBram Moolenaar    redir END
1126b544f3c8SBram Moolenaar    let l = split(result, "\n")
1127b544f3c8SBram Moolenaar    call assert_equal(["[[...]]",
1128b544f3c8SBram Moolenaar		     \ "[[...]]"], l)
1129b544f3c8SBram Moolenaar
1130b544f3c8SBram Moolenaar    " Empty dictionaries in a list
1131b544f3c8SBram Moolenaar    let a = {}
1132b544f3c8SBram Moolenaar    redir => result
1133b544f3c8SBram Moolenaar    echo [a, a, a]
1134b544f3c8SBram Moolenaar    echo string([a, a, a])
1135b544f3c8SBram Moolenaar    redir END
1136b544f3c8SBram Moolenaar    let l = split(result, "\n")
1137b544f3c8SBram Moolenaar    call assert_equal(["[{}, {}, {}]",
1138b544f3c8SBram Moolenaar		     \ "[{}, {}, {}]"], l)
1139b544f3c8SBram Moolenaar
1140b544f3c8SBram Moolenaar    " Empty dictionaries in a dictionary
1141b544f3c8SBram Moolenaar    let a = {}
1142b544f3c8SBram Moolenaar    let b = {"a": a, "b": a}
1143b544f3c8SBram Moolenaar    redir => result
1144b544f3c8SBram Moolenaar    echo b
1145b544f3c8SBram Moolenaar    echo string(b)
1146b544f3c8SBram Moolenaar    redir END
1147b544f3c8SBram Moolenaar    let l = split(result, "\n")
1148b544f3c8SBram Moolenaar    call assert_equal(["{'a': {}, 'b': {}}",
1149b544f3c8SBram Moolenaar		     \ "{'a': {}, 'b': {}}"], l)
1150b544f3c8SBram Moolenaar
1151b544f3c8SBram Moolenaar    " Empty lists in a list
1152b544f3c8SBram Moolenaar    let a = []
1153b544f3c8SBram Moolenaar    redir => result
1154b544f3c8SBram Moolenaar    echo [a, a, a]
1155b544f3c8SBram Moolenaar    echo string([a, a, a])
1156b544f3c8SBram Moolenaar    redir END
1157b544f3c8SBram Moolenaar    let l = split(result, "\n")
1158b544f3c8SBram Moolenaar    call assert_equal(["[[], [], []]",
1159b544f3c8SBram Moolenaar		     \ "[[], [], []]"], l)
1160b544f3c8SBram Moolenaar
1161b544f3c8SBram Moolenaar    " Empty lists in a dictionary
1162b544f3c8SBram Moolenaar    let a = []
1163b544f3c8SBram Moolenaar    let b = {"a": a, "b": a}
1164b544f3c8SBram Moolenaar    redir => result
1165b544f3c8SBram Moolenaar    echo b
1166b544f3c8SBram Moolenaar    echo string(b)
1167b544f3c8SBram Moolenaar    redir END
1168b544f3c8SBram Moolenaar    let l = split(result, "\n")
1169b544f3c8SBram Moolenaar    call assert_equal(["{'a': [], 'b': []}",
1170b544f3c8SBram Moolenaar		     \ "{'a': [], 'b': []}"], l)
1171b544f3c8SBram Moolenaar
1172b544f3c8SBram Moolenaar    " Dictionaries in a list
1173b544f3c8SBram Moolenaar    let a = {"one": "yes", "two": "yes", "three": "yes"}
1174b544f3c8SBram Moolenaar    redir => result
1175b544f3c8SBram Moolenaar    echo [a, a, a]
1176b544f3c8SBram Moolenaar    echo string([a, a, a])
1177b544f3c8SBram Moolenaar    redir END
1178b544f3c8SBram Moolenaar    let l = split(result, "\n")
1179b544f3c8SBram Moolenaar    call assert_equal(["[{'one': 'yes', 'two': 'yes', 'three': 'yes'}, {...}, {...}]",
1180b544f3c8SBram Moolenaar		     \ "[{'one': 'yes', 'two': 'yes', 'three': 'yes'}, {'one': 'yes', 'two': 'yes', 'three': 'yes'}, {'one': 'yes', 'two': 'yes', 'three': 'yes'}]"], l)
1181b544f3c8SBram Moolenaar
1182b544f3c8SBram Moolenaar    " Dictionaries in a dictionary
1183b544f3c8SBram Moolenaar    let a = {"one": "yes", "two": "yes", "three": "yes"}
1184b544f3c8SBram Moolenaar    let b = {"a": a, "b": a}
1185b544f3c8SBram Moolenaar    redir => result
1186b544f3c8SBram Moolenaar    echo b
1187b544f3c8SBram Moolenaar    echo string(b)
1188b544f3c8SBram Moolenaar    redir END
1189b544f3c8SBram Moolenaar    let l = split(result, "\n")
1190b544f3c8SBram Moolenaar    call assert_equal(["{'a': {'one': 'yes', 'two': 'yes', 'three': 'yes'}, 'b': {...}}",
1191b544f3c8SBram Moolenaar		     \ "{'a': {'one': 'yes', 'two': 'yes', 'three': 'yes'}, 'b': {'one': 'yes', 'two': 'yes', 'three': 'yes'}}"], l)
1192b544f3c8SBram Moolenaar
1193b544f3c8SBram Moolenaar    " Lists in a list
1194b544f3c8SBram Moolenaar    let a = [1, 2, 3]
1195b544f3c8SBram Moolenaar    redir => result
1196b544f3c8SBram Moolenaar    echo [a, a, a]
1197b544f3c8SBram Moolenaar    echo string([a, a, a])
1198b544f3c8SBram Moolenaar    redir END
1199b544f3c8SBram Moolenaar    let l = split(result, "\n")
1200b544f3c8SBram Moolenaar    call assert_equal(["[[1, 2, 3], [...], [...]]",
1201b544f3c8SBram Moolenaar		     \ "[[1, 2, 3], [1, 2, 3], [1, 2, 3]]"], l)
1202b544f3c8SBram Moolenaar
1203b544f3c8SBram Moolenaar    " Lists in a dictionary
1204b544f3c8SBram Moolenaar    let a = [1, 2, 3]
1205b544f3c8SBram Moolenaar    let b = {"a": a, "b": a}
1206b544f3c8SBram Moolenaar    redir => result
1207b544f3c8SBram Moolenaar    echo b
1208b544f3c8SBram Moolenaar    echo string(b)
1209b544f3c8SBram Moolenaar    redir END
1210b544f3c8SBram Moolenaar    let l = split(result, "\n")
1211b544f3c8SBram Moolenaar    call assert_equal(["{'a': [1, 2, 3], 'b': [...]}",
1212b544f3c8SBram Moolenaar		     \ "{'a': [1, 2, 3], 'b': [1, 2, 3]}"], l)
1213b544f3c8SBram Moolenaar
1214b544f3c8SBram Moolenaarendfunc
1215b544f3c8SBram Moolenaar
1216b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
1217b544f3c8SBram Moolenaar" Test 94:  64-bit Numbers					    {{{1
1218b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
1219b544f3c8SBram Moolenaar
1220b544f3c8SBram Moolenaarfunc Test_num64()
1221b544f3c8SBram Moolenaar    if !has('num64')
1222b544f3c8SBram Moolenaar	return
1223b544f3c8SBram Moolenaar    endif
1224b544f3c8SBram Moolenaar
1225b544f3c8SBram Moolenaar    call assert_notequal( 4294967296, 0)
1226b544f3c8SBram Moolenaar    call assert_notequal(-4294967296, 0)
1227b544f3c8SBram Moolenaar    call assert_equal( 4294967296,  0xFFFFffff + 1)
1228b544f3c8SBram Moolenaar    call assert_equal(-4294967296, -0xFFFFffff - 1)
1229b544f3c8SBram Moolenaar
1230b544f3c8SBram Moolenaar    call assert_equal( 9223372036854775807,  1 / 0)
1231b544f3c8SBram Moolenaar    call assert_equal(-9223372036854775807, -1 / 0)
1232b544f3c8SBram Moolenaar    call assert_equal(-9223372036854775807 - 1,  0 / 0)
1233b544f3c8SBram Moolenaar
1234b544f3c8SBram Moolenaar    call assert_equal( 0x7FFFffffFFFFffff, float2nr( 1.0e150))
1235b544f3c8SBram Moolenaar    call assert_equal(-0x7FFFffffFFFFffff, float2nr(-1.0e150))
1236b544f3c8SBram Moolenaar
1237b544f3c8SBram Moolenaar    let rng = range(0xFFFFffff, 0x100000001)
1238b544f3c8SBram Moolenaar    call assert_equal([0xFFFFffff, 0x100000000, 0x100000001], rng)
1239b544f3c8SBram Moolenaar    call assert_equal(0x100000001, max(rng))
1240b544f3c8SBram Moolenaar    call assert_equal(0xFFFFffff, min(rng))
1241b544f3c8SBram Moolenaar    call assert_equal(rng, sort(range(0x100000001, 0xFFFFffff, -1), 'N'))
1242b544f3c8SBram Moolenaarendfunc
1243b544f3c8SBram Moolenaar
1244b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
1245b544f3c8SBram Moolenaar" Test 95:  lines of :append, :change, :insert			    {{{1
1246b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
1247b544f3c8SBram Moolenaar
1248b544f3c8SBram Moolenaarfunction! DefineFunction(name, body)
1249b544f3c8SBram Moolenaar    let func = join(['function! ' . a:name . '()'] + a:body + ['endfunction'], "\n")
1250b544f3c8SBram Moolenaar    exec func
1251b544f3c8SBram Moolenaarendfunction
1252b544f3c8SBram Moolenaar
1253b544f3c8SBram Moolenaarfunc Test_script_lines()
1254b544f3c8SBram Moolenaar    " :append
1255b544f3c8SBram Moolenaar    try
1256b544f3c8SBram Moolenaar	call DefineFunction('T_Append', [
1257b544f3c8SBram Moolenaar		    \ 'append',
1258b544f3c8SBram Moolenaar		    \ 'py <<EOS',
1259b544f3c8SBram Moolenaar		    \ '.',
1260b544f3c8SBram Moolenaar		    \ ])
1261b544f3c8SBram Moolenaar    catch
126237175409SBram Moolenaar	call assert_report("Can't define function")
1263b544f3c8SBram Moolenaar    endtry
1264b544f3c8SBram Moolenaar    try
1265b544f3c8SBram Moolenaar	call DefineFunction('T_Append', [
1266b544f3c8SBram Moolenaar		    \ 'append',
1267b544f3c8SBram Moolenaar		    \ 'abc',
1268b544f3c8SBram Moolenaar		    \ ])
126937175409SBram Moolenaar	call assert_report("Shouldn't be able to define function")
1270b544f3c8SBram Moolenaar    catch
1271b544f3c8SBram Moolenaar	call assert_exception('Vim(function):E126: Missing :endfunction')
1272b544f3c8SBram Moolenaar    endtry
1273b544f3c8SBram Moolenaar
1274b544f3c8SBram Moolenaar    " :change
1275b544f3c8SBram Moolenaar    try
1276b544f3c8SBram Moolenaar	call DefineFunction('T_Change', [
1277b544f3c8SBram Moolenaar		    \ 'change',
1278b544f3c8SBram Moolenaar		    \ 'py <<EOS',
1279b544f3c8SBram Moolenaar		    \ '.',
1280b544f3c8SBram Moolenaar		    \ ])
1281b544f3c8SBram Moolenaar    catch
128237175409SBram Moolenaar	call assert_report("Can't define function")
1283b544f3c8SBram Moolenaar    endtry
1284b544f3c8SBram Moolenaar    try
1285b544f3c8SBram Moolenaar	call DefineFunction('T_Change', [
1286b544f3c8SBram Moolenaar		    \ 'change',
1287b544f3c8SBram Moolenaar		    \ 'abc',
1288b544f3c8SBram Moolenaar		    \ ])
128937175409SBram Moolenaar	call assert_report("Shouldn't be able to define function")
1290b544f3c8SBram Moolenaar    catch
1291b544f3c8SBram Moolenaar	call assert_exception('Vim(function):E126: Missing :endfunction')
1292b544f3c8SBram Moolenaar    endtry
1293b544f3c8SBram Moolenaar
1294b544f3c8SBram Moolenaar    " :insert
1295b544f3c8SBram Moolenaar    try
1296b544f3c8SBram Moolenaar	call DefineFunction('T_Insert', [
1297b544f3c8SBram Moolenaar		    \ 'insert',
1298b544f3c8SBram Moolenaar		    \ 'py <<EOS',
1299b544f3c8SBram Moolenaar		    \ '.',
1300b544f3c8SBram Moolenaar		    \ ])
1301b544f3c8SBram Moolenaar    catch
130237175409SBram Moolenaar	call assert_report("Can't define function")
1303b544f3c8SBram Moolenaar    endtry
1304b544f3c8SBram Moolenaar    try
1305b544f3c8SBram Moolenaar	call DefineFunction('T_Insert', [
1306b544f3c8SBram Moolenaar		    \ 'insert',
1307b544f3c8SBram Moolenaar		    \ 'abc',
1308b544f3c8SBram Moolenaar		    \ ])
130937175409SBram Moolenaar	call assert_report("Shouldn't be able to define function")
1310b544f3c8SBram Moolenaar    catch
1311b544f3c8SBram Moolenaar	call assert_exception('Vim(function):E126: Missing :endfunction')
1312b544f3c8SBram Moolenaar    endtry
1313b544f3c8SBram Moolenaarendfunc
1314b544f3c8SBram Moolenaar
1315b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
1316478af67dSBram Moolenaar" Test 96:  line continuation						    {{{1
1317478af67dSBram Moolenaar"
1318478af67dSBram Moolenaar"	    Undefined behavior was detected by ubsan with line continuation
1319478af67dSBram Moolenaar"	    after an empty line.
1320478af67dSBram Moolenaar"-------------------------------------------------------------------------------
1321478af67dSBram Moolenaarfunc Test_script_emty_line_continuation()
1322478af67dSBram Moolenaar
1323478af67dSBram Moolenaar    \
1324478af67dSBram Moolenaarendfunc
1325478af67dSBram Moolenaar
1326478af67dSBram Moolenaar"-------------------------------------------------------------------------------
1327863e80b4SBram Moolenaar" Test 97:  bitwise functions						    {{{1
1328863e80b4SBram Moolenaar"-------------------------------------------------------------------------------
1329863e80b4SBram Moolenaarfunc Test_bitwise_functions()
1330863e80b4SBram Moolenaar    " and
1331863e80b4SBram Moolenaar    call assert_equal(127, and(127, 127))
1332863e80b4SBram Moolenaar    call assert_equal(16, and(127, 16))
1333073e4b92SBram Moolenaar    eval 127->and(16)->assert_equal(16)
1334863e80b4SBram Moolenaar    call assert_equal(0, and(127, 128))
1335863e80b4SBram Moolenaar    call assert_fails("call and(1.0, 1)", 'E805:')
1336863e80b4SBram Moolenaar    call assert_fails("call and([], 1)", 'E745:')
1337863e80b4SBram Moolenaar    call assert_fails("call and({}, 1)", 'E728:')
1338863e80b4SBram Moolenaar    call assert_fails("call and(1, 1.0)", 'E805:')
1339863e80b4SBram Moolenaar    call assert_fails("call and(1, [])", 'E745:')
1340863e80b4SBram Moolenaar    call assert_fails("call and(1, {})", 'E728:')
1341863e80b4SBram Moolenaar    " or
1342863e80b4SBram Moolenaar    call assert_equal(23, or(16, 7))
1343863e80b4SBram Moolenaar    call assert_equal(15, or(8, 7))
1344073e4b92SBram Moolenaar    eval 8->or(7)->assert_equal(15)
1345863e80b4SBram Moolenaar    call assert_equal(123, or(0, 123))
1346863e80b4SBram Moolenaar    call assert_fails("call or(1.0, 1)", 'E805:')
1347863e80b4SBram Moolenaar    call assert_fails("call or([], 1)", 'E745:')
1348863e80b4SBram Moolenaar    call assert_fails("call or({}, 1)", 'E728:')
1349863e80b4SBram Moolenaar    call assert_fails("call or(1, 1.0)", 'E805:')
1350863e80b4SBram Moolenaar    call assert_fails("call or(1, [])", 'E745:')
1351863e80b4SBram Moolenaar    call assert_fails("call or(1, {})", 'E728:')
1352863e80b4SBram Moolenaar    " xor
1353863e80b4SBram Moolenaar    call assert_equal(0, xor(127, 127))
1354863e80b4SBram Moolenaar    call assert_equal(111, xor(127, 16))
1355073e4b92SBram Moolenaar    eval 127->xor(16)->assert_equal(111)
1356863e80b4SBram Moolenaar    call assert_equal(255, xor(127, 128))
1357863e80b4SBram Moolenaar    call assert_fails("call xor(1.0, 1)", 'E805:')
1358863e80b4SBram Moolenaar    call assert_fails("call xor([], 1)", 'E745:')
1359863e80b4SBram Moolenaar    call assert_fails("call xor({}, 1)", 'E728:')
1360863e80b4SBram Moolenaar    call assert_fails("call xor(1, 1.0)", 'E805:')
1361863e80b4SBram Moolenaar    call assert_fails("call xor(1, [])", 'E745:')
1362863e80b4SBram Moolenaar    call assert_fails("call xor(1, {})", 'E728:')
1363863e80b4SBram Moolenaar    " invert
1364863e80b4SBram Moolenaar    call assert_equal(65408, and(invert(127), 65535))
1365073e4b92SBram Moolenaar    eval 127->invert()->and(65535)->assert_equal(65408)
1366863e80b4SBram Moolenaar    call assert_equal(65519, and(invert(16), 65535))
1367863e80b4SBram Moolenaar    call assert_equal(65407, and(invert(128), 65535))
1368863e80b4SBram Moolenaar    call assert_fails("call invert(1.0)", 'E805:')
1369863e80b4SBram Moolenaar    call assert_fails("call invert([])", 'E745:')
1370863e80b4SBram Moolenaar    call assert_fails("call invert({})", 'E728:')
1371863e80b4SBram Moolenaarendfunc
1372863e80b4SBram Moolenaar
1373663bb233SBram Moolenaar" Test trailing text after :endfunction				    {{{1
1374663bb233SBram Moolenaarfunc Test_endfunction_trailing()
1375663bb233SBram Moolenaar    call assert_false(exists('*Xtest'))
1376663bb233SBram Moolenaar
1377663bb233SBram Moolenaar    exe "func Xtest()\necho 'hello'\nendfunc\nlet done = 'yes'"
1378663bb233SBram Moolenaar    call assert_true(exists('*Xtest'))
1379663bb233SBram Moolenaar    call assert_equal('yes', done)
1380663bb233SBram Moolenaar    delfunc Xtest
1381663bb233SBram Moolenaar    unlet done
1382663bb233SBram Moolenaar
1383663bb233SBram Moolenaar    exe "func Xtest()\necho 'hello'\nendfunc|let done = 'yes'"
1384663bb233SBram Moolenaar    call assert_true(exists('*Xtest'))
1385663bb233SBram Moolenaar    call assert_equal('yes', done)
1386663bb233SBram Moolenaar    delfunc Xtest
1387663bb233SBram Moolenaar    unlet done
1388663bb233SBram Moolenaar
138953564f7cSBram Moolenaar    " trailing line break
139053564f7cSBram Moolenaar    exe "func Xtest()\necho 'hello'\nendfunc\n"
139153564f7cSBram Moolenaar    call assert_true(exists('*Xtest'))
139253564f7cSBram Moolenaar    delfunc Xtest
139353564f7cSBram Moolenaar
1394663bb233SBram Moolenaar    set verbose=1
1395663bb233SBram Moolenaar    exe "func Xtest()\necho 'hello'\nendfunc \" garbage"
1396f8be461dSBram Moolenaar    call assert_notmatch('W22:', split(execute('1messages'), "\n")[0])
1397663bb233SBram Moolenaar    call assert_true(exists('*Xtest'))
1398663bb233SBram Moolenaar    delfunc Xtest
1399663bb233SBram Moolenaar
1400f8be461dSBram Moolenaar    exe "func Xtest()\necho 'hello'\nendfunc garbage"
1401f8be461dSBram Moolenaar    call assert_match('W22:', split(execute('1messages'), "\n")[0])
1402663bb233SBram Moolenaar    call assert_true(exists('*Xtest'))
1403663bb233SBram Moolenaar    delfunc Xtest
1404663bb233SBram Moolenaar    set verbose=0
140553564f7cSBram Moolenaar
140653564f7cSBram Moolenaar    function Foo()
140753564f7cSBram Moolenaar	echo 'hello'
140853564f7cSBram Moolenaar    endfunction | echo 'xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx'
140953564f7cSBram Moolenaar    delfunc Foo
1410663bb233SBram Moolenaarendfunc
1411663bb233SBram Moolenaar
1412d6abcd15SBram Moolenaarfunc Test_delfunction_force()
1413d6abcd15SBram Moolenaar    delfunc! Xtest
1414d6abcd15SBram Moolenaar    delfunc! Xtest
1415d6abcd15SBram Moolenaar    func Xtest()
1416d6abcd15SBram Moolenaar	echo 'nothing'
1417d6abcd15SBram Moolenaar    endfunc
1418d6abcd15SBram Moolenaar    delfunc! Xtest
1419d6abcd15SBram Moolenaar    delfunc! Xtest
1420d6abcd15SBram Moolenaarendfunc
1421d6abcd15SBram Moolenaar
14226f9a476bSBram Moolenaar" Test using bang after user command				    {{{1
14236f9a476bSBram Moolenaarfunc Test_user_command_with_bang()
14246f9a476bSBram Moolenaar    command -bang Nieuw let nieuw = 1
14256f9a476bSBram Moolenaar    Ni!
14266f9a476bSBram Moolenaar    call assert_equal(1, nieuw)
14276f9a476bSBram Moolenaar    unlet nieuw
14286f9a476bSBram Moolenaar    delcommand Nieuw
14296f9a476bSBram Moolenaarendfunc
14306f9a476bSBram Moolenaar
1431bb3e6416SBram Moolenaar" Test for script-local function
1432bb3e6416SBram Moolenaarfunc <SID>DoLast()
1433bb3e6416SBram Moolenaar  call append(line('$'), "last line")
1434bb3e6416SBram Moolenaarendfunc
1435bb3e6416SBram Moolenaar
1436bb3e6416SBram Moolenaarfunc s:DoNothing()
1437bb3e6416SBram Moolenaar  call append(line('$'), "nothing line")
1438bb3e6416SBram Moolenaarendfunc
1439bb3e6416SBram Moolenaar
1440bb3e6416SBram Moolenaarfunc Test_script_local_func()
1441bb3e6416SBram Moolenaar  set nocp viminfo+=nviminfo
1442bb3e6416SBram Moolenaar  new
1443bb3e6416SBram Moolenaar  nnoremap <buffer> _x	:call <SID>DoNothing()<bar>call <SID>DoLast()<bar>delfunc <SID>DoNothing<bar>delfunc <SID>DoLast<cr>
1444bb3e6416SBram Moolenaar
1445bb3e6416SBram Moolenaar  normal _x
1446bb3e6416SBram Moolenaar  call assert_equal('nothing line', getline(2))
1447bb3e6416SBram Moolenaar  call assert_equal('last line', getline(3))
1448bb3e6416SBram Moolenaar  enew! | close
1449bb3e6416SBram Moolenaarendfunc
1450bb3e6416SBram Moolenaar
1451ff697e6cSBram Moolenaarfunc Test_compound_assignment_operators()
1452ff697e6cSBram Moolenaar    " Test for number
1453ff697e6cSBram Moolenaar    let x = 1
1454ff697e6cSBram Moolenaar    let x += 10
1455ff697e6cSBram Moolenaar    call assert_equal(11, x)
1456ff697e6cSBram Moolenaar    let x -= 5
1457ff697e6cSBram Moolenaar    call assert_equal(6, x)
1458ff697e6cSBram Moolenaar    let x *= 4
1459ff697e6cSBram Moolenaar    call assert_equal(24, x)
1460ff697e6cSBram Moolenaar    let x /= 3
1461ff697e6cSBram Moolenaar    call assert_equal(8, x)
1462ff697e6cSBram Moolenaar    let x %= 3
1463ff697e6cSBram Moolenaar    call assert_equal(2, x)
1464ff697e6cSBram Moolenaar    let x .= 'n'
1465ff697e6cSBram Moolenaar    call assert_equal('2n', x)
1466ff697e6cSBram Moolenaar
1467e21c1580SBram Moolenaar    " Test special cases: division or modulus with 0.
1468e21c1580SBram Moolenaar    let x = 1
1469e21c1580SBram Moolenaar    let x /= 0
1470e21c1580SBram Moolenaar    if has('num64')
1471e21c1580SBram Moolenaar        call assert_equal(0x7FFFFFFFFFFFFFFF, x)
1472e21c1580SBram Moolenaar    else
1473e21c1580SBram Moolenaar        call assert_equal(0x7fffffff, x)
1474e21c1580SBram Moolenaar    endif
1475e21c1580SBram Moolenaar
1476e21c1580SBram Moolenaar    let x = -1
1477e21c1580SBram Moolenaar    let x /= 0
1478e21c1580SBram Moolenaar    if has('num64')
1479e21c1580SBram Moolenaar        call assert_equal(-0x7FFFFFFFFFFFFFFF, x)
1480e21c1580SBram Moolenaar    else
1481e21c1580SBram Moolenaar        call assert_equal(-0x7fffffff, x)
1482e21c1580SBram Moolenaar    endif
1483e21c1580SBram Moolenaar
1484e21c1580SBram Moolenaar    let x = 0
1485e21c1580SBram Moolenaar    let x /= 0
1486e21c1580SBram Moolenaar    if has('num64')
1487e21c1580SBram Moolenaar        call assert_equal(-0x7FFFFFFFFFFFFFFF - 1, x)
1488e21c1580SBram Moolenaar    else
1489e21c1580SBram Moolenaar        call assert_equal(-0x7FFFFFFF - 1, x)
1490e21c1580SBram Moolenaar    endif
1491e21c1580SBram Moolenaar
1492e21c1580SBram Moolenaar    let x = 1
1493e21c1580SBram Moolenaar    let x %= 0
1494e21c1580SBram Moolenaar    call assert_equal(0, x)
1495e21c1580SBram Moolenaar
1496e21c1580SBram Moolenaar    let x = -1
1497e21c1580SBram Moolenaar    let x %= 0
1498e21c1580SBram Moolenaar    call assert_equal(0, x)
1499e21c1580SBram Moolenaar
1500e21c1580SBram Moolenaar    let x = 0
1501e21c1580SBram Moolenaar    let x %= 0
1502e21c1580SBram Moolenaar    call assert_equal(0, x)
1503e21c1580SBram Moolenaar
1504ff697e6cSBram Moolenaar    " Test for string
1505ff697e6cSBram Moolenaar    let x = 'str'
1506ff697e6cSBram Moolenaar    let x .= 'ing'
1507ff697e6cSBram Moolenaar    call assert_equal('string', x)
1508ff697e6cSBram Moolenaar    let x += 1
1509ff697e6cSBram Moolenaar    call assert_equal(1, x)
1510ff697e6cSBram Moolenaar    let x -= 1.5
1511ff697e6cSBram Moolenaar    call assert_equal(-0.5, x)
1512ff697e6cSBram Moolenaar
1513ff697e6cSBram Moolenaar    if has('float')
1514ff697e6cSBram Moolenaar        " Test for float
1515ff697e6cSBram Moolenaar        let x = 0.5
1516ff697e6cSBram Moolenaar        let x += 4.5
1517ff697e6cSBram Moolenaar        call assert_equal(5.0, x)
1518ff697e6cSBram Moolenaar        let x -= 1.5
1519ff697e6cSBram Moolenaar        call assert_equal(3.5, x)
1520ff697e6cSBram Moolenaar        let x *= 3.0
1521ff697e6cSBram Moolenaar        call assert_equal(10.5, x)
1522ff697e6cSBram Moolenaar        let x /= 2.5
1523ff697e6cSBram Moolenaar        call assert_equal(4.2, x)
1524ff697e6cSBram Moolenaar        call assert_fails('let x %= 0.5', 'E734')
1525ff697e6cSBram Moolenaar        call assert_fails('let x .= "f"', 'E734')
1526ff697e6cSBram Moolenaar    endif
1527ff697e6cSBram Moolenaar
1528ff697e6cSBram Moolenaar    " Test for environment variable
1529ff697e6cSBram Moolenaar    let $FOO = 1
1530ff697e6cSBram Moolenaar    call assert_fails('let $FOO += 1', 'E734')
1531ff697e6cSBram Moolenaar    call assert_fails('let $FOO -= 1', 'E734')
1532ff697e6cSBram Moolenaar    call assert_fails('let $FOO *= 1', 'E734')
1533ff697e6cSBram Moolenaar    call assert_fails('let $FOO /= 1', 'E734')
1534ff697e6cSBram Moolenaar    call assert_fails('let $FOO %= 1', 'E734')
1535ff697e6cSBram Moolenaar    let $FOO .= 's'
1536ff697e6cSBram Moolenaar    call assert_equal('1s', $FOO)
1537ff697e6cSBram Moolenaar    unlet $FOO
1538ff697e6cSBram Moolenaar
1539ff697e6cSBram Moolenaar    " Test for option variable (type: number)
1540ff697e6cSBram Moolenaar    let &scrolljump = 1
1541ff697e6cSBram Moolenaar    let &scrolljump += 5
1542ff697e6cSBram Moolenaar    call assert_equal(6, &scrolljump)
1543ff697e6cSBram Moolenaar    let &scrolljump -= 2
1544ff697e6cSBram Moolenaar    call assert_equal(4, &scrolljump)
1545ff697e6cSBram Moolenaar    let &scrolljump *= 3
1546ff697e6cSBram Moolenaar    call assert_equal(12, &scrolljump)
1547ff697e6cSBram Moolenaar    let &scrolljump /= 2
1548ff697e6cSBram Moolenaar    call assert_equal(6, &scrolljump)
1549ff697e6cSBram Moolenaar    let &scrolljump %= 5
1550ff697e6cSBram Moolenaar    call assert_equal(1, &scrolljump)
1551ff697e6cSBram Moolenaar    call assert_fails('let &scrolljump .= "j"', 'E734')
1552ff697e6cSBram Moolenaar    set scrolljump&vim
1553ff697e6cSBram Moolenaar
1554ff697e6cSBram Moolenaar    " Test for register
1555ff697e6cSBram Moolenaar    let @/ = 1
1556ff697e6cSBram Moolenaar    call assert_fails('let @/ += 1', 'E734')
1557ff697e6cSBram Moolenaar    call assert_fails('let @/ -= 1', 'E734')
1558ff697e6cSBram Moolenaar    call assert_fails('let @/ *= 1', 'E734')
1559ff697e6cSBram Moolenaar    call assert_fails('let @/ /= 1', 'E734')
1560ff697e6cSBram Moolenaar    call assert_fails('let @/ %= 1', 'E734')
1561ff697e6cSBram Moolenaar    let @/ .= 's'
1562ff697e6cSBram Moolenaar    call assert_equal('1s', @/)
1563ff697e6cSBram Moolenaar    let @/ = ''
1564ff697e6cSBram Moolenaarendfunc
1565ff697e6cSBram Moolenaar
1566c3e92c16SBram Moolenaarfunc Test_refcount()
1567c3e92c16SBram Moolenaar    " Immediate values
1568c3e92c16SBram Moolenaar    call assert_equal(-1, test_refcount(1))
1569c3e92c16SBram Moolenaar    call assert_equal(-1, test_refcount('s'))
1570c3e92c16SBram Moolenaar    call assert_equal(-1, test_refcount(v:true))
1571c3e92c16SBram Moolenaar    call assert_equal(0, test_refcount([]))
1572c3e92c16SBram Moolenaar    call assert_equal(0, test_refcount({}))
1573c3e92c16SBram Moolenaar    call assert_equal(0, test_refcount(0zff))
1574c3e92c16SBram Moolenaar    call assert_equal(0, test_refcount({-> line('.')}))
1575c3e92c16SBram Moolenaar    if has('float')
1576c3e92c16SBram Moolenaar        call assert_equal(-1, test_refcount(0.1))
1577c3e92c16SBram Moolenaar    endif
1578c3e92c16SBram Moolenaar    if has('job')
1579c3e92c16SBram Moolenaar        call assert_equal(0, test_refcount(job_start([&shell, &shellcmdflag, 'echo .'])))
1580c3e92c16SBram Moolenaar    endif
1581c3e92c16SBram Moolenaar
1582c3e92c16SBram Moolenaar    " No refcount types
1583c3e92c16SBram Moolenaar    let x = 1
1584c3e92c16SBram Moolenaar    call assert_equal(-1, test_refcount(x))
1585c3e92c16SBram Moolenaar    let x = 's'
1586c3e92c16SBram Moolenaar    call assert_equal(-1, test_refcount(x))
1587c3e92c16SBram Moolenaar    let x = v:true
1588c3e92c16SBram Moolenaar    call assert_equal(-1, test_refcount(x))
1589c3e92c16SBram Moolenaar    if has('float')
1590c3e92c16SBram Moolenaar        let x = 0.1
1591c3e92c16SBram Moolenaar        call assert_equal(-1, test_refcount(x))
1592c3e92c16SBram Moolenaar    endif
1593c3e92c16SBram Moolenaar
1594c3e92c16SBram Moolenaar    " Check refcount
1595c3e92c16SBram Moolenaar    let x = []
1596c3e92c16SBram Moolenaar    call assert_equal(1, test_refcount(x))
1597c3e92c16SBram Moolenaar
1598c3e92c16SBram Moolenaar    let x = {}
1599*ce90e36fSBram Moolenaar    call assert_equal(1, x->test_refcount())
1600c3e92c16SBram Moolenaar
1601c3e92c16SBram Moolenaar    let x = 0zff
1602c3e92c16SBram Moolenaar    call assert_equal(1, test_refcount(x))
1603c3e92c16SBram Moolenaar
1604c3e92c16SBram Moolenaar    let X = {-> line('.')}
1605c3e92c16SBram Moolenaar    call assert_equal(1, test_refcount(X))
1606c3e92c16SBram Moolenaar    let Y = X
1607c3e92c16SBram Moolenaar    call assert_equal(2, test_refcount(X))
1608c3e92c16SBram Moolenaar
1609c3e92c16SBram Moolenaar    if has('job')
1610c3e92c16SBram Moolenaar        let job = job_start([&shell, &shellcmdflag, 'echo .'])
1611c3e92c16SBram Moolenaar        call assert_equal(1, test_refcount(job))
1612c3e92c16SBram Moolenaar        call assert_equal(1, test_refcount(job_getchannel(job)))
1613c3e92c16SBram Moolenaar        call assert_equal(1, test_refcount(job))
1614c3e92c16SBram Moolenaar    endif
1615c3e92c16SBram Moolenaar
1616c3e92c16SBram Moolenaar    " Function arguments, copying and unassigning
1617c3e92c16SBram Moolenaar    func ExprCheck(x, i)
1618c3e92c16SBram Moolenaar        let i = a:i + 1
1619c3e92c16SBram Moolenaar        call assert_equal(i, test_refcount(a:x))
1620c3e92c16SBram Moolenaar        let Y = a:x
1621c3e92c16SBram Moolenaar        call assert_equal(i + 1, test_refcount(a:x))
1622c3e92c16SBram Moolenaar        call assert_equal(test_refcount(a:x), test_refcount(Y))
1623c3e92c16SBram Moolenaar        let Y = 0
1624c3e92c16SBram Moolenaar        call assert_equal(i, test_refcount(a:x))
1625c3e92c16SBram Moolenaar    endfunc
1626c3e92c16SBram Moolenaar    call ExprCheck([], 0)
1627c3e92c16SBram Moolenaar    call ExprCheck({}, 0)
1628c3e92c16SBram Moolenaar    call ExprCheck(0zff, 0)
1629c3e92c16SBram Moolenaar    call ExprCheck({-> line('.')}, 0)
1630c3e92c16SBram Moolenaar    if has('job')
1631c3e92c16SBram Moolenaar	call ExprCheck(job, 1)
1632c3e92c16SBram Moolenaar	call ExprCheck(job_getchannel(job), 1)
1633c3e92c16SBram Moolenaar	call job_stop(job)
1634c3e92c16SBram Moolenaar    endif
1635c3e92c16SBram Moolenaar    delfunc ExprCheck
1636c3e92c16SBram Moolenaar
1637c3e92c16SBram Moolenaar    " Regarding function
1638c3e92c16SBram Moolenaar    func Func(x) abort
1639c3e92c16SBram Moolenaar        call assert_equal(2, test_refcount(function('Func')))
1640c3e92c16SBram Moolenaar        call assert_equal(0, test_refcount(funcref('Func')))
1641c3e92c16SBram Moolenaar    endfunc
1642c3e92c16SBram Moolenaar    call assert_equal(1, test_refcount(function('Func')))
1643c3e92c16SBram Moolenaar    call assert_equal(0, test_refcount(function('Func', [1])))
1644c3e92c16SBram Moolenaar    call assert_equal(0, test_refcount(funcref('Func')))
1645c3e92c16SBram Moolenaar    call assert_equal(0, test_refcount(funcref('Func', [1])))
1646c3e92c16SBram Moolenaar    let X = function('Func')
1647c3e92c16SBram Moolenaar    let Y = X
1648c3e92c16SBram Moolenaar    call assert_equal(1, test_refcount(X))
1649c3e92c16SBram Moolenaar    let X = function('Func', [1])
1650c3e92c16SBram Moolenaar    let Y = X
1651c3e92c16SBram Moolenaar    call assert_equal(2, test_refcount(X))
1652c3e92c16SBram Moolenaar    let X = funcref('Func')
1653c3e92c16SBram Moolenaar    let Y = X
1654c3e92c16SBram Moolenaar    call assert_equal(2, test_refcount(X))
1655c3e92c16SBram Moolenaar    let X = funcref('Func', [1])
1656c3e92c16SBram Moolenaar    let Y = X
1657c3e92c16SBram Moolenaar    call assert_equal(2, test_refcount(X))
1658c3e92c16SBram Moolenaar    unlet X
1659c3e92c16SBram Moolenaar    unlet Y
1660c3e92c16SBram Moolenaar    call Func(1)
1661c3e92c16SBram Moolenaar    delfunc Func
1662c3e92c16SBram Moolenaar
1663c3e92c16SBram Moolenaar    " Function with dict
1664c3e92c16SBram Moolenaar    func DictFunc() dict
1665c3e92c16SBram Moolenaar        call assert_equal(3, test_refcount(self))
1666c3e92c16SBram Moolenaar    endfunc
1667c3e92c16SBram Moolenaar    let d = {'Func': function('DictFunc')}
1668c3e92c16SBram Moolenaar    call assert_equal(1, test_refcount(d))
1669c3e92c16SBram Moolenaar    call assert_equal(0, test_refcount(d.Func))
1670c3e92c16SBram Moolenaar    call d.Func()
1671c3e92c16SBram Moolenaar    unlet d
1672c3e92c16SBram Moolenaar    delfunc DictFunc
1673c3e92c16SBram Moolenaarendfunc
1674c3e92c16SBram Moolenaar
16757d491c42SBram Moolenaarfunc Test_funccall_garbage_collect()
16766e5000d4SBram Moolenaar    func Func(x, ...)
16776e5000d4SBram Moolenaar        call add(a:x, a:000)
16786e5000d4SBram Moolenaar    endfunc
16796e5000d4SBram Moolenaar    call Func([], [])
16806e5000d4SBram Moolenaar    " Must not crash cause by invalid freeing
16816e5000d4SBram Moolenaar    call test_garbagecollect_now()
16826e5000d4SBram Moolenaar    call assert_true(v:true)
16836e5000d4SBram Moolenaar    delfunc Func
16846e5000d4SBram Moolenaarendfunc
16856e5000d4SBram Moolenaar
1686bc2cfe46SBram Moolenaarfunc Test_function_defined_line()
16878c5a278fSBram Moolenaar    CheckNotGui
1688bc2cfe46SBram Moolenaar
1689bc2cfe46SBram Moolenaar    let lines =<< trim [CODE]
1690bc2cfe46SBram Moolenaar    " F1
1691bc2cfe46SBram Moolenaar    func F1()
1692bc2cfe46SBram Moolenaar        " F2
1693bc2cfe46SBram Moolenaar        func F2()
1694bc2cfe46SBram Moolenaar            "
1695bc2cfe46SBram Moolenaar            "
1696bc2cfe46SBram Moolenaar            "
1697bc2cfe46SBram Moolenaar            return
1698bc2cfe46SBram Moolenaar        endfunc
1699bc2cfe46SBram Moolenaar        " F3
1700bc2cfe46SBram Moolenaar        execute "func F3()\n\n\n\nreturn\nendfunc"
1701bc2cfe46SBram Moolenaar        " F4
1702bc2cfe46SBram Moolenaar        execute "func F4()\n
1703bc2cfe46SBram Moolenaar                    \\n
1704bc2cfe46SBram Moolenaar                    \\n
1705bc2cfe46SBram Moolenaar                    \\n
1706bc2cfe46SBram Moolenaar                    \return\n
1707bc2cfe46SBram Moolenaar                    \endfunc"
1708bc2cfe46SBram Moolenaar    endfunc
1709bc2cfe46SBram Moolenaar    " F5
1710bc2cfe46SBram Moolenaar    execute "func F5()\n\n\n\nreturn\nendfunc"
1711bc2cfe46SBram Moolenaar    " F6
1712bc2cfe46SBram Moolenaar    execute "func F6()\n
1713bc2cfe46SBram Moolenaar                \\n
1714bc2cfe46SBram Moolenaar                \\n
1715bc2cfe46SBram Moolenaar                \\n
1716bc2cfe46SBram Moolenaar                \return\n
1717bc2cfe46SBram Moolenaar                \endfunc"
1718bc2cfe46SBram Moolenaar    call F1()
1719bc2cfe46SBram Moolenaar    verbose func F1
1720bc2cfe46SBram Moolenaar    verbose func F2
1721bc2cfe46SBram Moolenaar    verbose func F3
1722bc2cfe46SBram Moolenaar    verbose func F4
1723bc2cfe46SBram Moolenaar    verbose func F5
1724bc2cfe46SBram Moolenaar    verbose func F6
1725bc2cfe46SBram Moolenaar    qall!
1726bc2cfe46SBram Moolenaar    [CODE]
1727bc2cfe46SBram Moolenaar
1728bc2cfe46SBram Moolenaar    call writefile(lines, 'Xtest.vim')
172993344c2dSBram Moolenaar    let res = system(GetVimCommandClean() .. ' -es -X -S Xtest.vim')
1730bc2cfe46SBram Moolenaar    call assert_equal(0, v:shell_error)
1731bc2cfe46SBram Moolenaar
1732bc2cfe46SBram Moolenaar    let m = matchstr(res, 'function F1()[^[:print:]]*[[:print:]]*')
1733bc2cfe46SBram Moolenaar    call assert_match(' line 2$', m)
1734bc2cfe46SBram Moolenaar
1735bc2cfe46SBram Moolenaar    let m = matchstr(res, 'function F2()[^[:print:]]*[[:print:]]*')
1736bc2cfe46SBram Moolenaar    call assert_match(' line 4$', m)
1737bc2cfe46SBram Moolenaar
1738bc2cfe46SBram Moolenaar    let m = matchstr(res, 'function F3()[^[:print:]]*[[:print:]]*')
1739bc2cfe46SBram Moolenaar    call assert_match(' line 11$', m)
1740bc2cfe46SBram Moolenaar
1741bc2cfe46SBram Moolenaar    let m = matchstr(res, 'function F4()[^[:print:]]*[[:print:]]*')
1742bc2cfe46SBram Moolenaar    call assert_match(' line 13$', m)
1743bc2cfe46SBram Moolenaar
1744bc2cfe46SBram Moolenaar    let m = matchstr(res, 'function F5()[^[:print:]]*[[:print:]]*')
1745bc2cfe46SBram Moolenaar    call assert_match(' line 21$', m)
1746bc2cfe46SBram Moolenaar
1747bc2cfe46SBram Moolenaar    let m = matchstr(res, 'function F6()[^[:print:]]*[[:print:]]*')
1748bc2cfe46SBram Moolenaar    call assert_match(' line 23$', m)
1749bc2cfe46SBram Moolenaar
1750bc2cfe46SBram Moolenaar    call delete('Xtest.vim')
1751bc2cfe46SBram Moolenaarendfunc
1752bc2cfe46SBram Moolenaar
1753863e80b4SBram Moolenaar"-------------------------------------------------------------------------------
1754b544f3c8SBram Moolenaar" Modelines								    {{{1
1755b544f3c8SBram Moolenaar" vim: ts=8 sw=4 tw=80 fdm=marker
1756b544f3c8SBram Moolenaar"-------------------------------------------------------------------------------
1757