1" Tests for the List and Dict types 2 3func TearDown() 4 " Run garbage collection after every test 5 call test_garbagecollect_now() 6endfunc 7 8" Tests for List type 9 10" List creation 11func Test_list_create() 12 " Creating List directly with different types 13 let l = [1, 'as''d', [1, 2, function("strlen")], {'a': 1},] 14 call assert_equal("[1, 'as''d', [1, 2, function('strlen')], {'a': 1}]", string(l)) 15 call assert_equal({'a' : 1}, l[-1]) 16 call assert_equal(1, l[-4]) 17 let x = 10 18 try 19 let x = l[-5] 20 catch 21 call assert_match('E684:', v:exception) 22 endtry 23 call assert_equal(10, x) 24endfunc 25 26" This was allowed in legacy Vim script 27let s:list_with_spaces = [1 , 2 , 3] 28 29" List slices 30func Test_list_slice() 31 let l = [1, 'as''d', [1, 2, function("strlen")], {'a': 1},] 32 call assert_equal([1, 'as''d', [1, 2, function('strlen')], {'a': 1}], l[:]) 33 call assert_equal(['as''d', [1, 2, function('strlen')], {'a': 1}], l[1:]) 34 call assert_equal([1, 'as''d', [1, 2, function('strlen')]], l[:-2]) 35 call assert_equal([1, 'as''d', [1, 2, function('strlen')], {'a': 1}], l[0:8]) 36 call assert_equal([], l[8:-1]) 37 call assert_equal([], l[0:-10]) 38 " perform an operation on a list slice 39 let l = [1, 2, 3] 40 let l[:1] += [1, 2] 41 let l[2:] -= [1] 42 call assert_equal([2, 4, 2], l) 43endfunc 44 45" List identity 46func Test_list_identity() 47 let l = [1, 'as''d', [1, 2, function("strlen")], {'a': 1},] 48 let ll = l 49 let lx = copy(l) 50 call assert_true(l == ll) 51 call assert_false(l isnot ll) 52 call assert_true(l is ll) 53 call assert_true(l == lx) 54 call assert_false(l is lx) 55 call assert_true(l isnot lx) 56endfunc 57 58" removing items with :unlet 59func Test_list_unlet() 60 let l = [1, 'as''d', [1, 2, function("strlen")], {'a': 1},] 61 unlet l[2] 62 call assert_equal([1, 'as''d', {'a': 1}], l) 63 let l = range(8) 64 unlet l[:3] 65 unlet l[1:] 66 call assert_equal([4], l) 67 68 " removing items out of range: silently skip items that don't exist 69 let l = [0, 1, 2, 3] 70 call assert_fails('unlet l[2:1]', 'E684') 71 let l = [0, 1, 2, 3] 72 unlet l[2:2] 73 call assert_equal([0, 1, 3], l) 74 let l = [0, 1, 2, 3] 75 unlet l[2:3] 76 call assert_equal([0, 1], l) 77 let l = [0, 1, 2, 3] 78 unlet l[2:4] 79 call assert_equal([0, 1], l) 80 let l = [0, 1, 2, 3] 81 unlet l[2:5] 82 call assert_equal([0, 1], l) 83 let l = [0, 1, 2, 3] 84 call assert_fails('unlet l[-1:2]', 'E684') 85 let l = [0, 1, 2, 3] 86 unlet l[-2:2] 87 call assert_equal([0, 1, 3], l) 88 let l = [0, 1, 2, 3] 89 unlet l[-3:2] 90 call assert_equal([0, 3], l) 91 let l = [0, 1, 2, 3] 92 unlet l[-4:2] 93 call assert_equal([3], l) 94 let l = [0, 1, 2, 3] 95 unlet l[-5:2] 96 call assert_equal([3], l) 97 let l = [0, 1, 2, 3] 98 unlet l[-6:2] 99 call assert_equal([3], l) 100endfunc 101 102" assignment to a list 103func Test_list_assign() 104 let l = [0, 1, 2, 3] 105 let [va, vb] = l[2:3] 106 call assert_equal([2, 3], [va, vb]) 107 call assert_fails('let [va, vb] = l', 'E687') 108 call assert_fails('let [va, vb] = l[1:1]', 'E688') 109endfunc 110 111" test for range assign 112func Test_list_range_assign() 113 let l = [0] 114 let l[:] = [1, 2] 115 call assert_equal([1, 2], l) 116 let l[-4:-1] = [5, 6] 117 call assert_equal([5, 6], l) 118endfunc 119 120" Test removing items in list 121func Test_list_func_remove() 122 " Test removing 1 element 123 let l = [1, 2, 3, 4] 124 call assert_equal(1, remove(l, 0)) 125 call assert_equal([2, 3, 4], l) 126 127 let l = [1, 2, 3, 4] 128 call assert_equal(2, remove(l, 1)) 129 call assert_equal([1, 3, 4], l) 130 131 let l = [1, 2, 3, 4] 132 call assert_equal(4, remove(l, -1)) 133 call assert_equal([1, 2, 3], l) 134 135 " Test removing range of element(s) 136 let l = [1, 2, 3, 4] 137 call assert_equal([3], remove(l, 2, 2)) 138 call assert_equal([1, 2, 4], l) 139 140 let l = [1, 2, 3, 4] 141 call assert_equal([2, 3], remove(l, 1, 2)) 142 call assert_equal([1, 4], l) 143 144 let l = [1, 2, 3, 4] 145 call assert_equal([2, 3], remove(l, -3, -2)) 146 call assert_equal([1, 4], l) 147 148 " Test invalid cases 149 let l = [1, 2, 3, 4] 150 call assert_fails("call remove(l, 5)", 'E684:') 151 call assert_fails("call remove(l, 1, 5)", 'E684:') 152 call assert_fails("call remove(l, 3, 2)", 'E16:') 153 call assert_fails("call remove(1, 0)", 'E896:') 154 call assert_fails("call remove(l, l)", 'E745:') 155endfunc 156 157" List add() function 158func Test_list_add() 159 let l = [] 160 call add(l, 1) 161 call add(l, [2, 3]) 162 call add(l, []) 163 call add(l, test_null_list()) 164 call add(l, {'k' : 3}) 165 call add(l, {}) 166 call add(l, test_null_dict()) 167 call assert_equal([1, [2, 3], [], [], {'k' : 3}, {}, {}], l) 168 call assert_equal(1, add(test_null_list(), 4)) 169endfunc 170 171" Tests for Dictionary type 172 173func Test_dict() 174 " Creating Dictionary directly with different types 175 let d = {001: 'asd', 'b': [1, 2, function('strlen')], -1: {'a': 1},} 176 call assert_equal("{'1': 'asd', 'b': [1, 2, function('strlen')], '-1': {'a': 1}}", string(d)) 177 call assert_equal('asd', d.1) 178 call assert_equal(['-1', '1', 'b'], sort(keys(d))) 179 call assert_equal(['asd', [1, 2, function('strlen')], {'a': 1}], values(d)) 180 let v = [] 181 for [key, val] in items(d) 182 call extend(v, [key, val]) 183 unlet key val 184 endfor 185 call assert_equal(['1','asd','b',[1, 2, function('strlen')],'-1',{'a': 1}], v) 186 187 call extend(d, {3:33, 1:99}) 188 call extend(d, {'b':'bbb', 'c':'ccc'}, "keep") 189 call assert_fails("call extend(d, {3:333,4:444}, 'error')", 'E737') 190 call assert_equal({'c': 'ccc', '1': 99, 'b': [1, 2, function('strlen')], '3': 33, '-1': {'a': 1}}, d) 191 call filter(d, 'v:key =~ ''[ac391]''') 192 call assert_equal({'c': 'ccc', '1': 99, '3': 33, '-1': {'a': 1}}, d) 193 194 " duplicate key 195 call assert_fails("let d = {'k' : 10, 'k' : 20}", 'E721:') 196 " missing comma 197 call assert_fails("let d = {'k' : 10 'k' : 20}", 'E722:') 198 " missing curly brace 199 call assert_fails("let d = {'k' : 10,", 'E723:') 200 " invalid key 201 call assert_fails('let d = #{++ : 10}', 'E15:') 202 " wrong type for key 203 call assert_fails('let d={[] : 10}', 'E730:') 204 " undefined variable as value 205 call assert_fails("let d={'k' : i}", 'E121:') 206endfunc 207 208" This was allowed in legacy Vim script 209let s:dict_with_spaces = {'one' : 1 , 'two' : 2 , 'three' : 3} 210let s:dict_with_spaces_lit = #{one : 1 , two : 2 , three : 3} 211 212" Dictionary identity 213func Test_dict_identity() 214 let d = {001: 'asd', 'b': [1, 2, function('strlen')], -1: {'a': 1},} 215 let dd = d 216 let dx = copy(d) 217 call assert_true(d == dd) 218 call assert_false(d isnot dd) 219 call assert_true(d is dd) 220 call assert_true(d == dx) 221 call assert_false(d is dx) 222 call assert_true(d isnot dx) 223endfunc 224 225" removing items with :unlet 226func Test_dict_unlet() 227 let d = {'b':'bbb', '1': 99, '3': 33, '-1': {'a': 1}} 228 unlet d.b 229 unlet d[-1] 230 call assert_equal({'1': 99, '3': 33}, d) 231endfunc 232 233" manipulating a big Dictionary (hashtable.c has a border of 1000 entries) 234func Test_dict_big() 235 let d = {} 236 for i in range(1500) 237 let d[i] = 3000 - i 238 endfor 239 call assert_equal([3000, 2900, 2001, 1600, 1501], [d[0], d[100], d[999], d[1400], d[1499]]) 240 let str = '' 241 try 242 let n = d[1500] 243 catch 244 let str=substitute(v:exception, '\v(.{14}).*( \d{4}).*', '\1\2', '') 245 endtry 246 call assert_equal('Vim(let):E716: 1500', str) 247 248 " lookup each items 249 for i in range(1500) 250 call assert_equal(3000 - i, d[i]) 251 endfor 252 let i += 1 253 254 " delete even items 255 while i >= 2 256 let i -= 2 257 unlet d[i] 258 endwhile 259 call assert_equal('NONE', get(d, 1500 - 100, 'NONE')) 260 call assert_equal(2999, d[1]) 261 262 " delete odd items, checking value, one intentionally wrong 263 let d[33] = 999 264 let i = 1 265 while i < 1500 266 if i != 33 267 call assert_equal(3000 - i, d[i]) 268 else 269 call assert_equal(999, d[i]) 270 endif 271 unlet d[i] 272 let i += 2 273 endwhile 274 call assert_equal({}, d) 275 unlet d 276endfunc 277 278" Dictionary function 279func Test_dict_func() 280 let d = {} 281 func d.func(a) dict 282 return a:a . len(self.data) 283 endfunc 284 let d.data = [1,2,3] 285 call assert_equal('len: 3', d.func('len: ')) 286 let x = d.func('again: ') 287 call assert_equal('again: 3', x) 288 let Fn = d.func 289 call assert_equal('xxx3', Fn('xxx')) 290endfunc 291 292func Test_dict_assign() 293 let d = {} 294 let d.1 = 1 295 let d._ = 2 296 call assert_equal({'1': 1, '_': 2}, d) 297endfunc 298 299" Function in script-local List or Dict 300func Test_script_local_dict_func() 301 let g:dict = {} 302 function g:dict.func() dict 303 return 'g:dict.func' . self.foo[1] . self.foo[0]('asdf') 304 endfunc 305 let g:dict.foo = ['-', 2, 3] 306 call insert(g:dict.foo, function('strlen')) 307 call assert_equal('g:dict.func-4', g:dict.func()) 308 unlet g:dict 309endfunc 310 311" Test removing items in a dictionary 312func Test_dict_func_remove() 313 let d = {1:'a', 2:'b', 3:'c'} 314 call assert_equal('b', remove(d, 2)) 315 call assert_equal({1:'a', 3:'c'}, d) 316 317 call assert_fails("call remove(d, 1, 2)", 'E118:') 318 call assert_fails("call remove(d, 'a')", 'E716:') 319 call assert_fails("call remove(d, [])", 'E730:') 320endfunc 321 322" Nasty: remove func from Dict that's being called (works) 323func Test_dict_func_remove_in_use() 324 let d = {1:1} 325 func d.func(a) 326 return "a:" . a:a 327 endfunc 328 let expected = 'a:' . string(get(d, 'func')) 329 call assert_equal(expected, d.func(string(remove(d, 'func')))) 330endfunc 331 332func Test_dict_literal_keys() 333 call assert_equal({'one': 1, 'two2': 2, '3three': 3, '44': 4}, #{one: 1, two2: 2, 3three: 3, 44: 4},) 334 335 " why *{} cannot be used 336 let blue = 'blue' 337 call assert_equal('6', trim(execute('echo 2 *{blue: 3}.blue'))) 338endfunc 339 340" Nasty: deepcopy() dict that refers to itself (fails when noref used) 341func Test_dict_deepcopy() 342 let d = {1:1, 2:2} 343 let l = [4, d, 6] 344 let d[3] = l 345 let dc = deepcopy(d) 346 call assert_fails('call deepcopy(d, 1)', 'E698') 347 let l2 = [0, l, l, 3] 348 let l[1] = l2 349 let l3 = deepcopy(l2) 350 call assert_true(l3[1] is l3[2]) 351 call assert_fails("call deepcopy([1, 2], 2)", 'E474:') 352endfunc 353 354" Locked variables 355func Test_list_locked_var() 356 let expected = [ 357 \ [['0000-000', 'ppppppp'], 358 \ ['0000-000', 'ppppppp'], 359 \ ['0000-000', 'ppppppp']], 360 \ [['1000-000', 'ppppppF'], 361 \ ['0000-000', 'ppppppp'], 362 \ ['0000-000', 'ppppppp']], 363 \ [['1100-100', 'ppFppFF'], 364 \ ['0000-000', 'ppppppp'], 365 \ ['0000-000', 'ppppppp']], 366 \ [['1110-110', 'pFFpFFF'], 367 \ ['0010-010', 'pFppFpp'], 368 \ ['0000-000', 'ppppppp']], 369 \ [['1111-111', 'FFFFFFF'], 370 \ ['0011-011', 'FFpFFpp'], 371 \ ['0000-000', 'ppppppp']] 372 \ ] 373 for depth in range(5) 374 for u in range(3) 375 unlet! l 376 let l = [0, [1, [2, 3]], {4: 5, 6: {7: 8}}] 377 exe "lockvar " . depth . " l" 378 if u == 1 379 exe "unlockvar l" 380 elseif u == 2 381 exe "unlockvar " . depth . " l" 382 endif 383 let ps = islocked("l").islocked("l[1]").islocked("l[1][1]").islocked("l[1][1][0]").'-'.islocked("l[2]").islocked("l[2]['6']").islocked("l[2]['6'][7]") 384 call assert_equal(expected[depth][u][0], ps) 385 let ps = '' 386 try 387 let l[1][1][0] = 99 388 let ps .= 'p' 389 catch 390 let ps .= 'F' 391 endtry 392 try 393 let l[1][1] = [99] 394 let ps .= 'p' 395 catch 396 let ps .= 'F' 397 endtry 398 try 399 let l[1] = [99] 400 let ps .= 'p' 401 catch 402 let ps .= 'F' 403 endtry 404 try 405 let l[2]['6'][7] = 99 406 let ps .= 'p' 407 catch 408 let ps .= 'F' 409 endtry 410 try 411 let l[2][6] = {99: 99} 412 let ps .= 'p' 413 catch 414 let ps .= 'F' 415 endtry 416 try 417 let l[2] = {99: 99} 418 let ps .= 'p' 419 catch 420 let ps .= 'F' 421 endtry 422 try 423 let l = [99] 424 let ps .= 'p' 425 catch 426 let ps .= 'F' 427 endtry 428 call assert_equal(expected[depth][u][1], ps) 429 endfor 430 endfor 431 call assert_fails("let x=islocked('a b')", 'E488:') 432 let mylist = [1, 2, 3] 433 call assert_fails("let x = islocked('mylist[1:2]')", 'E786:') 434 let mydict = {'k' : 'v'} 435 call assert_fails("let x = islocked('mydict.a')", 'E716:') 436endfunc 437 438" Unletting locked variables 439func Test_list_locked_var_unlet() 440 let expected = [ 441 \ [['0000-000', 'ppppppp'], 442 \ ['0000-000', 'ppppppp'], 443 \ ['0000-000', 'ppppppp']], 444 \ [['1000-000', 'ppFppFp'], 445 \ ['0000-000', 'ppppppp'], 446 \ ['0000-000', 'ppppppp']], 447 \ [['1100-100', 'pFFpFFp'], 448 \ ['0000-000', 'ppppppp'], 449 \ ['0000-000', 'ppppppp']], 450 \ [['1110-110', 'FFFFFFp'], 451 \ ['0010-010', 'FppFppp'], 452 \ ['0000-000', 'ppppppp']], 453 \ [['1111-111', 'FFFFFFp'], 454 \ ['0011-011', 'FppFppp'], 455 \ ['0000-000', 'ppppppp']] 456 \ ] 457 458 for depth in range(5) 459 for u in range(3) 460 unlet! l 461 let l = [0, [1, [2, 3]], {4: 5, 6: {7: 8}}] 462 exe "lockvar " . depth . " l" 463 if u == 1 464 exe "unlockvar l" 465 elseif u == 2 466 exe "unlockvar " . depth . " l" 467 endif 468 let ps = islocked("l").islocked("l[1]").islocked("l[1][1]").islocked("l[1][1][0]").'-'.islocked("l[2]").islocked("l[2]['6']").islocked("l[2]['6'][7]") 469 call assert_equal(expected[depth][u][0], ps) 470 let ps = '' 471 try 472 unlet l[2]['6'][7] 473 let ps .= 'p' 474 catch 475 let ps .= 'F' 476 endtry 477 try 478 unlet l[2][6] 479 let ps .= 'p' 480 catch 481 let ps .= 'F' 482 endtry 483 try 484 unlet l[2] 485 let ps .= 'p' 486 catch 487 let ps .= 'F' 488 endtry 489 try 490 unlet l[1][1][0] 491 let ps .= 'p' 492 catch 493 let ps .= 'F' 494 endtry 495 try 496 unlet l[1][1] 497 let ps .= 'p' 498 catch 499 let ps .= 'F' 500 endtry 501 try 502 unlet l[1] 503 let ps .= 'p' 504 catch 505 let ps .= 'F' 506 endtry 507 try 508 unlet l 509 let ps .= 'p' 510 catch 511 let ps .= 'F' 512 endtry 513 call assert_equal(expected[depth][u][1], ps) 514 endfor 515 endfor 516endfunc 517 518" Locked variables and :unlet or list / dict functions 519 520" No :unlet after lock on dict: 521func Test_dict_lock_unlet() 522 unlet! d 523 let d = {'a': 99, 'b': 100} 524 lockvar 1 d 525 call assert_fails('unlet d.a', 'E741') 526endfunc 527 528" unlet after lock on dict item 529func Test_dict_item_lock_unlet() 530 unlet! d 531 let d = {'a': 99, 'b': 100} 532 lockvar d.a 533 unlet d.a 534 call assert_equal({'b' : 100}, d) 535endfunc 536 537" filter() after lock on dict item 538func Test_dict_lock_filter() 539 unlet! d 540 let d = {'a': 99, 'b': 100} 541 lockvar d.a 542 call filter(d, 'v:key != "a"') 543 call assert_equal({'b' : 100}, d) 544endfunc 545 546" map() after lock on dict 547func Test_dict_lock_map() 548 unlet! d 549 let d = {'a': 99, 'b': 100} 550 lockvar 1 d 551 call map(d, 'v:val + 200') 552 call assert_equal({'a' : 299, 'b' : 300}, d) 553endfunc 554 555" No extend() after lock on dict item 556func Test_dict_lock_extend() 557 unlet! d 558 let d = {'a': 99, 'b': 100} 559 lockvar d.a 560 call assert_fails("call extend(d, {'a' : 123})", 'E741') 561 call assert_equal({'a': 99, 'b': 100}, d) 562endfunc 563 564" Cannot use += with a locked dict 565func Test_dict_lock_operator() 566 unlet! d 567 let d = {} 568 lockvar d 569 call assert_fails("let d += {'k' : 10}", 'E741:') 570 unlockvar d 571endfunc 572 573" No remove() of write-protected scope-level variable 574func Tfunc1(this_is_a_long_parameter_name) 575 call assert_fails("call remove(a:, 'this_is_a_long_parameter_name')", 'E742') 576endfunc 577func Test_dict_scope_var_remove() 578 call Tfunc1('testval') 579endfunc 580 581" No extend() of write-protected scope-level variable 582func Test_dict_scope_var_extend() 583 call assert_fails("call extend(a:, {'this_is_a_long_parameter_name': 1234})", 'E742') 584endfunc 585 586func Tfunc2(this_is_a_long_parameter_name) 587 call assert_fails("call extend(a:, {'this_is_a_long_parameter_name': 1234})", 'E742') 588endfunc 589func Test_dict_scope_var_extend_overwrite() 590 call Tfunc2('testval') 591endfunc 592 593" No :unlet of variable in locked scope 594func Test_lock_var_unlet() 595 let b:testvar = 123 596 lockvar 1 b: 597 call assert_fails('unlet b:testvar', 'E741:') 598 unlockvar 1 b: 599 unlet! b:testvar 600endfunc 601 602" No :let += of locked list variable 603func Test_let_lock_list() 604 let l = ['a', 'b', 3] 605 lockvar 1 l 606 call assert_fails("let l += ['x']", 'E741:') 607 call assert_equal(['a', 'b', 3], l) 608 609 unlet l 610 let l = [1, 2, 3, 4] 611 lockvar! l 612 call assert_equal([1, 2, 3, 4], l) 613 unlockvar l[1] 614 call assert_fails('unlet l[0:1]', 'E741:') 615 call assert_equal([1, 2, 3, 4], l) 616 call assert_fails('unlet l[1:2]', 'E741:') 617 call assert_equal([1, 2, 3, 4], l) 618 unlockvar l[1] 619 call assert_fails('let l[0:1] = [0, 1]', 'E741:') 620 call assert_equal([1, 2, 3, 4], l) 621 call assert_fails('let l[1:2] = [0, 1]', 'E741:') 622 call assert_equal([1, 2, 3, 4], l) 623 unlet l 624endfunc 625 626" Locking part of the list 627func Test_let_lock_list_items() 628 let l = [1, 2, 3, 4] 629 lockvar l[2:] 630 call assert_equal(0, islocked('l[0]')) 631 call assert_equal(1, islocked('l[2]')) 632 call assert_equal(1, islocked('l[3]')) 633 call assert_fails('let l[2] = 10', 'E741:') 634 call assert_fails('let l[3] = 20', 'E741:') 635 unlet l 636endfunc 637 638" lockvar/islocked() triggering script autoloading 639func Test_lockvar_script_autoload() 640 let old_rtp = &rtp 641 set rtp+=./sautest 642 lockvar g:footest#x 643 unlockvar g:footest#x 644 call assert_equal(-1, 'g:footest#x'->islocked()) 645 call assert_equal(0, exists('g:footest#x')) 646 call assert_equal(1, g:footest#x) 647 let &rtp = old_rtp 648endfunc 649 650" a:000 function argument test 651func s:arg_list_test(...) 652 call assert_fails('let a:000 = [1, 2]', 'E46:') 653 call assert_fails('let a:000[0] = 9', 'E742:') 654 call assert_fails('let a:000[2] = [9, 10]', 'E742:') 655 call assert_fails('let a:000[3] = {9 : 10}', 'E742:') 656 657 " now the tests that should pass 658 let a:000[2][1] = 9 659 call extend(a:000[2], [5, 6]) 660 let a:000[3][5] = 8 661 let a:000[3]['a'] = 12 662 call assert_equal([1, 2, [3, 9, 5, 6], {'a': 12, '5': 8}], a:000) 663endfunc 664 665func Test_func_arg_list() 666 call s:arg_list_test(1, 2, [3, 4], {5: 6}) 667endfunc 668 669" Tests for reverse(), sort(), uniq() 670func Test_reverse_sort_uniq() 671 let l = ['-0', 'A11', 2, 2, 'xaaa', 4, 'foo', 'foo6', 'foo', [0, 1, 2], 'x8', [0, 1, 2], 1.5] 672 call assert_equal(['-0', 'A11', 2, 'xaaa', 4, 'foo', 'foo6', 'foo', [0, 1, 2], 'x8', [0, 1, 2], 1.5], uniq(copy(l))) 673 call assert_equal([1.5, [0, 1, 2], 'x8', [0, 1, 2], 'foo', 'foo6', 'foo', 4, 'xaaa', 2, 2, 'A11', '-0'], reverse(l)) 674 call assert_equal([1.5, [0, 1, 2], 'x8', [0, 1, 2], 'foo', 'foo6', 'foo', 4, 'xaaa', 2, 2, 'A11', '-0'], reverse(reverse(l))) 675 if has('float') 676 call assert_equal(['-0', 'A11', 'foo', 'foo', 'foo6', 'x8', 'xaaa', 1.5, 2, 2, 4, [0, 1, 2], [0, 1, 2]], sort(l)) 677 call assert_equal([[0, 1, 2], [0, 1, 2], 4, 2, 2, 1.5, 'xaaa', 'x8', 'foo6', 'foo', 'foo', 'A11', '-0'], reverse(sort(l))) 678 call assert_equal(['-0', 'A11', 'foo', 'foo', 'foo6', 'x8', 'xaaa', 1.5, 2, 2, 4, [0, 1, 2], [0, 1, 2]], sort(reverse(sort(l)))) 679 call assert_equal(['-0', 'A11', 'foo', 'foo6', 'x8', 'xaaa', 1.5, 2, 4, [0, 1, 2]], uniq(sort(l))) 680 681 let l = [7, 9, 'one', 18, 12, 22, 'two', 10.0e-16, -1, 'three', 0xff, 0.22, 'four'] 682 call assert_equal([-1, 'one', 'two', 'three', 'four', 1.0e-15, 0.22, 7, 9, 12, 18, 22, 255], sort(copy(l), 'n')) 683 684 let l = [7, 9, 18, 12, 22, 10.0e-16, -1, 0xff, 0, -0, 0.22, 'bar', 'BAR', 'Bar', 'Foo', 'FOO', 'foo', 'FOOBAR', {}, []] 685 call assert_equal(['bar', 'BAR', 'Bar', 'Foo', 'FOO', 'foo', 'FOOBAR', -1, 0, 0, 0.22, 1.0e-15, 12, 18, 22, 255, 7, 9, [], {}], sort(copy(l), 1)) 686 call assert_equal(['bar', 'BAR', 'Bar', 'Foo', 'FOO', 'foo', 'FOOBAR', -1, 0, 0, 0.22, 1.0e-15, 12, 18, 22, 255, 7, 9, [], {}], sort(copy(l), 'i')) 687 call assert_equal(['BAR', 'Bar', 'FOO', 'FOOBAR', 'Foo', 'bar', 'foo', -1, 0, 0, 0.22, 1.0e-15, 12, 18, 22, 255, 7, 9, [], {}], sort(copy(l))) 688 endif 689 690 call assert_fails('call reverse("")', 'E899:') 691 call assert_fails('call uniq([1, 2], {x, y -> []})', 'E745:') 692 call assert_fails("call sort([1, 2], function('min'), 1)", "E715:") 693 call assert_fails("call sort([1, 2], function('invalid_func'))", "E700:") 694 call assert_fails("call sort([1, 2], function('min'))", "E118:") 695endfunc 696 697" reduce a list or a blob 698func Test_reduce() 699 call assert_equal(1, reduce([], { acc, val -> acc + val }, 1)) 700 call assert_equal(10, reduce([1, 3, 5], { acc, val -> acc + val }, 1)) 701 call assert_equal(2 * (2 * ((2 * 1) + 2) + 3) + 4, reduce([2, 3, 4], { acc, val -> 2 * acc + val }, 1)) 702 call assert_equal('a x y z', ['x', 'y', 'z']->reduce({ acc, val -> acc .. ' ' .. val}, 'a')) 703 call assert_equal(#{ x: 1, y: 1, z: 1 }, ['x', 'y', 'z']->reduce({ acc, val -> extend(acc, { val: 1 }) }, {})) 704 call assert_equal([0, 1, 2, 3], reduce([1, 2, 3], function('add'), [0])) 705 706 let l = ['x', 'y', 'z'] 707 call assert_equal(42, reduce(l, function('get'), #{ x: #{ y: #{ z: 42 } } })) 708 call assert_equal(['x', 'y', 'z'], l) 709 710 call assert_equal(1, reduce([1], { acc, val -> acc + val })) 711 call assert_equal('x y z', reduce(['x', 'y', 'z'], { acc, val -> acc .. ' ' .. val })) 712 call assert_equal(120, range(1, 5)->reduce({ acc, val -> acc * val })) 713 call assert_fails("call reduce([], { acc, val -> acc + val })", 'E998: Reduce of an empty List with no initial value') 714 715 call assert_equal(1, reduce(0z, { acc, val -> acc + val }, 1)) 716 call assert_equal(1 + 0xaf + 0xbf + 0xcf, reduce(0zAFBFCF, { acc, val -> acc + val }, 1)) 717 call assert_equal(2 * (2 * 1 + 0xaf) + 0xbf, 0zAFBF->reduce({ acc, val -> 2 * acc + val }, 1)) 718 719 call assert_equal(0xff, reduce(0zff, { acc, val -> acc + val })) 720 call assert_equal(2 * (2 * 0xaf + 0xbf) + 0xcf, reduce(0zAFBFCF, { acc, val -> 2 * acc + val })) 721 call assert_fails("call reduce(0z, { acc, val -> acc + val })", 'E998: Reduce of an empty Blob with no initial value') 722 723 call assert_fails("call reduce({}, { acc, val -> acc + val }, 1)", 'E897:') 724 call assert_fails("call reduce(0, { acc, val -> acc + val }, 1)", 'E897:') 725 call assert_fails("call reduce('', { acc, val -> acc + val }, 1)", 'E897:') 726 727 let g:lut = [1, 2, 3, 4] 728 func EvilRemove() 729 call remove(g:lut, 1) 730 return 1 731 endfunc 732 call assert_fails("call reduce(g:lut, { acc, val -> EvilRemove() }, 1)", 'E742:') 733 unlet g:lut 734 delfunc EvilRemove 735 736 call assert_equal(42, reduce(test_null_list(), function('add'), 42)) 737 call assert_equal(42, reduce(test_null_blob(), function('add'), 42)) 738endfunc 739 740" splitting a string to a List using split() 741func Test_str_split() 742 call assert_equal(['aa', 'bb'], split(' aa bb ')) 743 call assert_equal(['aa', 'bb'], split(' aa bb ', '\W\+', 0)) 744 call assert_equal(['', 'aa', 'bb', ''], split(' aa bb ', '\W\+', 1)) 745 call assert_equal(['', '', 'aa', '', 'bb', '', ''], split(' aa bb ', '\W', 1)) 746 call assert_equal(['aa', '', 'bb'], split(':aa::bb:', ':', 0)) 747 call assert_equal(['', 'aa', '', 'bb', ''], split(':aa::bb:', ':', 1)) 748 call assert_equal(['aa', '', 'bb', 'cc', ''], split('aa,,bb, cc,', ',\s*', 1)) 749 call assert_equal(['a', 'b', 'c'], split('abc', '\zs')) 750 call assert_equal(['', 'a', '', 'b', '', 'c', ''], split('abc', '\zs', 1)) 751 call assert_fails("call split('abc', [])", 'E730:') 752 call assert_fails("call split('abc', 'b', [])", 'E745:') 753endfunc 754 755" compare recursively linked list and dict 756func Test_listdict_compare() 757 let l = [1, 2, 3, 4] 758 let d = {'1': 1, '2': l, '3': 3} 759 let l[1] = d 760 call assert_true(l == l) 761 call assert_true(d == d) 762 call assert_false(l != deepcopy(l)) 763 call assert_false(d != deepcopy(d)) 764 765 " comparison errors 766 call assert_fails('echo [1, 2] =~ {}', 'E691:') 767 call assert_fails('echo [1, 2] =~ [1, 2]', 'E692:') 768 call assert_fails('echo {} =~ 5', 'E735:') 769 call assert_fails('echo {} =~ {}', 'E736:') 770endfunc 771 772 " compare complex recursively linked list and dict 773func Test_listdict_compare_complex() 774 let l = [] 775 call add(l, l) 776 let dict4 = {"l": l} 777 call add(dict4.l, dict4) 778 let lcopy = deepcopy(l) 779 let dict4copy = deepcopy(dict4) 780 call assert_true(l == lcopy) 781 call assert_true(dict4 == dict4copy) 782endfunc 783 784" Test for extending lists and dictionaries 785func Test_listdict_extend() 786 " Test extend() with lists 787 788 " Pass the same List to extend() 789 let l = [1, 2, 3] 790 call assert_equal([1, 2, 3, 1, 2, 3], extend(l, l)) 791 call assert_equal([1, 2, 3, 1, 2, 3], l) 792 793 let l = [1, 2, 3] 794 call assert_equal([1, 2, 3, 4, 5, 6], extend(l, [4, 5, 6])) 795 call assert_equal([1, 2, 3, 4, 5, 6], l) 796 797 let l = [1, 2, 3] 798 call extend(l, [4, 5, 6], 0) 799 call assert_equal([4, 5, 6, 1, 2, 3], l) 800 801 let l = [1, 2, 3] 802 call extend(l, [4, 5, 6], 1) 803 call assert_equal([1, 4, 5, 6, 2, 3], l) 804 805 let l = [1, 2, 3] 806 call extend(l, [4, 5, 6], 3) 807 call assert_equal([1, 2, 3, 4, 5, 6], l) 808 809 let l = [1, 2, 3] 810 call extend(l, [4, 5, 6], -1) 811 call assert_equal([1, 2, 4, 5, 6, 3], l) 812 813 let l = [1, 2, 3] 814 call extend(l, [4, 5, 6], -3) 815 call assert_equal([4, 5, 6, 1, 2, 3], l) 816 817 let l = [1, 2, 3] 818 call assert_fails("call extend(l, [4, 5, 6], 4)", 'E684:') 819 call assert_fails("call extend(l, [4, 5, 6], -4)", 'E684:') 820 if has('float') 821 call assert_fails("call extend(l, [4, 5, 6], 1.2)", 'E805:') 822 endif 823 824 " Test extend() with dictionaries. 825 826 " Pass the same Dict to extend() 827 let d = { 'a': {'b': 'B'}} 828 call extend(d, d) 829 call assert_equal({'a': {'b': 'B'}}, d) 830 831 let d = {'a': 'A', 'b': 'B'} 832 call assert_equal({'a': 'A', 'b': 0, 'c': 'C'}, extend(d, {'b': 0, 'c':'C'})) 833 call assert_equal({'a': 'A', 'b': 0, 'c': 'C'}, d) 834 835 let d = {'a': 'A', 'b': 'B'} 836 call extend(d, {'a': 'A', 'b': 0, 'c': 'C'}, "force") 837 call assert_equal({'a': 'A', 'b': 0, 'c': 'C'}, d) 838 839 let d = {'a': 'A', 'b': 'B'} 840 call extend(d, {'b': 0, 'c':'C'}, "keep") 841 call assert_equal({'a': 'A', 'b': 'B', 'c': 'C'}, d) 842 843 let d = {'a': 'A', 'b': 'B'} 844 call assert_fails("call extend(d, {'b': 0, 'c':'C'}, 'error')", 'E737:') 845 call assert_fails("call extend(d, {'b': 0, 'c':'C'}, 'xxx')", 'E475:') 846 if has('float') 847 call assert_fails("call extend(d, {'b': 0, 'c':'C'}, 1.2)", 'E806:') 848 endif 849 call assert_equal({'a': 'A', 'b': 'B'}, d) 850 851 call assert_fails("call extend([1, 2], 1)", 'E712:') 852 call assert_fails("call extend([1, 2], {})", 'E712:') 853 854 " Extend g: dictionary with an invalid variable name 855 call assert_fails("call extend(g:, {'-!' : 10})", 'E461:') 856endfunc 857 858func s:check_scope_dict(x, fixed) 859 func s:gen_cmd(cmd, x) 860 return substitute(a:cmd, '\<x\ze:', a:x, 'g') 861 endfunc 862 863 let cmd = s:gen_cmd('let x:foo = 1', a:x) 864 if a:fixed 865 call assert_fails(cmd, 'E461') 866 else 867 exe cmd 868 exe s:gen_cmd('call assert_equal(1, x:foo)', a:x) 869 endif 870 871 let cmd = s:gen_cmd('let x:["bar"] = 2', a:x) 872 if a:fixed 873 call assert_fails(cmd, 'E461') 874 else 875 exe cmd 876 exe s:gen_cmd('call assert_equal(2, x:bar)', a:x) 877 endif 878 879 let cmd = s:gen_cmd('call extend(x:, {"baz": 3})', a:x) 880 if a:fixed 881 call assert_fails(cmd, 'E742') 882 else 883 exe cmd 884 exe s:gen_cmd('call assert_equal(3, x:baz)', a:x) 885 endif 886 887 if a:fixed 888 if a:x ==# 'a' 889 call assert_fails('unlet a:x', 'E795') 890 call assert_fails('call remove(a:, "x")', 'E742') 891 elseif a:x ==# 'v' 892 call assert_fails('unlet v:count', 'E795') 893 call assert_fails('call remove(v:, "count")', 'E742') 894 endif 895 else 896 exe s:gen_cmd('unlet x:foo', a:x) 897 exe s:gen_cmd('unlet x:bar', a:x) 898 exe s:gen_cmd('call remove(x:, "baz")', a:x) 899 endif 900 901 delfunc s:gen_cmd 902endfunc 903 904func Test_scope_dict() 905 " Test for g: 906 call s:check_scope_dict('g', v:false) 907 908 " Test for s: 909 call s:check_scope_dict('s', v:false) 910 911 " Test for l: 912 call s:check_scope_dict('l', v:false) 913 914 " Test for a: 915 call s:check_scope_dict('a', v:true) 916 917 " Test for b: 918 call s:check_scope_dict('b', v:false) 919 920 " Test for w: 921 call s:check_scope_dict('w', v:false) 922 923 " Test for t: 924 call s:check_scope_dict('t', v:false) 925 926 " Test for v: 927 call s:check_scope_dict('v', v:true) 928endfunc 929 930" Test for deep nesting of lists (> 100) 931func Test_deep_nested_list() 932 let deep_list = [] 933 let l = deep_list 934 for i in range(102) 935 let newlist = [] 936 call add(l, newlist) 937 let l = newlist 938 endfor 939 call add(l, 102) 940 941 call assert_fails('let m = deepcopy(deep_list)', 'E698:') 942 call assert_fails('lockvar 110 deep_list', 'E743:') 943 call assert_fails('unlockvar 110 deep_list', 'E743:') 944 call assert_fails('let x = execute("echo deep_list")', 'E724:') 945 call test_garbagecollect_now() 946 unlet deep_list 947endfunc 948 949" Test for deep nesting of dicts (> 100) 950func Test_deep_nested_dict() 951 let deep_dict = {} 952 let d = deep_dict 953 for i in range(102) 954 let newdict = {} 955 let d.k = newdict 956 let d = newdict 957 endfor 958 let d.k = 'v' 959 960 call assert_fails('let m = deepcopy(deep_dict)', 'E698:') 961 call assert_fails('lockvar 110 deep_dict', 'E743:') 962 call assert_fails('unlockvar 110 deep_dict', 'E743:') 963 call assert_fails('let x = execute("echo deep_dict")', 'E724:') 964 call test_garbagecollect_now() 965 unlet deep_dict 966endfunc 967 968" List and dict indexing tests 969func Test_listdict_index() 970 call assert_fails('echo function("min")[0]', 'E695:') 971 call assert_fails('echo v:true[0]', 'E909:') 972 let d = {'k' : 10} 973 call assert_fails('echo d.', 'E15:') 974 call assert_fails('echo d[1:2]', 'E719:') 975 call assert_fails("let v = [4, 6][{-> 1}]", 'E729:') 976 call assert_fails("let v = range(5)[2:[]]", 'E730:') 977 call assert_fails("let v = range(5)[2:{-> 2}(]", ['E15:', 'E116:']) 978 call assert_fails("let v = range(5)[2:3", 'E111:') 979 call assert_fails("let l = insert([1,2,3], 4, 10)", 'E684:') 980 call assert_fails("let l = insert([1,2,3], 4, -10)", 'E684:') 981 call assert_fails("let l = insert([1,2,3], 4, [])", 'E745:') 982 let l = [1, 2, 3] 983 call assert_fails("let l[i] = 3", 'E121:') 984 call assert_fails("let l[1.1] = 4", 'E806:') 985 call assert_fails("let l[:i] = [4, 5]", 'E121:') 986 call assert_fails("let l[:3.2] = [4, 5]", 'E806:') 987 let t = test_unknown() 988 call assert_fails("echo t[0]", 'E685:') 989endfunc 990 991" Test for a null list 992func Test_null_list() 993 let l = test_null_list() 994 call assert_equal(0, join(l)) 995 call assert_equal(0, len(l)) 996 call assert_equal(1, empty(l)) 997 call assert_fails('let s = join([1, 2], [])', 'E730:') 998 call assert_equal([], split(test_null_string())) 999 call assert_equal([], l[:2]) 1000 call assert_true([] == l) 1001 call assert_equal('[]', string(l)) 1002 call assert_equal(0, sort(l)) 1003 call assert_equal(0, uniq(l)) 1004 call assert_fails("let k = [] + l", 'E15:') 1005 call assert_fails("let k = l + []", 'E15:') 1006 call assert_equal(0, len(copy(l))) 1007 call assert_equal(0, count(l, 5)) 1008 call assert_equal([], deepcopy(l)) 1009 call assert_equal(5, get(l, 2, 5)) 1010 call assert_equal(-1, index(l, 2, 5)) 1011 call assert_equal(0, insert(l, 2, -1)) 1012 call assert_equal(0, min(l)) 1013 call assert_equal(0, max(l)) 1014 call assert_equal(0, remove(l, 0, 2)) 1015 call assert_equal([], repeat(l, 2)) 1016 call assert_equal(0, reverse(l)) 1017 call assert_equal(0, sort(l)) 1018 call assert_equal('[]', string(l)) 1019 call assert_equal(0, extend(l, l, 0)) 1020 lockvar l 1021 call assert_equal(1, islocked('l')) 1022 unlockvar l 1023endfunc 1024 1025" Test for a null dict 1026func Test_null_dict() 1027 call assert_equal(test_null_dict(), test_null_dict()) 1028 let d = test_null_dict() 1029 call assert_equal({}, d) 1030 call assert_equal(0, len(d)) 1031 call assert_equal(1, empty(d)) 1032 call assert_equal(0, items(d)) 1033 call assert_equal(0, keys(d)) 1034 call assert_equal(0, values(d)) 1035 call assert_false(has_key(d, 'k')) 1036 call assert_equal('{}', string(d)) 1037 call assert_fails('let x = d[10]') 1038 call assert_equal({}, {}) 1039 call assert_equal(0, len(copy(d))) 1040 call assert_equal(0, count(d, 'k')) 1041 call assert_equal({}, deepcopy(d)) 1042 call assert_equal(20, get(d, 'k', 20)) 1043 call assert_equal(0, min(d)) 1044 call assert_equal(0, max(d)) 1045 call assert_equal(0, remove(d, 'k')) 1046 call assert_equal('{}', string(d)) 1047 call assert_equal(0, extend(d, d, 0)) 1048 lockvar d 1049 call assert_equal(1, islocked('d')) 1050 unlockvar d 1051endfunc 1052 1053" vim: shiftwidth=2 sts=2 expandtab 1054