blob: ad322e68ca4a6d70f964c5737020ce41cb17e51f [file] [log] [blame]
Bram Moolenaardb913952012-06-29 12:54:53 +02001Tests for various python features. vim: set ft=vim :
2
Bram Moolenaar995a8cd2013-02-20 16:54:27 +01003NOTE: This will cause errors when run under valgrind.
4This would require recompiling Python with:
5 ./configure --without-pymalloc
6See http://svn.python.org/view/python/trunk/Misc/README.valgrind?view=markup
7
Bram Moolenaardb913952012-06-29 12:54:53 +02008STARTTEST
9:so small.vim
Bram Moolenaar8600e402013-05-30 13:28:41 +020010:set encoding=latin1
Bram Moolenaarfdde8802013-05-30 15:38:24 +020011:" HACK: currently crashes, skip the test
12:e! test.ok | wq! test.out
Bram Moolenaardb913952012-06-29 12:54:53 +020013:if !has('python') | e! test.ok | wq! test.out | endif
Bram Moolenaarc24c1ac2013-05-16 20:47:56 +020014:lang C
Bram Moolenaardb913952012-06-29 12:54:53 +020015:py import vim
16:fun Test()
17:let l = []
18:py l=vim.bindeval('l')
19:py f=vim.bindeval('function("strlen")')
20:" Extending List directly with different types
21:py l.extend([1, "as'd", [1, 2, f, {'a': 1}]])
22:$put =string(l)
23:$put =string(l[-1])
24:try
25: $put =string(l[-4])
26:catch
27: $put =v:exception[:13]
28:endtry
29:" List assignment
30:py l[0]=0
31:$put =string(l)
32:py l[-2]=f
33:$put =string(l)
34:"
35:" Extending Dictionary directly with different types
36:let d = {}
Bram Moolenaar355fd9b2013-05-30 13:14:13 +020037:fun d.f()
38: return 1
39:endfun
Bram Moolenaara9922d62013-05-30 13:01:18 +020040py << EOF
41d=vim.bindeval('d')
42d['1']='asd'
43d.update(b=[1, 2, f])
44d.update((('-1', {'a': 1}),))
45d.update({'0': -1})
46dk = d.keys()
47dv = d.values()
48di = d.items()
49dk.sort(key=repr)
50dv.sort(key=repr)
51di.sort(key=repr)
52EOF
Bram Moolenaar355fd9b2013-05-30 13:14:13 +020053:$put =pyeval('d[''f''](self={})')
Bram Moolenaara9922d62013-05-30 13:01:18 +020054:$put =pyeval('repr(dk)')
55:$put =substitute(pyeval('repr(dv)'),'0x\x\+','','g')
56:$put =substitute(pyeval('repr(di)'),'0x\x\+','','g')
Bram Moolenaar355fd9b2013-05-30 13:14:13 +020057:for [key, Val] in sort(items(d))
58: $put =string(key) . ' : ' . string(Val)
59: unlet key Val
Bram Moolenaardb913952012-06-29 12:54:53 +020060:endfor
61:"
62:" removing items with del
63:py del l[2]
64:$put =string(l)
65:let l = range(8)
66:py l=vim.bindeval('l')
67:try
68: py del l[:3]
69: py del l[1:]
70:catch
71: $put =v:exception
72:endtry
73:$put =string(l)
74:"
75:py del d['-1']
Bram Moolenaar355fd9b2013-05-30 13:14:13 +020076:py del d['f']
Bram Moolenaara9922d62013-05-30 13:01:18 +020077:$put =string(pyeval('d.get(''b'', 1)'))
78:$put =string(pyeval('d.pop(''b'')'))
79:$put =string(pyeval('d.get(''b'', 1)'))
80:$put =string(pyeval('d.pop(''1'', 2)'))
81:$put =string(pyeval('d.pop(''1'', 2)'))
82:$put =pyeval('repr(d.has_key(''0''))')
83:$put =pyeval('repr(d.has_key(''1''))')
84:$put =pyeval('repr(''0'' in d)')
85:$put =pyeval('repr(''1'' in d)')
86:$put =pyeval('repr(list(iter(d)))')
Bram Moolenaardb913952012-06-29 12:54:53 +020087:$put =string(d)
Bram Moolenaara9922d62013-05-30 13:01:18 +020088:$put =pyeval('repr(d.popitem(''0''))')
89:$put =pyeval('repr(d.get(''0''))')
90:$put =pyeval('repr(list(iter(d)))')
Bram Moolenaardb913952012-06-29 12:54:53 +020091:"
92:" removing items out of range: silently skip items that don't exist
93:let l = [0, 1, 2, 3]
94:py l=vim.bindeval('l')
95:" The following two ranges delete nothing as they match empty list:
96:py del l[2:1]
97:$put =string(l)
98:py del l[2:2]
99:$put =string(l)
100:py del l[2:3]
101:$put =string(l)
102:let l = [0, 1, 2, 3]
103:py l=vim.bindeval('l')
104:py del l[2:4]
105:$put =string(l)
106:let l = [0, 1, 2, 3]
107:py l=vim.bindeval('l')
108:py del l[2:5]
109:$put =string(l)
110:let l = [0, 1, 2, 3]
111:py l=vim.bindeval('l')
112:py del l[2:6]
113:$put =string(l)
114:let l = [0, 1, 2, 3]
115:py l=vim.bindeval('l')
116:" The following two ranges delete nothing as they match empty list:
117:py del l[-1:2]
118:$put =string(l)
119:py del l[-2:2]
120:$put =string(l)
121:py del l[-3:2]
122:$put =string(l)
123:let l = [0, 1, 2, 3]
124:py l=vim.bindeval('l')
125:py del l[-4:2]
126:$put =string(l)
127:let l = [0, 1, 2, 3]
128:py l=vim.bindeval('l')
129:py del l[-5:2]
130:$put =string(l)
131:let l = [0, 1, 2, 3]
132:py l=vim.bindeval('l')
133:py del l[-6:2]
134:$put =string(l)
135:"
136:" Slice assignment to a list
137:let l = [0, 1, 2, 3]
138:py l=vim.bindeval('l')
139:py l[0:0]=['a']
140:$put =string(l)
141:let l = [0, 1, 2, 3]
142:py l=vim.bindeval('l')
143:py l[1:2]=['b']
144:$put =string(l)
145:let l = [0, 1, 2, 3]
146:py l=vim.bindeval('l')
147:py l[2:4]=['c']
148:$put =string(l)
149:let l = [0, 1, 2, 3]
150:py l=vim.bindeval('l')
151:py l[4:4]=['d']
152:$put =string(l)
153:let l = [0, 1, 2, 3]
154:py l=vim.bindeval('l')
155:py l[-1:2]=['e']
156:$put =string(l)
157:let l = [0, 1, 2, 3]
158:py l=vim.bindeval('l')
159:py l[-10:2]=['f']
160:$put =string(l)
161:let l = [0, 1, 2, 3]
162:py l=vim.bindeval('l')
163:py l[2:-10]=['g']
164:$put =string(l)
165:let l = []
166:py l=vim.bindeval('l')
167:py l[0:0]=['h']
168:$put =string(l)
169:"
170:" Locked variables
171:let l = [0, 1, 2, 3]
172:py l=vim.bindeval('l')
173:lockvar! l
174:py l[2]='i'
175:$put =string(l)
176:unlockvar! l
177:"
178:" Function calls
179:function New(...)
180:return ['NewStart']+a:000+['NewEnd']
181:endfunction
182:function DictNew(...) dict
183:return ['DictNewStart']+a:000+['DictNewEnd', self]
184:endfunction
185:let l=[function('New'), function('DictNew')]
186:py l=vim.bindeval('l')
187:py l.extend(list(l[0](1, 2, 3)))
188:$put =string(l)
189:py l.extend(list(l[1](1, 2, 3, self={'a': 'b'})))
190:$put =string(l)
191:py l.extend([l[0].name])
192:$put =string(l)
193:try
194: py l[1](1, 2, 3)
195:catch
196: $put =v:exception[:16]
197:endtry
Bram Moolenaar355fd9b2013-05-30 13:14:13 +0200198:py f=l[0]
Bram Moolenaardb913952012-06-29 12:54:53 +0200199:delfunction New
200:try
Bram Moolenaar355fd9b2013-05-30 13:14:13 +0200201: py f(1, 2, 3)
Bram Moolenaardb913952012-06-29 12:54:53 +0200202:catch
203: $put =v:exception[:16]
204:endtry
205:if has('float')
206: let l=[0.0]
207: py l=vim.bindeval('l')
208: py l.extend([0.0])
209: $put =string(l)
210:else
211: $put ='[0.0, 0.0]'
212:endif
Bram Moolenaarc11073c2012-09-05 19:17:42 +0200213:let messages=[]
Bram Moolenaar03db85b2013-05-15 14:51:35 +0200214py <<EOF
Bram Moolenaarc11073c2012-09-05 19:17:42 +0200215d=vim.bindeval('{}')
216m=vim.bindeval('messages')
Bram Moolenaar03db85b2013-05-15 14:51:35 +0200217def em(expr, g=globals(), l=locals()):
218 try:
219 exec(expr, g, l)
220 except:
221 m.extend([sys.exc_type.__name__])
Bram Moolenaarc11073c2012-09-05 19:17:42 +0200222
Bram Moolenaar03db85b2013-05-15 14:51:35 +0200223em('d["abc"]')
224em('d["abc"]="\\0"')
225em('d["abc"]=vim')
226em('d[""]=1')
227em('d["a\\0b"]=1')
228em('d[u"a\\0b"]=1')
Bram Moolenaara9922d62013-05-30 13:01:18 +0200229
230em('d.pop("abc")')
231em('d.popitem("abc")')
Bram Moolenaarc11073c2012-09-05 19:17:42 +0200232EOF
233:$put =messages
Bram Moolenaar66b79852012-09-21 14:00:35 +0200234:unlet messages
235:" locked and scope attributes
236:let d={} | let dl={} | lockvar dl
237:for s in split("d dl v: g:")
238: let name=tr(s, ':', 's')
239: execute 'py '.name.'=vim.bindeval("'.s.'")'
240: let toput=s.' : '.join(map(['locked', 'scope'], 'v:val.":".pyeval(name.".".v:val)'), ';')
241: $put =toput
242:endfor
243:silent! let d.abc=1
244:silent! let dl.abc=1
245:py d.locked=True
246:py dl.locked=False
247:silent! let d.def=1
248:silent! let dl.def=1
249:put ='d:'.string(d)
250:put ='dl:'.string(dl)
251:unlet d dl
252:
253:let l=[] | let ll=[] | lockvar ll
254:for s in split("l ll")
255: let name=tr(s, ':', 's')
256: execute 'py '.name.'=vim.bindeval("'.s.'")'
257: let toput=s.' : locked:'.pyeval(name.'.locked')
258: $put =toput
259:endfor
260:silent! call extend(l, [0])
261:silent! call extend(ll, [0])
262:py l.locked=True
263:py ll.locked=False
264:silent! call extend(l, [1])
265:silent! call extend(ll, [1])
266:put ='l:'.string(l)
267:put ='ll:'.string(ll)
268:unlet l ll
Bram Moolenaardb913952012-06-29 12:54:53 +0200269:"
270:" pyeval()
271:let l=pyeval('range(3)')
272:$put =string(l)
273:let d=pyeval('{"a": "b", "c": 1, "d": ["e"]}')
274:$put =sort(items(d))
Bram Moolenaardb913952012-06-29 12:54:53 +0200275:if has('float')
276: let f=pyeval('0.0')
277: $put =string(f)
278:else
279: $put ='0.0'
280:endif
Bram Moolenaarc11073c2012-09-05 19:17:42 +0200281:" Invalid values:
282:for e in ['"\0"', '{"\0": 1}', 'undefined_name', 'vim']
283: try
284: let v=pyeval(e)
285: catch
286: let toput=e.":\t".v:exception[:13]
287: $put =toput
288: endtry
289:endfor
Bram Moolenaar76d711c2013-02-13 14:17:08 +0100290:"
291:" threading
292:let l = [0]
293:py l=vim.bindeval('l')
294:py <<EOF
295import threading
296import time
297
298class T(threading.Thread):
299 def __init__(self):
300 threading.Thread.__init__(self)
301 self.t = 0
302 self.running = True
303
304 def run(self):
305 while self.running:
306 self.t += 1
307 time.sleep(0.1)
308
309t = T()
310t.start()
311EOF
312:sleep 1
313:py t.running = False
314:py t.join()
315:py l[0] = t.t > 8 # check if the background thread is working
316:$put =string(l)
317:"
318:" settrace
319:let l = []
320:py l=vim.bindeval('l')
321:py <<EOF
322import sys
323
324def traceit(frame, event, arg):
325 global l
326 if event == "line":
327 l.extend([frame.f_lineno])
328 return traceit
329
330def trace_main():
331 for i in range(5):
332 pass
333EOF
334:py sys.settrace(traceit)
335:py trace_main()
336:py sys.settrace(None)
337:$put =string(l)
Bram Moolenaar24b11fb2013-04-05 19:32:36 +0200338:"
339:" Slice
340:py ll = vim.bindeval('[0, 1, 2, 3, 4, 5]')
341:py l = ll[:4]
342:$put =string(pyeval('l'))
343:py l = ll[2:]
344:$put =string(pyeval('l'))
345:py l = ll[:-4]
346:$put =string(pyeval('l'))
347:py l = ll[-2:]
348:$put =string(pyeval('l'))
349:py l = ll[2:4]
350:$put =string(pyeval('l'))
351:py l = ll[4:2]
352:$put =string(pyeval('l'))
353:py l = ll[-4:-2]
354:$put =string(pyeval('l'))
355:py l = ll[-2:-4]
356:$put =string(pyeval('l'))
357:py l = ll[:]
358:$put =string(pyeval('l'))
359:py l = ll[0:6]
360:$put =string(pyeval('l'))
361:py l = ll[-10:10]
362:$put =string(pyeval('l'))
Bram Moolenaar230bb3f2013-04-24 14:07:45 +0200363:"
364:" Vars
365:let g:foo = 'bac'
366:let w:abc = 'def'
367:let b:baz = 'bar'
Bram Moolenaara4720012013-05-15 16:27:37 +0200368:let t:bar = 'jkl'
Bram Moolenaar230bb3f2013-04-24 14:07:45 +0200369:try
370: throw "Abc"
371:catch
372: put =pyeval('vim.vvars[''exception'']')
373:endtry
374:put =pyeval('vim.vars[''foo'']')
375:put =pyeval('vim.current.window.vars[''abc'']')
376:put =pyeval('vim.current.buffer.vars[''baz'']')
Bram Moolenaara4720012013-05-15 16:27:37 +0200377:put =pyeval('vim.current.tabpage.vars[''bar'']')
Bram Moolenaar84e0f6c2013-05-06 03:52:55 +0200378:"
379:" Options
380:" paste: boolean, global
381:" previewheight number, global
382:" operatorfunc: string, global
383:" number: boolean, window-local
384:" numberwidth: number, window-local
385:" colorcolumn: string, window-local
386:" statusline: string, window-local/global
387:" autoindent: boolean, buffer-local
Bram Moolenaar55b8ad32013-05-17 13:38:04 +0200388:" shiftwidth: number, buffer-local
Bram Moolenaar84e0f6c2013-05-06 03:52:55 +0200389:" omnifunc: string, buffer-local
390:" preserveindent: boolean, buffer-local/global
391:" path: string, buffer-local/global
392:let g:bufs=[bufnr('%')]
393:new
394:let g:bufs+=[bufnr('%')]
395:vnew
396:let g:bufs+=[bufnr('%')]
397:wincmd j
398:vnew
399:let g:bufs+=[bufnr('%')]
400:wincmd l
401:fun RecVars(opt)
402: let gval =string(eval('&g:'.a:opt))
403: let wvals=join(map(range(1, 4), 'v:val.":".string(getwinvar(v:val, "&".a:opt))'))
404: let bvals=join(map(copy(g:bufs), 'v:val.":".string(getbufvar(v:val, "&".a:opt))'))
405: put =' G: '.gval
406: put =' W: '.wvals
407: put =' B: '.wvals
408:endfun
409py << EOF
410def e(s, g=globals(), l=locals()):
411 try:
412 exec(s, g, l)
Bram Moolenaar03db85b2013-05-15 14:51:35 +0200413 except:
Bram Moolenaara7b64ce2013-05-21 20:40:40 +0200414 vim.command('return ' + repr(sys.exc_type.__name__))
Bram Moolenaar84e0f6c2013-05-06 03:52:55 +0200415
416def ev(s, g=globals(), l=locals()):
417 try:
418 return eval(s, g, l)
Bram Moolenaar03db85b2013-05-15 14:51:35 +0200419 except:
Bram Moolenaara7b64ce2013-05-21 20:40:40 +0200420 vim.command('let exc=' + repr(sys.exc_type.__name__))
Bram Moolenaar84e0f6c2013-05-06 03:52:55 +0200421 return 0
422EOF
423:function E(s)
424: python e(vim.eval('a:s'))
425:endfunction
426:function Ev(s)
Bram Moolenaara7b64ce2013-05-21 20:40:40 +0200427: let r=pyeval('ev(vim.eval("a:s"))')
428: if exists('exc')
429: throw exc
430: endif
431: return r
Bram Moolenaar84e0f6c2013-05-06 03:52:55 +0200432:endfunction
433:py gopts1=vim.options
434:py wopts1=vim.windows[2].options
435:py wopts2=vim.windows[0].options
436:py wopts3=vim.windows[1].options
437:py bopts1=vim.buffers[vim.bindeval("g:bufs")[2]].options
438:py bopts2=vim.buffers[vim.bindeval("g:bufs")[1]].options
439:py bopts3=vim.buffers[vim.bindeval("g:bufs")[0]].options
440:let lst=[]
441:let lst+=[['paste', 1, 0, 1, 2, 1, 1, 0 ]]
442:let lst+=[['previewheight', 5, 1, 6, 'a', 0, 1, 0 ]]
443:let lst+=[['operatorfunc', 'A', 'B', 'C', 2, 0, 1, 0 ]]
444:let lst+=[['number', 0, 1, 1, 0, 1, 0, 1 ]]
445:let lst+=[['numberwidth', 2, 3, 5, -100, 0, 0, 1 ]]
446:let lst+=[['colorcolumn', '+1', '+2', '+3', 'abc', 0, 0, 1 ]]
447:let lst+=[['statusline', '1', '2', '4', 0, 0, 1, 1 ]]
448:let lst+=[['autoindent', 0, 1, 1, 2, 1, 0, 2 ]]
Bram Moolenaar55b8ad32013-05-17 13:38:04 +0200449:let lst+=[['shiftwidth', 0, 2, 1, 3, 0, 0, 2 ]]
Bram Moolenaar84e0f6c2013-05-06 03:52:55 +0200450:let lst+=[['omnifunc', 'A', 'B', 'C', 1, 0, 0, 2 ]]
451:let lst+=[['preserveindent', 0, 1, 1, 2, 1, 1, 2 ]]
452:let lst+=[['path', '.,,', ',,', '.', 0, 0, 1, 2 ]]
453:for [oname, oval1, oval2, oval3, invval, bool, global, local] in lst
454: py oname=vim.eval('oname')
455: py oval1=vim.bindeval('oval1')
456: py oval2=vim.bindeval('oval2')
457: py oval3=vim.bindeval('oval3')
458: if invval is 0 || invval is 1
459: py invval=bool(vim.bindeval('invval'))
460: else
461: py invval=vim.bindeval('invval')
462: endif
463: if bool
464: py oval1=bool(oval1)
465: py oval2=bool(oval2)
466: py oval3=bool(oval3)
467: endif
468: put ='>>> '.oname
469: for v in ['gopts1', 'wopts1', 'bopts1']
470: try
471: put =' p/'.v.': '.Ev('repr('.v.'['''.oname.'''])')
472: catch
473: put =' p/'.v.'! '.v:exception
474: endtry
Bram Moolenaara7b64ce2013-05-21 20:40:40 +0200475: let r=E(v.'['''.oname.''']=invval')
476: if r isnot 0
477: put =' inv: '.string(invval).'! '.r
478: endif
Bram Moolenaar84e0f6c2013-05-06 03:52:55 +0200479: for vv in (v is# 'gopts1' ? [v] : [v, v[:-2].'2', v[:-2].'3'])
480: let val=substitute(vv, '^.opts', 'oval', '')
Bram Moolenaara7b64ce2013-05-21 20:40:40 +0200481: let r=E(vv.'['''.oname.''']='.val)
482: if r isnot 0
483: put =' '.vv.'! '.r
484: endif
Bram Moolenaar84e0f6c2013-05-06 03:52:55 +0200485: endfor
486: endfor
487: call RecVars(oname)
488: for v in ['wopts3', 'bopts3']
Bram Moolenaara7b64ce2013-05-21 20:40:40 +0200489: let r=E('del '.v.'["'.oname.'"]')
490: if r isnot 0
491: put =' del '.v.'! '.r
492: endif
Bram Moolenaar84e0f6c2013-05-06 03:52:55 +0200493: endfor
494: call RecVars(oname)
Bram Moolenaar84e0f6c2013-05-06 03:52:55 +0200495:endfor
496:only
Bram Moolenaarb6c589a2013-05-15 14:39:52 +0200497:for buf in g:bufs[1:]
498: execute 'bwipeout!' buf
499:endfor
Bram Moolenaarbd80f352013-05-12 21:16:23 +0200500:"
501:" Test buffer object
502:vnew
503:put ='First line'
504:put ='Second line'
505:put ='Third line'
506:1 delete _
507:py b=vim.current.buffer
508:wincmd w
509:mark a
Bram Moolenaare9ba5162013-05-29 22:02:22 +0200510:augroup BUFS
511: autocmd BufFilePost * python cb.append(vim.eval('expand("<abuf>")') + ':BufFilePost:' + vim.eval('bufnr("%")'))
512: autocmd BufFilePre * python cb.append(vim.eval('expand("<abuf>")') + ':BufFilePre:' + vim.eval('bufnr("%")'))
513:augroup END
Bram Moolenaarbd80f352013-05-12 21:16:23 +0200514py << EOF
515cb = vim.current.buffer
516# Tests BufferAppend and BufferItem
517cb.append(b[0])
518# Tests BufferSlice and BufferAssSlice
519cb.append('abc') # Will be overwritten
520cb[-1:] = b[:-2]
521# Test BufferLength and BufferAssSlice
522cb.append('def') # Will not be overwritten
523cb[len(cb):] = b[:]
524# Test BufferAssItem and BufferMark
525cb.append('ghi') # Will be overwritten
526cb[-1] = repr((len(cb) - cb.mark('a')[0], cb.mark('a')[1]))
527# Test BufferRepr
528cb.append(repr(cb) + repr(b))
529# Modify foreign buffer
530b.append('foo')
531b[0]='bar'
532b[0:0]=['baz']
533vim.command('call append("$", getbufline(%i, 1, "$"))' % b.number)
Bram Moolenaare9ba5162013-05-29 22:02:22 +0200534# Test assigning to name property
535old_name = cb.name
536cb.name = 'foo'
537cb.append(cb.name[-11:])
538b.name = 'bar'
539cb.append(b.name[-11:])
540cb.name = old_name
541cb.append(cb.name[-17:])
Bram Moolenaarbd80f352013-05-12 21:16:23 +0200542# Test CheckBuffer
Bram Moolenaare9ba5162013-05-29 22:02:22 +0200543for _b in vim.buffers:
544 if _b is not cb:
545 vim.command('bwipeout! ' + str(_b.number))
546del _b
Bram Moolenaar9e822c02013-05-29 22:15:30 +0200547cb.append('valid: b:%s, cb:%s' % (repr(b.valid), repr(cb.valid)))
Bram Moolenaare9ba5162013-05-29 22:02:22 +0200548for expr in ('b[1]','b[:] = ["A", "B"]','b[:]','b.append("abc")', 'b.name = "!"'):
Bram Moolenaarbd80f352013-05-12 21:16:23 +0200549 try:
550 exec(expr)
551 except vim.error:
552 pass
553 else:
554 # Usually a SEGV here
555 # Should not happen in any case
556 cb.append('No exception for ' + expr)
Bram Moolenaare9ba5162013-05-29 22:02:22 +0200557vim.command('cd .')
Bram Moolenaarbd80f352013-05-12 21:16:23 +0200558EOF
Bram Moolenaare9ba5162013-05-29 22:02:22 +0200559:augroup BUFS
560: autocmd!
561:augroup END
562:augroup! BUFS
Bram Moolenaarb6c589a2013-05-15 14:39:52 +0200563:"
564:" Test vim.buffers object
565:set hidden
566:edit a
567:buffer #
568:edit b
569:buffer #
570:edit c
571:buffer #
572py << EOF
Bram Moolenaar2a0f3d32013-05-21 22:23:56 +0200573try:
574 from __builtin__ import next
575except ImportError:
576 next = lambda o: o.next()
Bram Moolenaarb6c589a2013-05-15 14:39:52 +0200577# Check GCing iterator that was not fully exhausted
578i = iter(vim.buffers)
579cb.append('i:' + str(next(i)))
580# and also check creating more then one iterator at a time
581i2 = iter(vim.buffers)
582cb.append('i2:' + str(next(i2)))
583cb.append('i:' + str(next(i)))
584# The following should trigger GC and not cause any problems
585del i
586del i2
587i3 = iter(vim.buffers)
588cb.append('i3:' + str(next(i3)))
589del i3
590
591prevnum = 0
592for b in vim.buffers:
593 # Check buffer order
594 if prevnum >= b.number:
595 cb.append('!!! Buffer numbers not in strictly ascending order')
596 # Check indexing: vim.buffers[number].number == number
597 cb.append(str(b.number) + ':' + repr(vim.buffers[b.number]) + '=' + repr(b))
598 prevnum = b.number
599
600cb.append(str(len(vim.buffers)))
601
602bnums = list(map(lambda b: b.number, vim.buffers))[1:]
603
604# Test wiping out buffer with existing iterator
605i4 = iter(vim.buffers)
606cb.append('i4:' + str(next(i4)))
607vim.command('bwipeout! ' + str(bnums.pop(0)))
608try:
609 next(i4)
610except vim.error:
611 pass
612else:
613 cb.append('!!!! No vim.error')
614i4 = iter(vim.buffers)
615vim.command('bwipeout! ' + str(bnums.pop(-1)))
616vim.command('bwipeout! ' + str(bnums.pop(-1)))
617cb.append('i4:' + str(next(i4)))
618try:
619 next(i4)
620except StopIteration:
621 cb.append('StopIteration')
622EOF
Bram Moolenaara4720012013-05-15 16:27:37 +0200623:"
624:" Test vim.{tabpage,window}list and vim.{tabpage,window} objects
625:tabnew 0
626:tabnew 1
627:vnew a.1
628:tabnew 2
629:vnew a.2
630:vnew b.2
631:vnew c.2
632py << EOF
633cb.append('Number of tabs: ' + str(len(vim.tabpages)))
634cb.append('Current tab pages:')
635def W(w):
Bram Moolenaar2a0f3d32013-05-21 22:23:56 +0200636 if repr(w).find('(unknown)') != -1:
Bram Moolenaara4720012013-05-15 16:27:37 +0200637 return '<window object (unknown)>'
638 else:
639 return repr(w)
Bram Moolenaar2a0f3d32013-05-21 22:23:56 +0200640
Bram Moolenaare9ba5162013-05-29 22:02:22 +0200641start = len(cb)
642
643def Cursor(w):
Bram Moolenaar2a0f3d32013-05-21 22:23:56 +0200644 if w.buffer is cb:
645 return repr((start - w.cursor[0], w.cursor[1]))
646 else:
647 return repr(w.cursor)
648
Bram Moolenaara4720012013-05-15 16:27:37 +0200649for t in vim.tabpages:
650 cb.append(' ' + repr(t) + '(' + str(t.number) + ')' + ': ' + str(len(t.windows)) + ' windows, current is ' + W(t.window))
651 cb.append(' Windows:')
652 for w in t.windows:
Bram Moolenaar2a0f3d32013-05-21 22:23:56 +0200653 cb.append(' ' + W(w) + '(' + str(w.number) + ')' + ': displays buffer ' + repr(w.buffer) + '; cursor is at ' + Cursor(w))
Bram Moolenaara4720012013-05-15 16:27:37 +0200654 # Other values depend on the size of the terminal, so they are checked partly:
655 for attr in ('height', 'row', 'width', 'col'):
656 try:
657 aval = getattr(w, attr)
658 if type(aval) is not long:
659 raise TypeError
660 if aval < 0:
661 raise ValueError
662 except Exception:
663 cb.append('!!!!!! Error while getting attribute ' + attr + ': ' + sys.exc_type.__name__)
664 w.cursor = (len(w.buffer), 0)
665cb.append('Number of windows in current tab page: ' + str(len(vim.windows)))
666if list(vim.windows) != list(vim.current.tabpage.windows):
667 cb.append('!!!!!! Windows differ')
668EOF
669:"
670:" Test vim.current
671py << EOF
672def H(o):
673 return repr(o)
674cb.append('Current tab page: ' + repr(vim.current.tabpage))
675cb.append('Current window: ' + repr(vim.current.window) + ': ' + H(vim.current.window) + ' is ' + H(vim.current.tabpage.window))
676cb.append('Current buffer: ' + repr(vim.current.buffer) + ': ' + H(vim.current.buffer) + ' is ' + H(vim.current.window.buffer)+ ' is ' + H(vim.current.tabpage.window.buffer))
677# Assigning: fails
678try:
679 vim.current.window = vim.tabpages[0].window
680except ValueError:
681 cb.append('ValueError at assigning foreign tab window')
682
683for attr in ('window', 'tabpage', 'buffer'):
684 try:
685 setattr(vim.current, attr, None)
686 except TypeError:
687 cb.append('Type error at assigning None to vim.current.' + attr)
688
689# Assigning: success
690vim.current.tabpage = vim.tabpages[-2]
691vim.current.buffer = cb
692vim.current.window = vim.windows[0]
693vim.current.window.cursor = (len(vim.current.buffer), 0)
694cb.append('Current tab page: ' + repr(vim.current.tabpage))
695cb.append('Current window: ' + repr(vim.current.window))
696cb.append('Current buffer: ' + repr(vim.current.buffer))
697cb.append('Current line: ' + repr(vim.current.line))
Bram Moolenaar9e822c02013-05-29 22:15:30 +0200698ws = list(vim.windows)
699ts = list(vim.tabpages)
Bram Moolenaara4720012013-05-15 16:27:37 +0200700for b in vim.buffers:
701 if b is not cb:
Bram Moolenaarcac867a2013-05-21 19:50:34 +0200702 vim.command('bwipeout! ' + str(b.number))
Bram Moolenaar9e822c02013-05-29 22:15:30 +0200703cb.append('w.valid: ' + repr([w.valid for w in ws]))
704cb.append('t.valid: ' + repr([t.valid for t in ts]))
Bram Moolenaara4720012013-05-15 16:27:37 +0200705EOF
706:tabonly!
707:only!
Bram Moolenaarcac867a2013-05-21 19:50:34 +0200708:"
709:" Test types
710py << EOF
711for expr, attr in (
712 ('vim.vars', 'Dictionary'),
713 ('vim.options', 'Options'),
714 ('vim.bindeval("{}")', 'Dictionary'),
715 ('vim.bindeval("[]")', 'List'),
716 ('vim.bindeval("function(\'tr\')")', 'Function'),
717 ('vim.current.buffer', 'Buffer'),
718 ('vim.current.range', 'Range'),
719 ('vim.current.window', 'Window'),
720 ('vim.current.tabpage', 'TabPage'),
721):
722 cb.append(expr + ':' + attr + ':' + repr(type(eval(expr)) is getattr(vim, attr)))
723EOF
Bram Moolenaara7b64ce2013-05-21 20:40:40 +0200724:"
Bram Moolenaardd8aca62013-05-29 22:36:10 +0200725:" Test __dir__() method
726py << EOF
727for name, o in (
728 ('current', vim.current),
729 ('buffer', vim.current.buffer),
730 ('window', vim.current.window),
731 ('tabpage', vim.current.tabpage),
732 ('range', vim.current.range),
733 ('dictionary', vim.bindeval('{}')),
734 ('list', vim.bindeval('[]')),
735 ('function', vim.bindeval('function("tr")')),
736 ('output', sys.stdout),
737 ):
738 cb.append(name + ':' + ','.join(dir(o)))
739del name
740del o
741EOF
742:"
Bram Moolenaara9922d62013-05-30 13:01:18 +0200743:" Test vim.*.__new__
744:$put =string(pyeval('vim.Dictionary({})'))
745:$put =string(pyeval('vim.Dictionary(a=1)'))
746:$put =string(pyeval('vim.Dictionary(((''a'', 1),))'))
Bram Moolenaar78cddbe2013-05-30 13:05:58 +0200747:$put =string(pyeval('vim.List()'))
748:$put =string(pyeval('vim.List(iter(''abc''))'))
Bram Moolenaar355fd9b2013-05-30 13:14:13 +0200749:$put =string(pyeval('vim.Function(''tr'')'))
Bram Moolenaar01a7a722013-05-30 12:26:58 +0200750:"
751:" Test stdout/stderr
752:redir => messages
753:py sys.stdout.write('abc') ; sys.stdout.write('def')
754:py sys.stderr.write('abc') ; sys.stderr.write('def')
755:py sys.stdout.writelines(iter('abc'))
756:py sys.stderr.writelines(iter('abc'))
757:redir END
758:$put =string(substitute(messages, '\d\+', '', 'g'))
Bram Moolenaara9922d62013-05-30 13:01:18 +0200759:" Test subclassing
Bram Moolenaar355fd9b2013-05-30 13:14:13 +0200760:fun Put(...)
761: $put =string(a:000)
762: return a:000
763:endfun
Bram Moolenaara9922d62013-05-30 13:01:18 +0200764py << EOF
765class DupDict(vim.Dictionary):
766 def __setitem__(self, key, value):
767 super(DupDict, self).__setitem__(key, value)
768 super(DupDict, self).__setitem__('dup_' + key, value)
769dd = DupDict()
770dd['a'] = 'b'
Bram Moolenaar78cddbe2013-05-30 13:05:58 +0200771
772class DupList(vim.List):
773 def __getitem__(self, idx):
774 return [super(DupList, self).__getitem__(idx)] * 2
775
776dl = DupList()
777dl2 = DupList(iter('abc'))
778dl.extend(dl2[0])
Bram Moolenaar355fd9b2013-05-30 13:14:13 +0200779
780class DupFun(vim.Function):
781 def __call__(self, arg):
782 return super(DupFun, self).__call__(arg, arg)
783
784df = DupFun('Put')
Bram Moolenaara9922d62013-05-30 13:01:18 +0200785EOF
786:$put =string(sort(keys(pyeval('dd'))))
Bram Moolenaar78cddbe2013-05-30 13:05:58 +0200787:$put =string(pyeval('dl'))
788:$put =string(pyeval('dl2'))
Bram Moolenaar355fd9b2013-05-30 13:14:13 +0200789:$put =string(pyeval('df(2)'))
Bram Moolenaar01a7a722013-05-30 12:26:58 +0200790:"
Bram Moolenaar8600e402013-05-30 13:28:41 +0200791:" Test errors
792:fun F() dict
793:endfun
794:fun D()
Bram Moolenaara7b64ce2013-05-21 20:40:40 +0200795:endfun
796py << EOF
797def ee(expr, g=globals(), l=locals()):
798 try:
799 exec(expr, g, l)
800 except:
Bram Moolenaar8600e402013-05-30 13:28:41 +0200801 cb.append(expr + ':' + repr(sys.exc_info()[:2]))
802 else:
803 cb.append(expr + ':NOT FAILED')
804d = vim.Dictionary()
805ned = vim.Dictionary(foo='bar', baz='abc')
806dl = vim.Dictionary(a=1)
807dl.locked = True
808l = vim.List()
809ll = vim.List('abc')
810ll.locked = True
811f = vim.Function('string')
812fd = vim.Function('F')
813fdel = vim.Function('D')
814vim.command('delfunction D')
815
816def subexpr_test(expr, name, subexprs):
817 cb.append('>>> Testing %s using %s' % (name, expr))
818 for subexpr in subexprs:
819 ee(expr % subexpr)
820 cb.append('<<< Finished')
821
822def stringtochars_test(expr):
823 return subexpr_test(expr, 'StringToChars', (
824 '1', # Fail type checks
825 'u"\\0"', # Fail PyString_AsStringAndSize(bytes, , NULL) check
826 '"\\0"', # Fail PyString_AsStringAndSize(object, , NULL) check
827 ))
828
829class Mapping(object):
830 def __init__(self, d):
831 self.d = d
832
833 def __getitem__(self, key):
834 return self.d[key]
835
836 def keys(self):
837 return self.d.keys()
838
839 def items(self):
840 return self.d.items()
841
842def convertfrompyobject_test(expr, recurse=True):
843 # pydict_to_tv
844 stringtochars_test(expr % '{%s : 1}')
845 if recurse:
846 convertfrompyobject_test(expr % '{"abc" : %s}', False)
847 # pymap_to_tv
848 stringtochars_test(expr % 'Mapping({%s : 1})')
849 if recurse:
850 convertfrompyobject_test(expr % 'Mapping({"abc" : %s})', False)
851 # pyseq_to_tv
852 iter_test(expr)
853 return subexpr_test(expr, 'ConvertFromPyObject', (
854 'None', # Not conversible
855 '{"": 1}', # Empty key not allowed
856 'FailingMapping()', #
857 'FailingMappingKey()', #
858 ))
859
860def convertfrompymapping_test(expr):
861 convertfrompyobject_test(expr)
862 return subexpr_test(expr, 'ConvertFromPyMapping', (
863 '[]',
864 ))
865
866def iter_test(expr):
867 return subexpr_test(expr, '*Iter*', (
868 'FailingIter()',
869 'FailingIterNext()',
870 ))
871
872class FailingTrue(object):
873 def __nonzero__(self):
874 raise NotImplementedError
875
876class FailingIter(object):
877 def __iter__(self):
878 raise NotImplementedError
879
880class FailingIterNext(object):
881 def __iter__(self):
882 return self
883
884 def next(self):
885 raise NotImplementedError
886
887class FailingMappingKey(object):
888 def __getitem__(self, item):
889 raise NotImplementedError
890
891 def keys(self):
892 return list("abc")
893
894class FailingMapping(object):
895 def __getitem__(self):
896 raise NotImplementedError
897
898 def keys(self):
899 raise NotImplementedError
900
901class FailingList(list):
902 def __getitem__(self, idx):
903 if i == 2:
904 raise NotImplementedError
905 else:
906 return super(FailingList, self).__getitem__(idx)
907
908cb.append("> Output")
909cb.append(">> OutputSetattr")
910ee('del sys.stdout.softspace')
911ee('sys.stdout.softspace = []')
912ee('sys.stdout.attr = None')
913cb.append(">> OutputWrite")
914ee('sys.stdout.write(None)')
915cb.append(">> OutputWriteLines")
916ee('sys.stdout.writelines(None)')
917ee('sys.stdout.writelines([1])')
918iter_test('sys.stdout.writelines(%s)')
919cb.append("> VimCommand")
920ee('vim.command(1)')
921#! Not checked: vim->python exceptions translating: checked later
922cb.append("> VimToPython")
923#! Not checked: everything: needs errors in internal python functions
924cb.append("> VimEval")
925ee('vim.eval(1)')
926#! Not checked: everything: needs errors in internal python functions
927cb.append("> VimEvalPy")
928ee('vim.bindeval(1)')
929#! Not checked: vim->python exceptions translating: checked later
930cb.append("> VimStrwidth")
931ee('vim.strwidth(1)')
932cb.append("> Dictionary")
933cb.append(">> DictionaryConstructor")
934ee('vim.Dictionary("abc")')
935##! Not checked: py_dict_alloc failure
936cb.append(">> DictionarySetattr")
937ee('del d.locked')
938ee('d.locked = FailingTrue()')
939ee('vim.vvars.locked = False')
940ee('d.scope = True')
941ee('d.xxx = True')
942cb.append(">> _DictionaryItem")
943ee('d.get("a", 2, 3)')
944stringtochars_test('d.get(%s)')
945ee('d.pop("a")')
946ee('dl.pop("a")')
947cb.append(">> DictionaryIterNext")
948ee('for i in ned: ned["a"] = 1')
949cb.append(">> DictionaryAssItem")
950ee('dl["b"] = 1')
951stringtochars_test('d[%s] = 1')
952convertfrompyobject_test('d["a"] = %s')
953cb.append(">> DictionaryUpdate")
954cb.append(">>> kwargs")
955cb.append(">>> iter")
956ee('d.update(FailingMapping())')
957ee('d.update([FailingIterNext()])')
958iter_test('d.update(%s)')
959convertfrompyobject_test('d.update(%s)')
960stringtochars_test('d.update(((%s, 0),))')
961convertfrompyobject_test('d.update((("a", %s),))')
962cb.append(">> DictionaryPopItem")
963ee('d.popitem(1, 2)')
964cb.append(">> DictionaryHasKey")
965ee('d.has_key()')
966cb.append("> List")
967cb.append(">> ListConstructor")
968ee('vim.List(1, 2)')
969ee('vim.List(a=1)')
970iter_test('vim.List(%s)')
971convertfrompyobject_test('vim.List([%s])')
972cb.append(">> ListItem")
973ee('l[1000]')
974cb.append(">> ListAssItem")
975ee('ll[1] = 2')
976ee('l[1000] = 3')
977cb.append(">> ListAssSlice")
978ee('ll[1:100] = "abc"')
979iter_test('l[:] = %s')
980convertfrompyobject_test('l[:] = [%s]')
981cb.append(">> ListConcatInPlace")
982iter_test('l.extend(%s)')
983convertfrompyobject_test('l.extend([%s])')
984cb.append(">> ListSetattr")
985ee('del l.locked')
986ee('l.locked = FailingTrue()')
987ee('l.xxx = True')
988cb.append("> Function")
989cb.append(">> FunctionConstructor")
990ee('vim.Function("123")')
991ee('vim.Function("xxx_non_existent_function_xxx")')
992ee('vim.Function("xxx#non#existent#function#xxx")')
993cb.append(">> FunctionCall")
994convertfrompyobject_test('f(%s)')
995convertfrompymapping_test('fd(self=%s)')
996cb.append("> TabPage")
997cb.append(">> TabPageAttr")
998ee('vim.current.tabpage.xxx')
999cb.append("> TabList")
1000cb.append(">> TabListItem")
1001ee('vim.tabpages[1000]')
1002cb.append("> Window")
1003cb.append(">> WindowAttr")
1004ee('vim.current.window.xxx')
1005cb.append(">> WindowSetattr")
1006ee('vim.current.window.buffer = 0')
1007ee('vim.current.window.cursor = (10000000000, 100000000)')
1008ee('vim.current.window.cursor = True')
1009ee('vim.current.window.height = "abc"')
1010ee('vim.current.window.width = "abc"')
1011ee('vim.current.window.xxxxxx = True')
1012cb.append("> WinList")
1013cb.append(">> WinListItem")
1014ee('vim.windows[1000]')
1015cb.append("> Buffer")
1016cb.append(">> StringToLine (indirect)")
1017ee('vim.current.buffer[0] = "\\na"')
1018cb.append(">> SetBufferLine (indirect)")
1019ee('vim.current.buffer[0] = True')
1020cb.append(">> SetBufferLines (indirect)")
1021ee('vim.current.buffer[:] = True')
1022ee('vim.current.buffer[:] = ["\\na", "bc"]')
1023cb.append(">> InsertBufferLines (indirect)")
1024ee('vim.current.buffer.append(None)')
1025ee('vim.current.buffer.append(["\\na", "bc"])')
1026ee('vim.current.buffer.append("\\nbc")')
1027cb.append(">> RBItem")
1028ee('vim.current.buffer[10000000000]')
1029cb.append(">> RBAsItem")
1030ee('vim.current.buffer[10000000000] = ""')
1031cb.append(">> BufferAttr")
1032ee('vim.current.buffer.xxx')
1033cb.append(">> BufferSetattr")
1034ee('vim.current.buffer.name = True')
1035ee('vim.current.buffer.xxx = True')
1036cb.append(">> BufferMark")
1037ee('vim.current.buffer.mark(0)')
1038ee('vim.current.buffer.mark("abc")')
1039ee('vim.current.buffer.mark("!")')
1040cb.append(">> BufferRange")
1041ee('vim.current.buffer.range(1, 2, 3)')
1042cb.append("> BufMap")
1043cb.append(">> BufMapItem")
1044ee('vim.buffers[None]')
1045ee('vim.buffers[100000000]')
1046cb.append("> Current")
1047cb.append(">> CurrentGetattr")
1048ee('vim.current.xxx')
1049cb.append(">> CurrentSetattr")
1050ee('vim.current.line = True')
1051ee('vim.current.buffer = True')
1052ee('vim.current.window = True')
1053ee('vim.current.tabpage = True')
1054ee('vim.current.xxx = True')
1055EOF
1056:"
1057:" Test exceptions
1058:fun Exe(e)
1059: execute a:e
1060:endfun
1061py << EOF
Bram Moolenaara7b64ce2013-05-21 20:40:40 +02001062Exe = vim.bindeval('function("Exe")')
1063ee('vim.command("throw \'abc\'")')
1064ee('Exe("throw \'def\'")')
1065ee('vim.eval("Exe(\'throw \'\'ghi\'\'\')")')
1066ee('vim.eval("Exe(\'echoerr \'\'jkl\'\'\')")')
1067ee('vim.eval("Exe(\'xxx_non_existent_command_xxx\')")')
1068ee('vim.bindeval("Exe(\'xxx_non_existent_command_xxx\')")')
1069EOF
Bram Moolenaardb913952012-06-29 12:54:53 +02001070:endfun
1071:"
1072:call Test()
1073:"
1074:delfunc Test
1075:call garbagecollect(1)
1076:"
1077:/^start:/,$wq! test.out
Bram Moolenaar66b79852012-09-21 14:00:35 +02001078:call getchar()
Bram Moolenaardb913952012-06-29 12:54:53 +02001079ENDTEST
1080
1081start: