blob: 535a1437827cc497a0df96bc1e25e3155d94b8cb [file] [log] [blame]
Bram Moolenaardb913952012-06-29 12:54:53 +02001Tests for various python features. vim: set ft=vim :
2
3STARTTEST
4:so small.vim
Bram Moolenaar9f3685a2013-06-12 14:20:36 +02005:set noswapfile
Bram Moolenaardb913952012-06-29 12:54:53 +02006:if !has('python3') | e! test.ok | wq! test.out | endif
Bram Moolenaarc24c1ac2013-05-16 20:47:56 +02007:lang C
Bram Moolenaardb913952012-06-29 12:54:53 +02008:fun Test()
Bram Moolenaar841fbd22013-06-23 14:37:07 +02009:py3 import vim
Bram Moolenaardb913952012-06-29 12:54:53 +020010:let l = []
11:py3 l=vim.bindeval('l')
12:py3 f=vim.bindeval('function("strlen")')
13:" Extending List directly with different types
14:py3 l+=[1, "as'd", [1, 2, f, {'a': 1}]]
15:$put =string(l)
16:$put =string(l[-1])
17:try
18: $put =string(l[-4])
19:catch
20: $put =v:exception[:13]
21:endtry
22:" List assignment
23:py3 l[0]=0
24:$put =string(l)
25:py3 l[-2]=f
26:$put =string(l)
27:"
28:" Extending Dictionary directly with different types
29:let d = {}
Bram Moolenaar355fd9b2013-05-30 13:14:13 +020030:fun d.f()
31: return 1
32:endfun
Bram Moolenaara9922d62013-05-30 13:01:18 +020033py3 << EOF
34d=vim.bindeval('d')
35d['1']='asd'
Bram Moolenaar2d5f38f2014-02-11 18:47:27 +010036d.update() # Must not do anything, including throwing errors
Bram Moolenaara9922d62013-05-30 13:01:18 +020037d.update(b=[1, 2, f])
38d.update((('-1', {'a': 1}),))
39d.update({'0': -1})
40dk = d.keys()
41dv = d.values()
42di = d.items()
43dk.sort(key=repr)
44dv.sort(key=repr)
45di.sort(key=repr)
46EOF
Bram Moolenaar355fd9b2013-05-30 13:14:13 +020047:$put =py3eval('d[''f''](self={})')
Bram Moolenaara9922d62013-05-30 13:01:18 +020048:$put =py3eval('repr(dk)')
49:$put =substitute(py3eval('repr(dv)'),'0x\x\+','','g')
50:$put =substitute(py3eval('repr(di)'),'0x\x\+','','g')
Bram Moolenaar355fd9b2013-05-30 13:14:13 +020051:for [key, Val] in sort(items(d))
52: $put =string(key) . ' : ' . string(Val)
53: unlet key Val
Bram Moolenaardb913952012-06-29 12:54:53 +020054:endfor
Bram Moolenaar841fbd22013-06-23 14:37:07 +020055:py3 del dk
56:py3 del di
57:py3 del dv
Bram Moolenaardb913952012-06-29 12:54:53 +020058:"
59:" removing items with del
60:py3 del l[2]
61:$put =string(l)
62:let l = range(8)
63:py3 l=vim.bindeval('l')
64:try
65: py3 del l[:3]
66: py3 del l[1:]
67:catch
68: $put =v:exception
69:endtry
70:$put =string(l)
71:"
72:py3 del d['-1']
Bram Moolenaar355fd9b2013-05-30 13:14:13 +020073:py3 del d['f']
Bram Moolenaara9922d62013-05-30 13:01:18 +020074:$put =string(py3eval('d.get(''b'', 1)'))
75:$put =string(py3eval('d.pop(''b'')'))
76:$put =string(py3eval('d.get(''b'', 1)'))
77:$put =string(py3eval('d.pop(''1'', 2)'))
78:$put =string(py3eval('d.pop(''1'', 2)'))
79:$put =py3eval('repr(d.has_key(''0''))')
80:$put =py3eval('repr(d.has_key(''1''))')
81:$put =py3eval('repr(''0'' in d)')
82:$put =py3eval('repr(''1'' in d)')
83:$put =py3eval('repr(list(iter(d)))')
Bram Moolenaardb913952012-06-29 12:54:53 +020084:$put =string(d)
Bram Moolenaarde71b562013-06-02 17:41:54 +020085:$put =py3eval('repr(d.popitem())')
Bram Moolenaara9922d62013-05-30 13:01:18 +020086:$put =py3eval('repr(d.get(''0''))')
87:$put =py3eval('repr(list(iter(d)))')
Bram Moolenaardb913952012-06-29 12:54:53 +020088:"
89:" removing items out of range: silently skip items that don't exist
90:let l = [0, 1, 2, 3]
91:py3 l=vim.bindeval('l')
92:" The following two ranges delete nothing as they match empty list:
93:py3 del l[2:1]
94:$put =string(l)
95:py3 del l[2:2]
96:$put =string(l)
97:py3 del l[2:3]
98:$put =string(l)
99:let l = [0, 1, 2, 3]
100:py3 l=vim.bindeval('l')
101:py3 del l[2:4]
102:$put =string(l)
103:let l = [0, 1, 2, 3]
104:py3 l=vim.bindeval('l')
105:py3 del l[2:5]
106:$put =string(l)
107:let l = [0, 1, 2, 3]
108:py3 l=vim.bindeval('l')
109:py3 del l[2:6]
110:$put =string(l)
111:let l = [0, 1, 2, 3]
112:py3 l=vim.bindeval('l')
113:" The following two ranges delete nothing as they match empty list:
114:py3 del l[-1:2]
115:$put =string(l)
116:py3 del l[-2:2]
117:$put =string(l)
118:py3 del l[-3:2]
119:$put =string(l)
120:let l = [0, 1, 2, 3]
121:py3 l=vim.bindeval('l')
122:py3 del l[-4:2]
123:$put =string(l)
124:let l = [0, 1, 2, 3]
125:py3 l=vim.bindeval('l')
126:py3 del l[-5:2]
127:$put =string(l)
128:let l = [0, 1, 2, 3]
129:py3 l=vim.bindeval('l')
130:py3 del l[-6:2]
131:$put =string(l)
Bram Moolenaar063a46b2014-01-14 16:36:51 +0100132:let l = [0, 1, 2, 3]
133:py3 l=vim.bindeval('l')
134:py3 del l[::2]
135:$put =string(l)
136:let l = [0, 1, 2, 3]
137:py3 l=vim.bindeval('l')
138:py3 del l[3:0:-2]
139:$put =string(l)
140:let l = [0, 1, 2, 3]
141:py3 l=vim.bindeval('l')
142:py3 del l[2:4:-2]
143:$put =string(l)
Bram Moolenaardb913952012-06-29 12:54:53 +0200144:"
145:" Slice assignment to a list
146:let l = [0, 1, 2, 3]
147:py3 l=vim.bindeval('l')
148:py3 l[0:0]=['a']
149:$put =string(l)
150:let l = [0, 1, 2, 3]
151:py3 l=vim.bindeval('l')
152:py3 l[1:2]=['b']
153:$put =string(l)
154:let l = [0, 1, 2, 3]
155:py3 l=vim.bindeval('l')
156:py3 l[2:4]=['c']
157:$put =string(l)
158:let l = [0, 1, 2, 3]
159:py3 l=vim.bindeval('l')
160:py3 l[4:4]=['d']
161:$put =string(l)
162:let l = [0, 1, 2, 3]
163:py3 l=vim.bindeval('l')
164:py3 l[-1:2]=['e']
165:$put =string(l)
166:let l = [0, 1, 2, 3]
167:py3 l=vim.bindeval('l')
168:py3 l[-10:2]=['f']
169:$put =string(l)
170:let l = [0, 1, 2, 3]
171:py3 l=vim.bindeval('l')
172:py3 l[2:-10]=['g']
173:$put =string(l)
174:let l = []
175:py3 l=vim.bindeval('l')
176:py3 l[0:0]=['h']
177:$put =string(l)
Bram Moolenaar063a46b2014-01-14 16:36:51 +0100178:let l = range(8)
179:py3 l=vim.bindeval('l')
180:py3 l[2:6:2] = [10, 20]
181:$put =string(l)
182:let l = range(8)
183:py3 l=vim.bindeval('l')
184:py3 l[6:2:-2] = [10, 20]
185:$put =string(l)
186:let l = range(8)
187:py3 l=vim.bindeval('l')
188:py3 l[6:2] = ()
189:$put =string(l)
190:let l = range(8)
191:py3 l=vim.bindeval('l')
192:py3 l[6:2:1] = ()
193:$put =string(l)
194:let l = range(8)
195:py3 l=vim.bindeval('l')
196:py3 l[2:2:1] = ()
197:$put =string(l)
Bram Moolenaardb913952012-06-29 12:54:53 +0200198:"
199:" Locked variables
200:let l = [0, 1, 2, 3]
201:py3 l=vim.bindeval('l')
202:lockvar! l
203:py3 l[2]='i'
204:$put =string(l)
205:unlockvar! l
206:"
207:" Function calls
Bram Moolenaar9fee7d42013-11-28 17:04:43 +0100208py3 << EOF
209import sys
210import re
211
212py33_type_error_pattern = re.compile('^__call__\(\) takes (\d+) positional argument but (\d+) were given$')
213
214def ee(expr, g=globals(), l=locals()):
215 cb = vim.current.buffer
216 try:
217 try:
218 exec(expr, g, l)
219 except Exception as e:
220 if sys.version_info >= (3, 3) and e.__class__ is AttributeError and str(e).find('has no attribute')>=0 and not str(e).startswith("'vim."):
221 cb.append(expr + ':' + repr((e.__class__, AttributeError(str(e)[str(e).rfind(" '") + 2:-1]))))
222 elif sys.version_info >= (3, 3) and e.__class__ is ImportError and str(e).find('No module named \'') >= 0:
223 cb.append(expr + ':' + repr((e.__class__, ImportError(str(e).replace("'", '')))))
224 elif sys.version_info >= (3, 3) and e.__class__ is TypeError:
225 m = py33_type_error_pattern.search(str(e))
226 if m:
227 msg = '__call__() takes exactly {0} positional argument ({1} given)'.format(m.group(1), m.group(2))
228 cb.append(expr + ':' + repr((e.__class__, TypeError(msg))))
229 else:
230 cb.append(expr + ':' + repr((e.__class__, e)))
Bram Moolenaar29e19512016-02-28 20:13:18 +0100231 elif sys.version_info >= (3, 5) and e.__class__ is ValueError and str(e) == 'embedded null byte':
232 msg = cb.append(expr + ':' + repr((TypeError, TypeError('expected bytes with no null'))))
Bram Moolenaar9fee7d42013-11-28 17:04:43 +0100233 else:
234 cb.append(expr + ':' + repr((e.__class__, e)))
235 else:
236 cb.append(expr + ':NOT FAILED')
237 except Exception as e:
238 cb.append(expr + '::' + repr((e.__class__, e)))
239EOF
Bram Moolenaar841fbd22013-06-23 14:37:07 +0200240:fun New(...)
241: return ['NewStart']+a:000+['NewEnd']
242:endfun
243:fun DictNew(...) dict
244: return ['DictNewStart']+a:000+['DictNewEnd', self]
245:endfun
Bram Moolenaardb913952012-06-29 12:54:53 +0200246:let l=[function('New'), function('DictNew')]
247:py3 l=vim.bindeval('l')
248:py3 l.extend(list(l[0](1, 2, 3)))
249:$put =string(l)
250:py3 l.extend(list(l[1](1, 2, 3, self={'a': 'b'})))
251:$put =string(l)
252:py3 l+=[l[0].name]
253:$put =string(l)
Bram Moolenaar9fee7d42013-11-28 17:04:43 +0100254:py3 ee('l[1](1, 2, 3)')
Bram Moolenaar355fd9b2013-05-30 13:14:13 +0200255:py3 f=l[0]
Bram Moolenaardb913952012-06-29 12:54:53 +0200256:delfunction New
Bram Moolenaar9fee7d42013-11-28 17:04:43 +0100257:py3 ee('f(1, 2, 3)')
Bram Moolenaardb913952012-06-29 12:54:53 +0200258:if has('float')
259: let l=[0.0]
260: py3 l=vim.bindeval('l')
261: py3 l.extend([0.0])
262: $put =string(l)
263:else
264: $put ='[0.0, 0.0]'
265:endif
Bram Moolenaarc11073c2012-09-05 19:17:42 +0200266:let messages=[]
Bram Moolenaar841fbd22013-06-23 14:37:07 +0200267:delfunction DictNew
268py3 <<EOF
Bram Moolenaar29e19512016-02-28 20:13:18 +0100269import sys
Bram Moolenaarc11073c2012-09-05 19:17:42 +0200270d=vim.bindeval('{}')
271m=vim.bindeval('messages')
Bram Moolenaara9922d62013-05-30 13:01:18 +0200272def em(expr, g=globals(), l=locals()):
273 try:
274 exec(expr, g, l)
275 except Exception as e:
Bram Moolenaar29e19512016-02-28 20:13:18 +0100276 if sys.version_info >= (3, 5) and e.__class__ is ValueError and str(e) == 'embedded null byte':
277 m.extend([TypeError.__name__])
278 else:
279 m.extend([e.__class__.__name__])
Bram Moolenaarc11073c2012-09-05 19:17:42 +0200280
Bram Moolenaar841fbd22013-06-23 14:37:07 +0200281em('d["abc1"]')
282em('d["abc1"]="\\0"')
283em('d["abc1"]=vim')
Bram Moolenaara9922d62013-05-30 13:01:18 +0200284em('d[""]=1')
285em('d["a\\0b"]=1')
286em('d[b"a\\0b"]=1')
Bram Moolenaarc11073c2012-09-05 19:17:42 +0200287
Bram Moolenaar841fbd22013-06-23 14:37:07 +0200288em('d.pop("abc1")')
Bram Moolenaarde71b562013-06-02 17:41:54 +0200289em('d.popitem()')
Bram Moolenaar841fbd22013-06-23 14:37:07 +0200290del em
291del m
Bram Moolenaarc11073c2012-09-05 19:17:42 +0200292EOF
293:$put =messages
Bram Moolenaar66b79852012-09-21 14:00:35 +0200294:unlet messages
295:" locked and scope attributes
296:let d={} | let dl={} | lockvar dl
297:for s in split("d dl v: g:")
298: let name=tr(s, ':', 's')
299: execute 'py3 '.name.'=vim.bindeval("'.s.'")'
300: let toput=s.' : '.join(map(['locked', 'scope'], 'v:val.":".py3eval(name.".".v:val)'), ';')
301: $put =toput
302:endfor
Bram Moolenaar841fbd22013-06-23 14:37:07 +0200303:silent! let d.abc2=1
304:silent! let dl.abc3=1
Bram Moolenaar66b79852012-09-21 14:00:35 +0200305:py3 d.locked=True
306:py3 dl.locked=False
307:silent! let d.def=1
308:silent! let dl.def=1
309:put ='d:'.string(d)
310:put ='dl:'.string(dl)
311:unlet d dl
312:
313:let l=[] | let ll=[] | lockvar ll
314:for s in split("l ll")
315: let name=tr(s, ':', 's')
316: execute 'py3 '.name.'=vim.bindeval("'.s.'")'
317: let toput=s.' : locked:'.py3eval(name.'.locked')
318: $put =toput
319:endfor
320:silent! call extend(l, [0])
321:silent! call extend(ll, [0])
322:py3 l.locked=True
323:py3 ll.locked=False
324:silent! call extend(l, [1])
325:silent! call extend(ll, [1])
326:put ='l:'.string(l)
327:put ='ll:'.string(ll)
328:unlet l ll
Bram Moolenaardb913952012-06-29 12:54:53 +0200329:"
330:" py3eval()
331:let l=py3eval('[0, 1, 2]')
332:$put =string(l)
333:let d=py3eval('{"a": "b", "c": 1, "d": ["e"]}')
334:$put =sort(items(d))
Bram Moolenaar77324fc2016-01-17 22:37:03 +0100335:let v:errmsg = ''
336:$put ='py3eval(\"None\") = ' . py3eval('None') . v:errmsg
Bram Moolenaardb913952012-06-29 12:54:53 +0200337:if has('float')
338: let f=py3eval('0.0')
339: $put =string(f)
340:else
341: $put ='0.0'
342:endif
Bram Moolenaarc11073c2012-09-05 19:17:42 +0200343:" Invalid values:
344:for e in ['"\0"', '{"\0": 1}', 'undefined_name', 'vim']
345: try
346: let v=py3eval(e)
347: catch
348: let toput=e.":\t".v:exception[:13]
349: $put =toput
350: endtry
351:endfor
Bram Moolenaar76d711c2013-02-13 14:17:08 +0100352:"
353:" threading
354:let l = [0]
355:py3 l=vim.bindeval('l')
Bram Moolenaardee2e312013-06-23 16:35:47 +0200356py3 <<EOF
Bram Moolenaar76d711c2013-02-13 14:17:08 +0100357import threading
358import time
359
360class T(threading.Thread):
361 def __init__(self):
362 threading.Thread.__init__(self)
363 self.t = 0
364 self.running = True
365
366 def run(self):
367 while self.running:
368 self.t += 1
369 time.sleep(0.1)
370
371t = T()
Bram Moolenaar841fbd22013-06-23 14:37:07 +0200372del T
Bram Moolenaar76d711c2013-02-13 14:17:08 +0100373t.start()
374EOF
375:sleep 1
376:py3 t.running = False
377:py3 t.join()
Bram Moolenaar52f6ae12015-12-29 16:34:06 +0100378:" Check if the background thread is working. Count should be 10, but on a
379:" busy system (AppVeyor) it can be much lower.
380:py3 l[0] = t.t > 4
Bram Moolenaar841fbd22013-06-23 14:37:07 +0200381:py3 del time
382:py3 del threading
Bram Moolenaar9fee7d42013-11-28 17:04:43 +0100383:py3 del t
Bram Moolenaar76d711c2013-02-13 14:17:08 +0100384:$put =string(l)
385:"
386:" settrace
387:let l = []
388:py3 l=vim.bindeval('l')
Bram Moolenaardee2e312013-06-23 16:35:47 +0200389py3 <<EOF
Bram Moolenaar76d711c2013-02-13 14:17:08 +0100390import sys
391
392def traceit(frame, event, arg):
393 global l
394 if event == "line":
395 l += [frame.f_lineno]
396 return traceit
397
398def trace_main():
399 for i in range(5):
400 pass
401EOF
402:py3 sys.settrace(traceit)
403:py3 trace_main()
Bram Moolenaardee2e312013-06-23 16:35:47 +0200404:py3 sys.settrace(None)
Bram Moolenaar841fbd22013-06-23 14:37:07 +0200405:py3 del traceit
406:py3 del trace_main
Bram Moolenaar76d711c2013-02-13 14:17:08 +0100407:$put =string(l)
Bram Moolenaar230bb3f2013-04-24 14:07:45 +0200408:"
Bram Moolenaar063a46b2014-01-14 16:36:51 +0100409:" Slice
410:py3 ll = vim.bindeval('[0, 1, 2, 3, 4, 5]')
411:py3 l = ll[:4]
412:$put =string(py3eval('l'))
413:py3 l = ll[2:]
414:$put =string(py3eval('l'))
415:py3 l = ll[:-4]
416:$put =string(py3eval('l'))
417:py3 l = ll[-2:]
418:$put =string(py3eval('l'))
419:py3 l = ll[2:4]
420:$put =string(py3eval('l'))
421:py3 l = ll[4:2]
422:$put =string(py3eval('l'))
423:py3 l = ll[-4:-2]
424:$put =string(py3eval('l'))
425:py3 l = ll[-2:-4]
426:$put =string(py3eval('l'))
427:py3 l = ll[:]
428:$put =string(py3eval('l'))
429:py3 l = ll[0:6]
430:$put =string(py3eval('l'))
431:py3 l = ll[-10:10]
432:$put =string(py3eval('l'))
433:py3 l = ll[4:2:-1]
434:$put =string(py3eval('l'))
435:py3 l = ll[::2]
436:$put =string(py3eval('l'))
437:py3 l = ll[4:2:1]
438:$put =string(py3eval('l'))
439:py3 del l
440:"
Bram Moolenaar230bb3f2013-04-24 14:07:45 +0200441:" Vars
442:let g:foo = 'bac'
Bram Moolenaar841fbd22013-06-23 14:37:07 +0200443:let w:abc3 = 'def'
Bram Moolenaar230bb3f2013-04-24 14:07:45 +0200444:let b:baz = 'bar'
Bram Moolenaara4720012013-05-15 16:27:37 +0200445:let t:bar = 'jkl'
Bram Moolenaar230bb3f2013-04-24 14:07:45 +0200446:try
447: throw "Abc"
448:catch
449: put =py3eval('vim.vvars[''exception'']')
450:endtry
451:put =py3eval('vim.vars[''foo'']')
Bram Moolenaar841fbd22013-06-23 14:37:07 +0200452:put =py3eval('vim.current.window.vars[''abc3'']')
Bram Moolenaar230bb3f2013-04-24 14:07:45 +0200453:put =py3eval('vim.current.buffer.vars[''baz'']')
Bram Moolenaara4720012013-05-15 16:27:37 +0200454:put =py3eval('vim.current.tabpage.vars[''bar'']')
Bram Moolenaar84e0f6c2013-05-06 03:52:55 +0200455:"
456:" Options
457:" paste: boolean, global
458:" previewheight number, global
459:" operatorfunc: string, global
460:" number: boolean, window-local
461:" numberwidth: number, window-local
462:" colorcolumn: string, window-local
463:" statusline: string, window-local/global
464:" autoindent: boolean, buffer-local
Bram Moolenaar55b8ad32013-05-17 13:38:04 +0200465:" shiftwidth: number, buffer-local
Bram Moolenaar84e0f6c2013-05-06 03:52:55 +0200466:" omnifunc: string, buffer-local
467:" preserveindent: boolean, buffer-local/global
468:" path: string, buffer-local/global
469:let g:bufs=[bufnr('%')]
470:new
471:let g:bufs+=[bufnr('%')]
472:vnew
473:let g:bufs+=[bufnr('%')]
474:wincmd j
475:vnew
476:let g:bufs+=[bufnr('%')]
477:wincmd l
478:fun RecVars(opt)
479: let gval =string(eval('&g:'.a:opt))
480: let wvals=join(map(range(1, 4), 'v:val.":".string(getwinvar(v:val, "&".a:opt))'))
481: let bvals=join(map(copy(g:bufs), 'v:val.":".string(getbufvar(v:val, "&".a:opt))'))
482: put =' G: '.gval
483: put =' W: '.wvals
484: put =' B: '.wvals
485:endfun
486py3 << EOF
487def e(s, g=globals(), l=locals()):
488 try:
489 exec(s, g, l)
490 except Exception as e:
Bram Moolenaara7b64ce2013-05-21 20:40:40 +0200491 vim.command('return ' + repr(e.__class__.__name__))
Bram Moolenaar84e0f6c2013-05-06 03:52:55 +0200492
493def ev(s, g=globals(), l=locals()):
494 try:
495 return eval(s, g, l)
496 except Exception as e:
Bram Moolenaara7b64ce2013-05-21 20:40:40 +0200497 vim.command('let exc=' + repr(e.__class__.__name__))
Bram Moolenaar84e0f6c2013-05-06 03:52:55 +0200498 return 0
499EOF
Bram Moolenaar841fbd22013-06-23 14:37:07 +0200500:fun E(s)
Bram Moolenaar84e0f6c2013-05-06 03:52:55 +0200501: python3 e(vim.eval('a:s'))
Bram Moolenaar841fbd22013-06-23 14:37:07 +0200502:endfun
503:fun Ev(s)
Bram Moolenaara7b64ce2013-05-21 20:40:40 +0200504: let r=py3eval('ev(vim.eval("a:s"))')
505: if exists('exc')
506: throw exc
507: endif
508: return r
Bram Moolenaar841fbd22013-06-23 14:37:07 +0200509:endfun
Bram Moolenaar84e0f6c2013-05-06 03:52:55 +0200510:py3 gopts1=vim.options
511:py3 wopts1=vim.windows[2].options
512:py3 wopts2=vim.windows[0].options
513:py3 wopts3=vim.windows[1].options
514:py3 bopts1=vim.buffers[vim.bindeval("g:bufs")[2]].options
515:py3 bopts2=vim.buffers[vim.bindeval("g:bufs")[1]].options
516:py3 bopts3=vim.buffers[vim.bindeval("g:bufs")[0]].options
Bram Moolenaar1028f4d2014-01-14 16:55:00 +0100517:$put ='wopts iters equal: '.py3eval('list(wopts1) == list(wopts2)')
518:$put ='bopts iters equal: '.py3eval('list(bopts1) == list(bopts2)')
519:py3 gset=set(iter(gopts1))
520:py3 wset=set(iter(wopts1))
521:py3 bset=set(iter(bopts1))
Bram Moolenaar04188112013-06-01 20:32:12 +0200522:set path=.,..,,
Bram Moolenaar84e0f6c2013-05-06 03:52:55 +0200523:let lst=[]
524:let lst+=[['paste', 1, 0, 1, 2, 1, 1, 0 ]]
525:let lst+=[['previewheight', 5, 1, 6, 'a', 0, 1, 0 ]]
526:let lst+=[['operatorfunc', 'A', 'B', 'C', 2, 0, 1, 0 ]]
527:let lst+=[['number', 0, 1, 1, 0, 1, 0, 1 ]]
528:let lst+=[['numberwidth', 2, 3, 5, -100, 0, 0, 1 ]]
Bram Moolenaar841fbd22013-06-23 14:37:07 +0200529:let lst+=[['colorcolumn', '+1', '+2', '+3', 'abc4', 0, 0, 1 ]]
Bram Moolenaar84e0f6c2013-05-06 03:52:55 +0200530:let lst+=[['statusline', '1', '2', '4', 0, 0, 1, 1 ]]
531:let lst+=[['autoindent', 0, 1, 1, 2, 1, 0, 2 ]]
Bram Moolenaar55b8ad32013-05-17 13:38:04 +0200532:let lst+=[['shiftwidth', 0, 2, 1, 3, 0, 0, 2 ]]
Bram Moolenaar84e0f6c2013-05-06 03:52:55 +0200533:let lst+=[['omnifunc', 'A', 'B', 'C', 1, 0, 0, 2 ]]
534:let lst+=[['preserveindent', 0, 1, 1, 2, 1, 1, 2 ]]
535:let lst+=[['path', '.,,', ',,', '.', 0, 0, 1, 2 ]]
536:for [oname, oval1, oval2, oval3, invval, bool, global, local] in lst
537: py3 oname=vim.eval('oname')
538: py3 oval1=vim.bindeval('oval1')
539: py3 oval2=vim.bindeval('oval2')
540: py3 oval3=vim.bindeval('oval3')
541: if invval is 0 || invval is 1
542: py3 invval=bool(vim.bindeval('invval'))
543: else
544: py3 invval=vim.bindeval('invval')
545: endif
546: if bool
547: py3 oval1=bool(oval1)
548: py3 oval2=bool(oval2)
549: py3 oval3=bool(oval3)
550: endif
551: put ='>>> '.oname
Bram Moolenaar1028f4d2014-01-14 16:55:00 +0100552: $put =' g/w/b:'.py3eval('oname in gset').'/'.py3eval('oname in wset').'/'.py3eval('oname in bset')
553: $put =' g/w/b (in):'.py3eval('oname in gopts1').'/'.py3eval('oname in wopts1').'/'.py3eval('oname in bopts1')
Bram Moolenaar84e0f6c2013-05-06 03:52:55 +0200554: for v in ['gopts1', 'wopts1', 'bopts1']
555: try
556: put =' p/'.v.': '.Ev('repr('.v.'['''.oname.'''])')
557: catch
558: put =' p/'.v.'! '.v:exception
559: endtry
Bram Moolenaara7b64ce2013-05-21 20:40:40 +0200560: let r=E(v.'['''.oname.''']=invval')
561: if r isnot 0
562: put =' inv: '.string(invval).'! '.r
563: endif
Bram Moolenaar84e0f6c2013-05-06 03:52:55 +0200564: for vv in (v is# 'gopts1' ? [v] : [v, v[:-2].'2', v[:-2].'3'])
565: let val=substitute(vv, '^.opts', 'oval', '')
Bram Moolenaara7b64ce2013-05-21 20:40:40 +0200566: let r=E(vv.'['''.oname.''']='.val)
567: if r isnot 0
568: put =' '.vv.'! '.r
569: endif
Bram Moolenaar84e0f6c2013-05-06 03:52:55 +0200570: endfor
571: endfor
572: call RecVars(oname)
573: for v in ['wopts3', 'bopts3']
Bram Moolenaara7b64ce2013-05-21 20:40:40 +0200574: let r=E('del '.v.'["'.oname.'"]')
575: if r isnot 0
576: put =' del '.v.'! '.r
577: endif
Bram Moolenaar84e0f6c2013-05-06 03:52:55 +0200578: endfor
579: call RecVars(oname)
Bram Moolenaar84e0f6c2013-05-06 03:52:55 +0200580:endfor
Bram Moolenaar841fbd22013-06-23 14:37:07 +0200581:delfunction RecVars
582:delfunction E
583:delfunction Ev
584:py3 del ev
585:py3 del e
Bram Moolenaar84e0f6c2013-05-06 03:52:55 +0200586:only
Bram Moolenaarb6c589a2013-05-15 14:39:52 +0200587:for buf in g:bufs[1:]
588: execute 'bwipeout!' buf
589:endfor
Bram Moolenaar841fbd22013-06-23 14:37:07 +0200590:py3 del gopts1
591:py3 del wopts1
592:py3 del wopts2
593:py3 del wopts3
594:py3 del bopts1
595:py3 del bopts2
596:py3 del bopts3
597:py3 del oval1
598:py3 del oval2
599:py3 del oval3
600:py3 del oname
601:py3 del invval
Bram Moolenaarbd80f352013-05-12 21:16:23 +0200602:"
603:" Test buffer object
604:vnew
605:put ='First line'
606:put ='Second line'
607:put ='Third line'
608:1 delete _
609:py3 b=vim.current.buffer
610:wincmd w
611:mark a
Bram Moolenaare9ba5162013-05-29 22:02:22 +0200612:augroup BUFS
613: autocmd BufFilePost * python3 cb.append(vim.eval('expand("<abuf>")') + ':BufFilePost:' + vim.eval('bufnr("%")'))
614: autocmd BufFilePre * python3 cb.append(vim.eval('expand("<abuf>")') + ':BufFilePre:' + vim.eval('bufnr("%")'))
615:augroup END
Bram Moolenaarbd80f352013-05-12 21:16:23 +0200616py3 << EOF
617cb = vim.current.buffer
618# Tests BufferAppend and BufferItem
619cb.append(b[0])
620# Tests BufferSlice and BufferAssSlice
Bram Moolenaar841fbd22013-06-23 14:37:07 +0200621cb.append('abc5') # Will be overwritten
Bram Moolenaarbd80f352013-05-12 21:16:23 +0200622cb[-1:] = b[:-2]
623# Test BufferLength and BufferAssSlice
624cb.append('def') # Will not be overwritten
625cb[len(cb):] = b[:]
626# Test BufferAssItem and BufferMark
627cb.append('ghi') # Will be overwritten
628cb[-1] = repr((len(cb) - cb.mark('a')[0], cb.mark('a')[1]))
629# Test BufferRepr
630cb.append(repr(cb) + repr(b))
631# Modify foreign buffer
632b.append('foo')
633b[0]='bar'
634b[0:0]=['baz']
635vim.command('call append("$", getbufline(%i, 1, "$"))' % b.number)
Bram Moolenaare9ba5162013-05-29 22:02:22 +0200636# Test assigning to name property
Bram Moolenaar04188112013-06-01 20:32:12 +0200637import os
Bram Moolenaare9ba5162013-05-29 22:02:22 +0200638old_name = cb.name
639cb.name = 'foo'
Bram Moolenaar04188112013-06-01 20:32:12 +0200640cb.append(cb.name[-11:].replace(os.path.sep, '/'))
Bram Moolenaare9ba5162013-05-29 22:02:22 +0200641b.name = 'bar'
Bram Moolenaar04188112013-06-01 20:32:12 +0200642cb.append(b.name[-11:].replace(os.path.sep, '/'))
Bram Moolenaare9ba5162013-05-29 22:02:22 +0200643cb.name = old_name
Bram Moolenaar04188112013-06-01 20:32:12 +0200644cb.append(cb.name[-17:].replace(os.path.sep, '/'))
Bram Moolenaar841fbd22013-06-23 14:37:07 +0200645del old_name
Bram Moolenaarbd80f352013-05-12 21:16:23 +0200646# Test CheckBuffer
Bram Moolenaare9ba5162013-05-29 22:02:22 +0200647for _b in vim.buffers:
648 if _b is not cb:
649 vim.command('bwipeout! ' + str(_b.number))
650del _b
Bram Moolenaar9e822c02013-05-29 22:15:30 +0200651cb.append('valid: b:%s, cb:%s' % (repr(b.valid), repr(cb.valid)))
Bram Moolenaar841fbd22013-06-23 14:37:07 +0200652for expr in ('b[1]','b[:] = ["A", "B"]','b[:]','b.append("abc6")'):
Bram Moolenaarbd80f352013-05-12 21:16:23 +0200653 try:
654 exec(expr)
655 except vim.error:
656 pass
657 else:
658 # Usually a SEGV here
659 # Should not happen in any case
660 cb.append('No exception for ' + expr)
Bram Moolenaare9ba5162013-05-29 22:02:22 +0200661vim.command('cd .')
Bram Moolenaar841fbd22013-06-23 14:37:07 +0200662del b
Bram Moolenaarbd80f352013-05-12 21:16:23 +0200663EOF
Bram Moolenaarb6c589a2013-05-15 14:39:52 +0200664:"
665:" Test vim.buffers object
666:set hidden
667:edit a
668:buffer #
669:edit b
670:buffer #
671:edit c
672:buffer #
673py3 << EOF
674# Check GCing iterator that was not fully exhausted
675i = iter(vim.buffers)
676cb.append('i:' + str(next(i)))
Bram Moolenaar038e5d42014-03-12 15:26:40 +0100677# and also check creating more than one iterator at a time
Bram Moolenaarb6c589a2013-05-15 14:39:52 +0200678i2 = iter(vim.buffers)
679cb.append('i2:' + str(next(i2)))
680cb.append('i:' + str(next(i)))
681# The following should trigger GC and not cause any problems
682del i
683del i2
684i3 = iter(vim.buffers)
685cb.append('i3:' + str(next(i3)))
686del i3
687
688prevnum = 0
689for b in vim.buffers:
690 # Check buffer order
691 if prevnum >= b.number:
692 cb.append('!!! Buffer numbers not in strictly ascending order')
693 # Check indexing: vim.buffers[number].number == number
694 cb.append(str(b.number) + ':' + repr(vim.buffers[b.number]) + '=' + repr(b))
695 prevnum = b.number
Bram Moolenaar841fbd22013-06-23 14:37:07 +0200696del prevnum
Bram Moolenaarb6c589a2013-05-15 14:39:52 +0200697
698cb.append(str(len(vim.buffers)))
699
700bnums = list(map(lambda b: b.number, vim.buffers))[1:]
701
702# Test wiping out buffer with existing iterator
703i4 = iter(vim.buffers)
704cb.append('i4:' + str(next(i4)))
705vim.command('bwipeout! ' + str(bnums.pop(0)))
706try:
707 next(i4)
708except vim.error:
709 pass
710else:
711 cb.append('!!!! No vim.error')
712i4 = iter(vim.buffers)
713vim.command('bwipeout! ' + str(bnums.pop(-1)))
714vim.command('bwipeout! ' + str(bnums.pop(-1)))
715cb.append('i4:' + str(next(i4)))
716try:
717 next(i4)
718except StopIteration:
719 cb.append('StopIteration')
Bram Moolenaar841fbd22013-06-23 14:37:07 +0200720del i4
721del bnums
Bram Moolenaarb6c589a2013-05-15 14:39:52 +0200722EOF
Bram Moolenaara4720012013-05-15 16:27:37 +0200723:"
724:" Test vim.{tabpage,window}list and vim.{tabpage,window} objects
725:tabnew 0
726:tabnew 1
727:vnew a.1
728:tabnew 2
729:vnew a.2
730:vnew b.2
731:vnew c.2
732py3 << EOF
Bram Moolenaar2a0f3d32013-05-21 22:23:56 +0200733cb.append('Number of tabs: ' + str(len(vim.tabpages)))
734cb.append('Current tab pages:')
735
Bram Moolenaara4720012013-05-15 16:27:37 +0200736def W(w):
737 if '(unknown)' in repr(w):
738 return '<window object (unknown)>'
739 else:
740 return repr(w)
Bram Moolenaar2a0f3d32013-05-21 22:23:56 +0200741
742def Cursor(w, start=len(cb)):
743 if w.buffer is cb:
744 return repr((start - w.cursor[0], w.cursor[1]))
745 else:
746 return repr(w.cursor)
747
Bram Moolenaara4720012013-05-15 16:27:37 +0200748for t in vim.tabpages:
749 cb.append(' ' + repr(t) + '(' + str(t.number) + ')' + ': ' + str(len(t.windows)) + ' windows, current is ' + W(t.window))
750 cb.append(' Windows:')
751 for w in t.windows:
Bram Moolenaar2a0f3d32013-05-21 22:23:56 +0200752 cb.append(' ' + W(w) + '(' + str(w.number) + ')' + ': displays buffer ' + repr(w.buffer) + '; cursor is at ' + Cursor(w))
Bram Moolenaara4720012013-05-15 16:27:37 +0200753 # Other values depend on the size of the terminal, so they are checked partly:
754 for attr in ('height', 'row', 'width', 'col'):
755 try:
756 aval = getattr(w, attr)
757 if type(aval) is not int:
758 raise TypeError
759 if aval < 0:
760 raise ValueError
761 except Exception as e:
762 cb.append('!!!!!! Error while getting attribute ' + attr + ': ' + e.__class__.__name__)
Bram Moolenaar841fbd22013-06-23 14:37:07 +0200763 del aval
764 del attr
Bram Moolenaara4720012013-05-15 16:27:37 +0200765 w.cursor = (len(w.buffer), 0)
Bram Moolenaar841fbd22013-06-23 14:37:07 +0200766del W
767del Cursor
Bram Moolenaara4720012013-05-15 16:27:37 +0200768cb.append('Number of windows in current tab page: ' + str(len(vim.windows)))
769if list(vim.windows) != list(vim.current.tabpage.windows):
770 cb.append('!!!!!! Windows differ')
771EOF
772:"
773:" Test vim.current
774py3 << EOF
775def H(o):
776 return repr(o)
777cb.append('Current tab page: ' + repr(vim.current.tabpage))
778cb.append('Current window: ' + repr(vim.current.window) + ': ' + H(vim.current.window) + ' is ' + H(vim.current.tabpage.window))
779cb.append('Current buffer: ' + repr(vim.current.buffer) + ': ' + H(vim.current.buffer) + ' is ' + H(vim.current.window.buffer)+ ' is ' + H(vim.current.tabpage.window.buffer))
Bram Moolenaar841fbd22013-06-23 14:37:07 +0200780del H
Bram Moolenaara4720012013-05-15 16:27:37 +0200781# Assigning: fails
782try:
783 vim.current.window = vim.tabpages[0].window
784except ValueError:
785 cb.append('ValueError at assigning foreign tab window')
786
787for attr in ('window', 'tabpage', 'buffer'):
788 try:
789 setattr(vim.current, attr, None)
790 except TypeError:
791 cb.append('Type error at assigning None to vim.current.' + attr)
Bram Moolenaar841fbd22013-06-23 14:37:07 +0200792del attr
Bram Moolenaara4720012013-05-15 16:27:37 +0200793
794# Assigning: success
795vim.current.tabpage = vim.tabpages[-2]
796vim.current.buffer = cb
797vim.current.window = vim.windows[0]
798vim.current.window.cursor = (len(vim.current.buffer), 0)
799cb.append('Current tab page: ' + repr(vim.current.tabpage))
800cb.append('Current window: ' + repr(vim.current.window))
801cb.append('Current buffer: ' + repr(vim.current.buffer))
802cb.append('Current line: ' + repr(vim.current.line))
Bram Moolenaar9e822c02013-05-29 22:15:30 +0200803ws = list(vim.windows)
804ts = list(vim.tabpages)
Bram Moolenaara4720012013-05-15 16:27:37 +0200805for b in vim.buffers:
806 if b is not cb:
807 vim.command('bwipeout! ' + str(b.number))
Bram Moolenaar841fbd22013-06-23 14:37:07 +0200808del b
Bram Moolenaar9e822c02013-05-29 22:15:30 +0200809cb.append('w.valid: ' + repr([w.valid for w in ws]))
810cb.append('t.valid: ' + repr([t.valid for t in ts]))
Bram Moolenaar841fbd22013-06-23 14:37:07 +0200811del w
812del t
813del ts
814del ws
Bram Moolenaara4720012013-05-15 16:27:37 +0200815EOF
816:tabonly!
817:only!
Bram Moolenaarcac867a2013-05-21 19:50:34 +0200818:"
819:" Test types
820py3 << EOF
821for expr, attr in (
822 ('vim.vars', 'Dictionary'),
823 ('vim.options', 'Options'),
824 ('vim.bindeval("{}")', 'Dictionary'),
825 ('vim.bindeval("[]")', 'List'),
826 ('vim.bindeval("function(\'tr\')")', 'Function'),
827 ('vim.current.buffer', 'Buffer'),
828 ('vim.current.range', 'Range'),
829 ('vim.current.window', 'Window'),
830 ('vim.current.tabpage', 'TabPage'),
831):
832 cb.append(expr + ':' + attr + ':' + repr(type(eval(expr)) is getattr(vim, attr)))
Bram Moolenaar841fbd22013-06-23 14:37:07 +0200833del expr
834del attr
Bram Moolenaarcac867a2013-05-21 19:50:34 +0200835EOF
Bram Moolenaara7b64ce2013-05-21 20:40:40 +0200836:"
Bram Moolenaardd8aca62013-05-29 22:36:10 +0200837:" Test __dir__() method
838py3 << EOF
839for name, o in (
840 ('current', vim.current),
841 ('buffer', vim.current.buffer),
842 ('window', vim.current.window),
843 ('tabpage', vim.current.tabpage),
844 ('range', vim.current.range),
845 ('dictionary', vim.bindeval('{}')),
846 ('list', vim.bindeval('[]')),
847 ('function', vim.bindeval('function("tr")')),
848 ('output', sys.stdout),
849 ):
850 cb.append(name + ':' + ','.join(dir(o)))
851del name
852del o
853EOF
854:"
Bram Moolenaar78cddbe2013-05-30 13:05:58 +0200855:" Test vim.*.__new__
Bram Moolenaara9922d62013-05-30 13:01:18 +0200856:$put =string(py3eval('vim.Dictionary({})'))
857:$put =string(py3eval('vim.Dictionary(a=1)'))
858:$put =string(py3eval('vim.Dictionary(((''a'', 1),))'))
Bram Moolenaar78cddbe2013-05-30 13:05:58 +0200859:$put =string(py3eval('vim.List()'))
Bram Moolenaar841fbd22013-06-23 14:37:07 +0200860:$put =string(py3eval('vim.List(iter(''abc7''))'))
Bram Moolenaar355fd9b2013-05-30 13:14:13 +0200861:$put =string(py3eval('vim.Function(''tr'')'))
Bram Moolenaar01a7a722013-05-30 12:26:58 +0200862:"
863:" Test stdout/stderr
864:redir => messages
Bram Moolenaar841fbd22013-06-23 14:37:07 +0200865:py3 sys.stdout.write('abc8') ; sys.stdout.write('def')
866:py3 sys.stderr.write('abc9') ; sys.stderr.write('def')
867:py3 sys.stdout.writelines(iter('abcA'))
868:py3 sys.stderr.writelines(iter('abcB'))
Bram Moolenaar01a7a722013-05-30 12:26:58 +0200869:redir END
870:$put =string(substitute(messages, '\d\+', '', 'g'))
Bram Moolenaara9922d62013-05-30 13:01:18 +0200871:" Test subclassing
Bram Moolenaar355fd9b2013-05-30 13:14:13 +0200872:fun Put(...)
873: $put =string(a:000)
874: return a:000
875:endfun
Bram Moolenaara9922d62013-05-30 13:01:18 +0200876py3 << EOF
877class DupDict(vim.Dictionary):
878 def __setitem__(self, key, value):
879 super(DupDict, self).__setitem__(key, value)
880 super(DupDict, self).__setitem__('dup_' + key, value)
881dd = DupDict()
882dd['a'] = 'b'
Bram Moolenaar78cddbe2013-05-30 13:05:58 +0200883
884class DupList(vim.List):
885 def __getitem__(self, idx):
886 return [super(DupList, self).__getitem__(idx)] * 2
887
888dl = DupList()
Bram Moolenaar841fbd22013-06-23 14:37:07 +0200889dl2 = DupList(iter('abcC'))
Bram Moolenaar78cddbe2013-05-30 13:05:58 +0200890dl.extend(dl2[0])
Bram Moolenaar355fd9b2013-05-30 13:14:13 +0200891
892class DupFun(vim.Function):
893 def __call__(self, arg):
894 return super(DupFun, self).__call__(arg, arg)
895
896df = DupFun('Put')
Bram Moolenaara9922d62013-05-30 13:01:18 +0200897EOF
898:$put =string(sort(keys(py3eval('dd'))))
Bram Moolenaar78cddbe2013-05-30 13:05:58 +0200899:$put =string(py3eval('dl'))
900:$put =string(py3eval('dl2'))
Bram Moolenaar355fd9b2013-05-30 13:14:13 +0200901:$put =string(py3eval('df(2)'))
Bram Moolenaar841fbd22013-06-23 14:37:07 +0200902:$put =string(py3eval('dl') is# py3eval('dl'))
903:$put =string(py3eval('dd') is# py3eval('dd'))
904:$put =string(py3eval('df'))
905:delfunction Put
906py3 << EOF
907del DupDict
908del DupList
909del DupFun
910del dd
911del dl
912del dl2
913del df
914EOF
Bram Moolenaar01a7a722013-05-30 12:26:58 +0200915:"
Bram Moolenaarf4258302013-06-02 18:20:17 +0200916:" Test chdir
917py3 << EOF
918import os
919fnamemodify = vim.Function('fnamemodify')
920cb.append(str(fnamemodify('.', ':p:h:t')))
921cb.append(vim.eval('@%'))
922os.chdir('..')
Bram Moolenaar91376b62015-12-11 16:17:02 +0100923path = fnamemodify('.', ':p:h:t')
924if path != b'src':
925 # Running tests from a shadow directory, so move up another level
926 # This will result in @% looking like shadow/testdir/test87.in, hence the
927 # slicing to remove the leading path and path separator
928 os.chdir('..')
929 cb.append(str(fnamemodify('.', ':p:h:t')))
930 cb.append(vim.eval('@%')[len(path)+1:].replace(os.path.sep, '/'))
931 os.chdir(path)
932else:
933 cb.append(str(fnamemodify('.', ':p:h:t')))
934 cb.append(vim.eval('@%').replace(os.path.sep, '/'))
935del path
Bram Moolenaarf4258302013-06-02 18:20:17 +0200936os.chdir('testdir')
937cb.append(str(fnamemodify('.', ':p:h:t')))
938cb.append(vim.eval('@%'))
Bram Moolenaar841fbd22013-06-23 14:37:07 +0200939del fnamemodify
Bram Moolenaarf4258302013-06-02 18:20:17 +0200940EOF
941:"
Bram Moolenaar8600e402013-05-30 13:28:41 +0200942:" Test errors
943:fun F() dict
944:endfun
945:fun D()
946:endfun
947py3 << EOF
Bram Moolenaar8600e402013-05-30 13:28:41 +0200948d = vim.Dictionary()
Bram Moolenaar841fbd22013-06-23 14:37:07 +0200949ned = vim.Dictionary(foo='bar', baz='abcD')
Bram Moolenaar8600e402013-05-30 13:28:41 +0200950dl = vim.Dictionary(a=1)
951dl.locked = True
952l = vim.List()
Bram Moolenaar841fbd22013-06-23 14:37:07 +0200953ll = vim.List('abcE')
Bram Moolenaar8600e402013-05-30 13:28:41 +0200954ll.locked = True
Bram Moolenaar063a46b2014-01-14 16:36:51 +0100955nel = vim.List('abcO')
Bram Moolenaar8600e402013-05-30 13:28:41 +0200956f = vim.Function('string')
957fd = vim.Function('F')
958fdel = vim.Function('D')
959vim.command('delfunction D')
960
961def subexpr_test(expr, name, subexprs):
962 cb.append('>>> Testing %s using %s' % (name, expr))
963 for subexpr in subexprs:
964 ee(expr % subexpr)
965 cb.append('<<< Finished')
966
967def stringtochars_test(expr):
968 return subexpr_test(expr, 'StringToChars', (
969 '1', # Fail type checks
Bram Moolenaar96c7dfd2013-05-31 18:46:11 +0200970 'b"\\0"', # Fail PyString_AsStringAndSize(object, , NULL) check
971 '"\\0"', # Fail PyString_AsStringAndSize(bytes, , NULL) check
Bram Moolenaar8600e402013-05-30 13:28:41 +0200972 ))
973
974class Mapping(object):
975 def __init__(self, d):
976 self.d = d
977
978 def __getitem__(self, key):
979 return self.d[key]
980
981 def keys(self):
982 return self.d.keys()
983
984 def items(self):
985 return self.d.items()
986
987def convertfrompyobject_test(expr, recurse=True):
988 # pydict_to_tv
989 stringtochars_test(expr % '{%s : 1}')
990 if recurse:
Bram Moolenaar841fbd22013-06-23 14:37:07 +0200991 convertfrompyobject_test(expr % '{"abcF" : %s}', False)
Bram Moolenaar8600e402013-05-30 13:28:41 +0200992 # pymap_to_tv
993 stringtochars_test(expr % 'Mapping({%s : 1})')
994 if recurse:
Bram Moolenaar841fbd22013-06-23 14:37:07 +0200995 convertfrompyobject_test(expr % 'Mapping({"abcG" : %s})', False)
Bram Moolenaar8600e402013-05-30 13:28:41 +0200996 # pyseq_to_tv
997 iter_test(expr)
998 return subexpr_test(expr, 'ConvertFromPyObject', (
999 'None', # Not conversible
Bram Moolenaar78b59572013-06-02 18:54:21 +02001000 '{b"": 1}', # Empty key not allowed
1001 '{"": 1}', # Same, but with unicode object
Bram Moolenaar8600e402013-05-30 13:28:41 +02001002 'FailingMapping()', #
1003 'FailingMappingKey()', #
Bram Moolenaardee2e312013-06-23 16:35:47 +02001004 'FailingNumber()', #
Bram Moolenaar8600e402013-05-30 13:28:41 +02001005 ))
1006
1007def convertfrompymapping_test(expr):
1008 convertfrompyobject_test(expr)
1009 return subexpr_test(expr, 'ConvertFromPyMapping', (
1010 '[]',
1011 ))
1012
1013def iter_test(expr):
1014 return subexpr_test(expr, '*Iter*', (
1015 'FailingIter()',
1016 'FailingIterNext()',
1017 ))
1018
Bram Moolenaardee2e312013-06-23 16:35:47 +02001019def number_test(expr, natural=False, unsigned=False):
1020 if natural:
1021 unsigned = True
1022 return subexpr_test(expr, 'NumberToLong', (
1023 '[]',
1024 'None',
1025 ) + (('-1',) if unsigned else ())
1026 + (('0',) if natural else ()))
1027
Bram Moolenaar8600e402013-05-30 13:28:41 +02001028class FailingTrue(object):
1029 def __bool__(self):
Bram Moolenaardee2e312013-06-23 16:35:47 +02001030 raise NotImplementedError('bool')
Bram Moolenaar8600e402013-05-30 13:28:41 +02001031
1032class FailingIter(object):
1033 def __iter__(self):
Bram Moolenaardee2e312013-06-23 16:35:47 +02001034 raise NotImplementedError('iter')
Bram Moolenaar8600e402013-05-30 13:28:41 +02001035
1036class FailingIterNext(object):
1037 def __iter__(self):
1038 return self
1039
1040 def __next__(self):
Bram Moolenaardee2e312013-06-23 16:35:47 +02001041 raise NotImplementedError('next')
Bram Moolenaar8600e402013-05-30 13:28:41 +02001042
Bram Moolenaar063a46b2014-01-14 16:36:51 +01001043class FailingIterNextN(object):
1044 def __init__(self, n):
1045 self.n = n
1046
1047 def __iter__(self):
1048 return self
1049
1050 def __next__(self):
1051 if self.n:
1052 self.n -= 1
1053 return 1
1054 else:
1055 raise NotImplementedError('next N')
1056
Bram Moolenaar8600e402013-05-30 13:28:41 +02001057class FailingMappingKey(object):
1058 def __getitem__(self, item):
Bram Moolenaardee2e312013-06-23 16:35:47 +02001059 raise NotImplementedError('getitem:mappingkey')
Bram Moolenaar8600e402013-05-30 13:28:41 +02001060
1061 def keys(self):
Bram Moolenaar841fbd22013-06-23 14:37:07 +02001062 return list("abcH")
Bram Moolenaar8600e402013-05-30 13:28:41 +02001063
1064class FailingMapping(object):
1065 def __getitem__(self):
Bram Moolenaardee2e312013-06-23 16:35:47 +02001066 raise NotImplementedError('getitem:mapping')
Bram Moolenaar8600e402013-05-30 13:28:41 +02001067
1068 def keys(self):
Bram Moolenaardee2e312013-06-23 16:35:47 +02001069 raise NotImplementedError('keys')
Bram Moolenaar8600e402013-05-30 13:28:41 +02001070
1071class FailingList(list):
1072 def __getitem__(self, idx):
1073 if i == 2:
Bram Moolenaardee2e312013-06-23 16:35:47 +02001074 raise NotImplementedError('getitem:list')
Bram Moolenaar8600e402013-05-30 13:28:41 +02001075 else:
1076 return super(FailingList, self).__getitem__(idx)
1077
Bram Moolenaardee2e312013-06-23 16:35:47 +02001078class NoArgsCall(object):
1079 def __call__(self):
1080 pass
1081
1082class FailingCall(object):
1083 def __call__(self, path):
1084 raise NotImplementedError('call')
1085
1086class FailingNumber(object):
1087 def __int__(self):
1088 raise NotImplementedError('int')
1089
Bram Moolenaar8600e402013-05-30 13:28:41 +02001090cb.append("> Output")
1091cb.append(">> OutputSetattr")
1092ee('del sys.stdout.softspace')
Bram Moolenaardee2e312013-06-23 16:35:47 +02001093number_test('sys.stdout.softspace = %s', unsigned=True)
1094number_test('sys.stderr.softspace = %s', unsigned=True)
Bram Moolenaard4247472015-11-02 13:28:59 +01001095ee('assert sys.stdout.isatty()==False')
1096ee('assert sys.stdout.seekable()==False')
1097ee('sys.stdout.close()')
1098ee('sys.stdout.flush()')
1099ee('assert sys.stderr.isatty()==False')
1100ee('assert sys.stderr.seekable()==False')
1101ee('sys.stderr.close()')
1102ee('sys.stderr.flush()')
Bram Moolenaar8600e402013-05-30 13:28:41 +02001103ee('sys.stdout.attr = None')
1104cb.append(">> OutputWrite")
Bram Moolenaard4247472015-11-02 13:28:59 +01001105ee('assert sys.stdout.writable()==True')
1106ee('assert sys.stdout.readable()==False')
1107ee('assert sys.stderr.writable()==True')
1108ee('assert sys.stderr.readable()==False')
Bram Moolenaar8600e402013-05-30 13:28:41 +02001109ee('sys.stdout.write(None)')
1110cb.append(">> OutputWriteLines")
1111ee('sys.stdout.writelines(None)')
1112ee('sys.stdout.writelines([1])')
1113iter_test('sys.stdout.writelines(%s)')
1114cb.append("> VimCommand")
Bram Moolenaardee2e312013-06-23 16:35:47 +02001115stringtochars_test('vim.command(%s)')
1116ee('vim.command("", 2)')
Bram Moolenaar8600e402013-05-30 13:28:41 +02001117#! Not checked: vim->python exceptions translating: checked later
1118cb.append("> VimToPython")
1119#! Not checked: everything: needs errors in internal python functions
1120cb.append("> VimEval")
Bram Moolenaardee2e312013-06-23 16:35:47 +02001121stringtochars_test('vim.eval(%s)')
1122ee('vim.eval("", FailingTrue())')
Bram Moolenaar8600e402013-05-30 13:28:41 +02001123#! Not checked: everything: needs errors in internal python functions
1124cb.append("> VimEvalPy")
Bram Moolenaardee2e312013-06-23 16:35:47 +02001125stringtochars_test('vim.bindeval(%s)')
1126ee('vim.eval("", 2)')
Bram Moolenaar8600e402013-05-30 13:28:41 +02001127#! Not checked: vim->python exceptions translating: checked later
1128cb.append("> VimStrwidth")
Bram Moolenaardee2e312013-06-23 16:35:47 +02001129stringtochars_test('vim.strwidth(%s)')
1130cb.append("> VimForeachRTP")
1131ee('vim.foreach_rtp(None)')
1132ee('vim.foreach_rtp(NoArgsCall())')
1133ee('vim.foreach_rtp(FailingCall())')
1134ee('vim.foreach_rtp(int, 2)')
1135cb.append('> import')
1136old_rtp = vim.options['rtp']
1137vim.options['rtp'] = os.getcwd().replace(',', '\\,').replace('\\', '\\\\')
1138ee('import xxx_no_such_module_xxx')
1139ee('import failing_import')
1140ee('import failing')
1141vim.options['rtp'] = old_rtp
1142del old_rtp
Bram Moolenaar1028f4d2014-01-14 16:55:00 +01001143cb.append("> Options")
1144cb.append(">> OptionsItem")
1145ee('vim.options["abcQ"]')
1146ee('vim.options[""]')
1147stringtochars_test('vim.options[%s]')
1148cb.append(">> OptionsContains")
1149stringtochars_test('%s in vim.options')
Bram Moolenaar8600e402013-05-30 13:28:41 +02001150cb.append("> Dictionary")
1151cb.append(">> DictionaryConstructor")
Bram Moolenaar841fbd22013-06-23 14:37:07 +02001152ee('vim.Dictionary("abcI")')
Bram Moolenaar8600e402013-05-30 13:28:41 +02001153##! Not checked: py_dict_alloc failure
1154cb.append(">> DictionarySetattr")
1155ee('del d.locked')
1156ee('d.locked = FailingTrue()')
1157ee('vim.vvars.locked = False')
1158ee('d.scope = True')
1159ee('d.xxx = True')
1160cb.append(">> _DictionaryItem")
1161ee('d.get("a", 2, 3)')
1162stringtochars_test('d.get(%s)')
1163ee('d.pop("a")')
1164ee('dl.pop("a")')
Bram Moolenaarba2d7ff2013-11-04 00:34:53 +01001165cb.append(">> DictionaryContains")
1166ee('"" in d')
1167ee('0 in d')
Bram Moolenaar8600e402013-05-30 13:28:41 +02001168cb.append(">> DictionaryIterNext")
1169ee('for i in ned: ned["a"] = 1')
Bram Moolenaar841fbd22013-06-23 14:37:07 +02001170del i
Bram Moolenaar8600e402013-05-30 13:28:41 +02001171cb.append(">> DictionaryAssItem")
1172ee('dl["b"] = 1')
1173stringtochars_test('d[%s] = 1')
1174convertfrompyobject_test('d["a"] = %s')
1175cb.append(">> DictionaryUpdate")
1176cb.append(">>> kwargs")
1177cb.append(">>> iter")
1178ee('d.update(FailingMapping())')
1179ee('d.update([FailingIterNext()])')
Bram Moolenaar063a46b2014-01-14 16:36:51 +01001180ee('d.update([FailingIterNextN(1)])')
Bram Moolenaar8600e402013-05-30 13:28:41 +02001181iter_test('d.update(%s)')
1182convertfrompyobject_test('d.update(%s)')
1183stringtochars_test('d.update(((%s, 0),))')
1184convertfrompyobject_test('d.update((("a", %s),))')
1185cb.append(">> DictionaryPopItem")
1186ee('d.popitem(1, 2)')
1187cb.append(">> DictionaryHasKey")
1188ee('d.has_key()')
1189cb.append("> List")
1190cb.append(">> ListConstructor")
1191ee('vim.List(1, 2)')
1192ee('vim.List(a=1)')
1193iter_test('vim.List(%s)')
1194convertfrompyobject_test('vim.List([%s])')
1195cb.append(">> ListItem")
1196ee('l[1000]')
1197cb.append(">> ListAssItem")
1198ee('ll[1] = 2')
1199ee('l[1000] = 3')
1200cb.append(">> ListAssSlice")
Bram Moolenaar841fbd22013-06-23 14:37:07 +02001201ee('ll[1:100] = "abcJ"')
Bram Moolenaar8600e402013-05-30 13:28:41 +02001202iter_test('l[:] = %s')
Bram Moolenaar063a46b2014-01-14 16:36:51 +01001203ee('nel[1:10:2] = "abcK"')
1204cb.append(repr(tuple(nel)))
1205ee('nel[1:10:2] = "a"')
1206cb.append(repr(tuple(nel)))
1207ee('nel[1:1:-1] = "a"')
1208cb.append(repr(tuple(nel)))
1209ee('nel[:] = FailingIterNextN(2)')
1210cb.append(repr(tuple(nel)))
Bram Moolenaar8600e402013-05-30 13:28:41 +02001211convertfrompyobject_test('l[:] = [%s]')
1212cb.append(">> ListConcatInPlace")
1213iter_test('l.extend(%s)')
1214convertfrompyobject_test('l.extend([%s])')
1215cb.append(">> ListSetattr")
1216ee('del l.locked')
1217ee('l.locked = FailingTrue()')
1218ee('l.xxx = True')
1219cb.append("> Function")
1220cb.append(">> FunctionConstructor")
1221ee('vim.Function("123")')
1222ee('vim.Function("xxx_non_existent_function_xxx")')
1223ee('vim.Function("xxx#non#existent#function#xxx")')
1224cb.append(">> FunctionCall")
1225convertfrompyobject_test('f(%s)')
1226convertfrompymapping_test('fd(self=%s)')
1227cb.append("> TabPage")
1228cb.append(">> TabPageAttr")
1229ee('vim.current.tabpage.xxx')
1230cb.append("> TabList")
1231cb.append(">> TabListItem")
1232ee('vim.tabpages[1000]')
1233cb.append("> Window")
1234cb.append(">> WindowAttr")
1235ee('vim.current.window.xxx')
1236cb.append(">> WindowSetattr")
1237ee('vim.current.window.buffer = 0')
Bram Moolenaar96c7dfd2013-05-31 18:46:11 +02001238ee('vim.current.window.cursor = (100000000, 100000000)')
Bram Moolenaar8600e402013-05-30 13:28:41 +02001239ee('vim.current.window.cursor = True')
Bram Moolenaardee2e312013-06-23 16:35:47 +02001240number_test('vim.current.window.height = %s', unsigned=True)
1241number_test('vim.current.window.width = %s', unsigned=True)
Bram Moolenaar8600e402013-05-30 13:28:41 +02001242ee('vim.current.window.xxxxxx = True')
1243cb.append("> WinList")
1244cb.append(">> WinListItem")
1245ee('vim.windows[1000]')
1246cb.append("> Buffer")
1247cb.append(">> StringToLine (indirect)")
1248ee('vim.current.buffer[0] = "\\na"')
Bram Moolenaardee2e312013-06-23 16:35:47 +02001249ee('vim.current.buffer[0] = b"\\na"')
Bram Moolenaar8600e402013-05-30 13:28:41 +02001250cb.append(">> SetBufferLine (indirect)")
1251ee('vim.current.buffer[0] = True')
Bram Moolenaar841fbd22013-06-23 14:37:07 +02001252cb.append(">> SetBufferLineList (indirect)")
Bram Moolenaar8600e402013-05-30 13:28:41 +02001253ee('vim.current.buffer[:] = True')
1254ee('vim.current.buffer[:] = ["\\na", "bc"]')
1255cb.append(">> InsertBufferLines (indirect)")
1256ee('vim.current.buffer.append(None)')
1257ee('vim.current.buffer.append(["\\na", "bc"])')
1258ee('vim.current.buffer.append("\\nbc")')
1259cb.append(">> RBItem")
Bram Moolenaar96c7dfd2013-05-31 18:46:11 +02001260ee('vim.current.buffer[100000000]')
Bram Moolenaar8600e402013-05-30 13:28:41 +02001261cb.append(">> RBAsItem")
Bram Moolenaar96c7dfd2013-05-31 18:46:11 +02001262ee('vim.current.buffer[100000000] = ""')
Bram Moolenaar8600e402013-05-30 13:28:41 +02001263cb.append(">> BufferAttr")
1264ee('vim.current.buffer.xxx')
1265cb.append(">> BufferSetattr")
1266ee('vim.current.buffer.name = True')
1267ee('vim.current.buffer.xxx = True')
1268cb.append(">> BufferMark")
1269ee('vim.current.buffer.mark(0)')
Bram Moolenaar841fbd22013-06-23 14:37:07 +02001270ee('vim.current.buffer.mark("abcM")')
Bram Moolenaar8600e402013-05-30 13:28:41 +02001271ee('vim.current.buffer.mark("!")')
1272cb.append(">> BufferRange")
1273ee('vim.current.buffer.range(1, 2, 3)')
1274cb.append("> BufMap")
1275cb.append(">> BufMapItem")
Bram Moolenaar8600e402013-05-30 13:28:41 +02001276ee('vim.buffers[100000000]')
Bram Moolenaardee2e312013-06-23 16:35:47 +02001277number_test('vim.buffers[%s]', natural=True)
Bram Moolenaar8600e402013-05-30 13:28:41 +02001278cb.append("> Current")
1279cb.append(">> CurrentGetattr")
1280ee('vim.current.xxx')
1281cb.append(">> CurrentSetattr")
1282ee('vim.current.line = True')
1283ee('vim.current.buffer = True')
1284ee('vim.current.window = True')
1285ee('vim.current.tabpage = True')
1286ee('vim.current.xxx = True')
Bram Moolenaar841fbd22013-06-23 14:37:07 +02001287del d
1288del ned
1289del dl
1290del l
1291del ll
Bram Moolenaar063a46b2014-01-14 16:36:51 +01001292del nel
Bram Moolenaar841fbd22013-06-23 14:37:07 +02001293del f
1294del fd
1295del fdel
1296del subexpr_test
1297del stringtochars_test
1298del Mapping
1299del convertfrompyobject_test
1300del convertfrompymapping_test
1301del iter_test
Bram Moolenaardee2e312013-06-23 16:35:47 +02001302del number_test
Bram Moolenaar841fbd22013-06-23 14:37:07 +02001303del FailingTrue
1304del FailingIter
1305del FailingIterNext
Bram Moolenaar063a46b2014-01-14 16:36:51 +01001306del FailingIterNextN
Bram Moolenaar841fbd22013-06-23 14:37:07 +02001307del FailingMapping
1308del FailingMappingKey
1309del FailingList
Bram Moolenaardee2e312013-06-23 16:35:47 +02001310del NoArgsCall
1311del FailingCall
1312del FailingNumber
Bram Moolenaar8600e402013-05-30 13:28:41 +02001313EOF
Bram Moolenaar841fbd22013-06-23 14:37:07 +02001314:delfunction F
Bram Moolenaar8600e402013-05-30 13:28:41 +02001315:"
Bram Moolenaara9f22202013-06-11 18:48:21 +02001316:" Test import
1317py3 << EOF
Bram Moolenaar9f3685a2013-06-12 14:20:36 +02001318sys.path.insert(0, os.path.join(os.getcwd(), 'python_before'))
1319sys.path.append(os.path.join(os.getcwd(), 'python_after'))
Bram Moolenaara9f22202013-06-11 18:48:21 +02001320vim.options['rtp'] = os.getcwd().replace(',', '\\,').replace('\\', '\\\\')
Bram Moolenaardee2e312013-06-23 16:35:47 +02001321l = []
1322def callback(path):
1323 l.append(os.path.relpath(path))
1324vim.foreach_rtp(callback)
1325cb.append(repr(l))
1326del l
1327def callback(path):
1328 return os.path.relpath(path)
1329cb.append(repr(vim.foreach_rtp(callback)))
1330del callback
Bram Moolenaara9f22202013-06-11 18:48:21 +02001331from module import dir as d
1332from modulex import ddir
1333cb.append(d + ',' + ddir)
Bram Moolenaar9f3685a2013-06-12 14:20:36 +02001334import before
1335cb.append(before.dir)
1336import after
1337cb.append(after.dir)
Bram Moolenaardee2e312013-06-23 16:35:47 +02001338import topmodule as tm
1339import topmodule.submodule as tms
1340import topmodule.submodule.subsubmodule.subsubsubmodule as tmsss
Bram Moolenaar877aa002013-06-26 21:49:51 +02001341cb.append(tm.__file__.replace(os.path.sep, '/')[-len('modulex/topmodule/__init__.py'):])
1342cb.append(tms.__file__.replace(os.path.sep, '/')[-len('modulex/topmodule/submodule/__init__.py'):])
1343cb.append(tmsss.__file__.replace(os.path.sep, '/')[-len('modulex/topmodule/submodule/subsubmodule/subsubsubmodule.py'):])
Bram Moolenaar841fbd22013-06-23 14:37:07 +02001344del before
1345del after
1346del d
1347del ddir
Bram Moolenaardee2e312013-06-23 16:35:47 +02001348del tm
1349del tms
1350del tmsss
Bram Moolenaara9f22202013-06-11 18:48:21 +02001351EOF
Bram Moolenaarc09a6d62013-06-10 21:27:29 +02001352:"
Bram Moolenaara7b64ce2013-05-21 20:40:40 +02001353:" Test exceptions
1354:fun Exe(e)
1355: execute a:e
1356:endfun
1357py3 << EOF
Bram Moolenaara7b64ce2013-05-21 20:40:40 +02001358Exe = vim.bindeval('function("Exe")')
Bram Moolenaar841fbd22013-06-23 14:37:07 +02001359ee('vim.command("throw \'abcN\'")')
Bram Moolenaara7b64ce2013-05-21 20:40:40 +02001360ee('Exe("throw \'def\'")')
1361ee('vim.eval("Exe(\'throw \'\'ghi\'\'\')")')
1362ee('vim.eval("Exe(\'echoerr \'\'jkl\'\'\')")')
1363ee('vim.eval("Exe(\'xxx_non_existent_command_xxx\')")')
Bram Moolenaar9fee7d42013-11-28 17:04:43 +01001364ee('vim.eval("xxx_unknown_function_xxx()")')
Bram Moolenaara7b64ce2013-05-21 20:40:40 +02001365ee('vim.bindeval("Exe(\'xxx_non_existent_command_xxx\')")')
Bram Moolenaar841fbd22013-06-23 14:37:07 +02001366del Exe
1367EOF
1368:delfunction Exe
1369:"
Bram Moolenaard6b8a522013-11-11 01:05:48 +01001370:" Regression: interrupting vim.command propagates to next vim.command
1371py3 << EOF
1372def test_keyboard_interrupt():
1373 try:
1374 vim.command('while 1 | endwhile')
1375 except KeyboardInterrupt:
1376 cb.append('Caught KeyboardInterrupt')
1377 except Exception as e:
1378 cb.append('!!!!!!!! Caught exception: ' + repr(e))
1379 else:
1380 cb.append('!!!!!!!! No exception')
1381 try:
1382 vim.command('$ put =\'Running :put\'')
1383 except KeyboardInterrupt:
1384 cb.append('!!!!!!!! Caught KeyboardInterrupt')
1385 except Exception as e:
1386 cb.append('!!!!!!!! Caught exception: ' + repr(e))
1387 else:
1388 cb.append('No exception')
1389EOF
1390:debuggreedy
1391:call inputsave()
1392:call feedkeys("s\ns\ns\ns\nq\n")
1393:redir => output
1394:debug silent! py3 test_keyboard_interrupt()
1395:redir END
1396:0 debuggreedy
Bram Moolenaar56f62272014-09-29 18:08:59 +02001397:call inputrestore()
Bram Moolenaard6b8a522013-11-11 01:05:48 +01001398:silent $put =output
1399:unlet output
1400:py3 del test_keyboard_interrupt
1401:"
Bram Moolenaar841fbd22013-06-23 14:37:07 +02001402:" Cleanup
1403py3 << EOF
1404del cb
1405del ee
1406del sys
1407del os
1408del vim
Bram Moolenaara7b64ce2013-05-21 20:40:40 +02001409EOF
Bram Moolenaardb913952012-06-29 12:54:53 +02001410:endfun
1411:"
Bram Moolenaar841fbd22013-06-23 14:37:07 +02001412:fun RunTest()
1413:let checkrefs = !empty($PYTHONDUMPREFS)
1414:let start = getline(1, '$')
1415:for i in range(checkrefs ? 10 : 1)
1416: if i != 0
1417: %d _
1418: call setline(1, start)
1419: endif
1420: call Test()
1421: if i == 0
1422: let result = getline(1, '$')
1423: endif
1424:endfor
1425:if checkrefs
1426: %d _
1427: call setline(1, result)
1428:endif
1429:endfun
Bram Moolenaardb913952012-06-29 12:54:53 +02001430:"
Bram Moolenaar841fbd22013-06-23 14:37:07 +02001431:call RunTest()
1432:delfunction RunTest
1433:delfunction Test
Bram Moolenaardb913952012-06-29 12:54:53 +02001434:call garbagecollect(1)
1435:"
1436:/^start:/,$wq! test.out
Bram Moolenaardee2e312013-06-23 16:35:47 +02001437:" vim: et ts=4 isk-=\:
Bram Moolenaarb6c589a2013-05-15 14:39:52 +02001438:call getchar()
Bram Moolenaardb913952012-06-29 12:54:53 +02001439ENDTEST
1440
1441start: