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