blob: 967cd8fc4209808d313935b56c3ab8727a11e1d6 [file] [log] [blame]
Bram Moolenaarc6df10e2017-07-29 20:15:08 +02001" Tests for the terminal window.
2
Bram Moolenaarea5d6fa2017-08-18 21:07:11 +02003if !has('terminal')
Bram Moolenaarc6df10e2017-07-29 20:15:08 +02004 finish
5endif
6
7source shared.vim
8
Bram Moolenaarb81bc772017-08-11 22:45:01 +02009let s:python = PythonProg()
10
Bram Moolenaar94053a52017-08-01 21:44:33 +020011" Open a terminal with a shell, assign the job to g:job and return the buffer
12" number.
Bram Moolenaar05aafed2017-08-11 19:12:11 +020013func Run_shell_in_terminal(options)
Bram Moolenaarba6febd2017-10-30 21:56:23 +010014 if has('win32')
15 let buf = term_start([&shell,'/k'], a:options)
16 else
17 let buf = term_start(&shell, a:options)
18 endif
Bram Moolenaarc6df10e2017-07-29 20:15:08 +020019
20 let termlist = term_list()
21 call assert_equal(1, len(termlist))
22 call assert_equal(buf, termlist[0])
23
24 let g:job = term_getjob(buf)
25 call assert_equal(v:t_job, type(g:job))
26
Bram Moolenaar35422f42017-08-05 16:33:56 +020027 let string = string({'job': term_getjob(buf)})
28 call assert_match("{'job': 'process \\d\\+ run'}", string)
29
Bram Moolenaar94053a52017-08-01 21:44:33 +020030 return buf
31endfunc
32
33" Stops the shell started by Run_shell_in_terminal().
34func Stop_shell_in_terminal(buf)
35 call term_sendkeys(a:buf, "exit\r")
Bram Moolenaarc6df10e2017-07-29 20:15:08 +020036 call WaitFor('job_status(g:job) == "dead"')
37 call assert_equal('dead', job_status(g:job))
Bram Moolenaar20e6cd02017-08-01 20:25:22 +020038endfunc
39
40func Test_terminal_basic()
Bram Moolenaarb00fdf62017-09-21 22:16:21 +020041 au BufWinEnter * if &buftype == 'terminal' | let b:done = 'yes' | endif
Bram Moolenaar05aafed2017-08-11 19:12:11 +020042 let buf = Run_shell_in_terminal({})
Bram Moolenaarb00fdf62017-09-21 22:16:21 +020043
Bram Moolenaar7c9aec42017-08-03 13:51:25 +020044 if has("unix")
Bram Moolenaar2dc9d262017-09-08 14:39:30 +020045 call assert_match('^/dev/', job_info(g:job).tty_out)
46 call assert_match('^/dev/', term_gettty(''))
Bram Moolenaar7c9aec42017-08-03 13:51:25 +020047 else
Bram Moolenaar2dc9d262017-09-08 14:39:30 +020048 call assert_match('^\\\\.\\pipe\\', job_info(g:job).tty_out)
49 call assert_match('^\\\\.\\pipe\\', term_gettty(''))
Bram Moolenaar7c9aec42017-08-03 13:51:25 +020050 endif
Bram Moolenaar2bb7b6b2017-08-13 20:58:33 +020051 call assert_equal('t', mode())
Bram Moolenaarb00fdf62017-09-21 22:16:21 +020052 call assert_equal('yes', b:done)
Bram Moolenaar2bb7b6b2017-08-13 20:58:33 +020053 call assert_match('%aR[^\n]*running]', execute('ls'))
54
Bram Moolenaar94053a52017-08-01 21:44:33 +020055 call Stop_shell_in_terminal(buf)
56 call term_wait(buf)
Bram Moolenaar2bb7b6b2017-08-13 20:58:33 +020057 call assert_equal('n', mode())
58 call assert_match('%aF[^\n]*finished]', execute('ls'))
Bram Moolenaar20e6cd02017-08-01 20:25:22 +020059
Bram Moolenaar94053a52017-08-01 21:44:33 +020060 " closing window wipes out the terminal buffer a with finished job
61 close
62 call assert_equal("", bufname(buf))
63
Bram Moolenaarb00fdf62017-09-21 22:16:21 +020064 au! BufWinEnter
Bram Moolenaar20e6cd02017-08-01 20:25:22 +020065 unlet g:job
66endfunc
67
68func Test_terminal_make_change()
Bram Moolenaar05aafed2017-08-11 19:12:11 +020069 let buf = Run_shell_in_terminal({})
Bram Moolenaar94053a52017-08-01 21:44:33 +020070 call Stop_shell_in_terminal(buf)
Bram Moolenaar20e6cd02017-08-01 20:25:22 +020071 call term_wait(buf)
72
73 setlocal modifiable
74 exe "normal Axxx\<Esc>"
75 call assert_fails(buf . 'bwipe', 'E517')
76 undo
77
Bram Moolenaarc6df10e2017-07-29 20:15:08 +020078 exe buf . 'bwipe'
79 unlet g:job
80endfunc
81
Bram Moolenaar94053a52017-08-01 21:44:33 +020082func Test_terminal_wipe_buffer()
Bram Moolenaar05aafed2017-08-11 19:12:11 +020083 let buf = Run_shell_in_terminal({})
Bram Moolenaareb44a682017-08-03 22:44:55 +020084 call assert_fails(buf . 'bwipe', 'E517')
85 exe buf . 'bwipe!'
Bram Moolenaar94053a52017-08-01 21:44:33 +020086 call WaitFor('job_status(g:job) == "dead"')
87 call assert_equal('dead', job_status(g:job))
88 call assert_equal("", bufname(buf))
89
90 unlet g:job
91endfunc
92
Bram Moolenaar8adb0d02017-09-17 19:08:02 +020093func Test_terminal_split_quit()
94 let buf = Run_shell_in_terminal({})
95 call term_wait(buf)
96 split
97 quit!
98 call term_wait(buf)
99 sleep 50m
100 call assert_equal('run', job_status(g:job))
101
102 quit!
103 call WaitFor('job_status(g:job) == "dead"')
104 call assert_equal('dead', job_status(g:job))
105
106 exe buf . 'bwipe'
107 unlet g:job
108endfunc
109
Bram Moolenaar94053a52017-08-01 21:44:33 +0200110func Test_terminal_hide_buffer()
Bram Moolenaar05aafed2017-08-11 19:12:11 +0200111 let buf = Run_shell_in_terminal({})
Bram Moolenaar97a80e42017-08-30 13:31:49 +0200112 setlocal bufhidden=hide
Bram Moolenaar94053a52017-08-01 21:44:33 +0200113 quit
114 for nr in range(1, winnr('$'))
115 call assert_notequal(winbufnr(nr), buf)
116 endfor
117 call assert_true(bufloaded(buf))
118 call assert_true(buflisted(buf))
119
120 exe 'split ' . buf . 'buf'
121 call Stop_shell_in_terminal(buf)
122 exe buf . 'bwipe'
123
124 unlet g:job
125endfunc
126
Bram Moolenaar3c3a80d2017-08-03 17:06:45 +0200127func! s:Nasty_exit_cb(job, st)
128 exe g:buf . 'bwipe!'
129 let g:buf = 0
130endfunc
131
Bram Moolenaar9d189612017-09-09 18:11:00 +0200132func Get_cat_123_cmd()
133 if has('win32')
134 return 'cmd /c "cls && color 2 && echo 123"'
135 else
136 call writefile(["\<Esc>[32m123"], 'Xtext')
137 return "cat Xtext"
138 endif
139endfunc
140
Bram Moolenaar3c3a80d2017-08-03 17:06:45 +0200141func Test_terminal_nasty_cb()
Bram Moolenaar33a43be2017-08-06 21:36:22 +0200142 let cmd = Get_cat_123_cmd()
Bram Moolenaar3c3a80d2017-08-03 17:06:45 +0200143 let g:buf = term_start(cmd, {'exit_cb': function('s:Nasty_exit_cb')})
144 let g:job = term_getjob(g:buf)
145
146 call WaitFor('job_status(g:job) == "dead"')
147 call WaitFor('g:buf == 0')
148 unlet g:buf
149 unlet g:job
150 call delete('Xtext')
151endfunc
152
Bram Moolenaarc6df10e2017-07-29 20:15:08 +0200153func Check_123(buf)
Bram Moolenaar5c838a32017-08-02 22:10:34 +0200154 let l = term_scrape(a:buf, 0)
155 call assert_true(len(l) == 0)
156 let l = term_scrape(a:buf, 999)
157 call assert_true(len(l) == 0)
Bram Moolenaar9c844842017-08-01 18:41:21 +0200158 let l = term_scrape(a:buf, 1)
Bram Moolenaarc6df10e2017-07-29 20:15:08 +0200159 call assert_true(len(l) > 0)
160 call assert_equal('1', l[0].chars)
161 call assert_equal('2', l[1].chars)
162 call assert_equal('3', l[2].chars)
163 call assert_equal('#00e000', l[0].fg)
164 if &background == 'light'
165 call assert_equal('#ffffff', l[0].bg)
166 else
167 call assert_equal('#000000', l[0].bg)
168 endif
169
Bram Moolenaar5c838a32017-08-02 22:10:34 +0200170 let l = term_getline(a:buf, -1)
171 call assert_equal('', l)
172 let l = term_getline(a:buf, 0)
173 call assert_equal('', l)
174 let l = term_getline(a:buf, 999)
175 call assert_equal('', l)
Bram Moolenaar9c844842017-08-01 18:41:21 +0200176 let l = term_getline(a:buf, 1)
Bram Moolenaarc6df10e2017-07-29 20:15:08 +0200177 call assert_equal('123', l)
178endfunc
179
Bram Moolenaar33a43be2017-08-06 21:36:22 +0200180func Test_terminal_scrape_123()
181 let cmd = Get_cat_123_cmd()
Bram Moolenaarc6df10e2017-07-29 20:15:08 +0200182 let buf = term_start(cmd)
183
184 let termlist = term_list()
185 call assert_equal(1, len(termlist))
186 call assert_equal(buf, termlist[0])
187
Bram Moolenaarf144a3f2017-07-30 18:02:12 +0200188 " Nothing happens with invalid buffer number
189 call term_wait(1234)
190
Bram Moolenaarc6df10e2017-07-29 20:15:08 +0200191 call term_wait(buf)
Bram Moolenaare88fc7a2017-09-03 20:59:40 +0200192 let g:buf = buf
Bram Moolenaar17833372017-09-04 22:23:19 +0200193 " On MS-Windows we first get a startup message of two lines, wait for the
Bram Moolenaar1bfdc072017-09-05 20:19:42 +0200194 " "cls" to happen, after that we have one line with three characters.
195 call WaitFor('len(term_scrape(g:buf, 1)) == 3')
Bram Moolenaarc6df10e2017-07-29 20:15:08 +0200196 call Check_123(buf)
197
198 " Must still work after the job ended.
199 let g:job = term_getjob(buf)
200 call WaitFor('job_status(g:job) == "dead"')
201 call term_wait(buf)
202 call Check_123(buf)
203
204 exe buf . 'bwipe'
Bram Moolenaarf144a3f2017-07-30 18:02:12 +0200205 call delete('Xtext')
Bram Moolenaarc6df10e2017-07-29 20:15:08 +0200206endfunc
Bram Moolenaarcfcc0222017-08-05 17:13:48 +0200207
Bram Moolenaar33a43be2017-08-06 21:36:22 +0200208func Test_terminal_scrape_multibyte()
209 if !has('multi_byte')
210 return
211 endif
212 call writefile(["léttrs"], 'Xtext')
213 if has('win32')
Bram Moolenaar36783932017-08-14 23:07:30 +0200214 " Run cmd with UTF-8 codepage to make the type command print the expected
215 " multibyte characters.
216 let g:buf = term_start("cmd /K chcp 65001")
217 call term_sendkeys(g:buf, "type Xtext\<CR>")
218 call term_sendkeys(g:buf, "exit\<CR>")
219 let g:line = 4
Bram Moolenaar33a43be2017-08-06 21:36:22 +0200220 else
Bram Moolenaar36783932017-08-14 23:07:30 +0200221 let g:buf = term_start("cat Xtext")
222 let g:line = 1
Bram Moolenaar33a43be2017-08-06 21:36:22 +0200223 endif
Bram Moolenaar33a43be2017-08-06 21:36:22 +0200224
Bram Moolenaara038cb52017-09-11 20:45:23 +0200225 call WaitFor('len(term_scrape(g:buf, g:line)) >= 7 && term_scrape(g:buf, g:line)[0].chars == "l"')
Bram Moolenaar36783932017-08-14 23:07:30 +0200226 let l = term_scrape(g:buf, g:line)
Bram Moolenaar33a43be2017-08-06 21:36:22 +0200227 call assert_true(len(l) >= 7)
228 call assert_equal('l', l[0].chars)
229 call assert_equal('é', l[1].chars)
230 call assert_equal(1, l[1].width)
231 call assert_equal('t', l[2].chars)
232 call assert_equal('t', l[3].chars)
233 call assert_equal('ま', l[4].chars)
234 call assert_equal(2, l[4].width)
235 call assert_equal('r', l[5].chars)
236 call assert_equal('s', l[6].chars)
237
Bram Moolenaarc0870612017-08-14 22:01:16 +0200238 let g:job = term_getjob(g:buf)
Bram Moolenaar33a43be2017-08-06 21:36:22 +0200239 call WaitFor('job_status(g:job) == "dead"')
Bram Moolenaarc0870612017-08-14 22:01:16 +0200240 call term_wait(g:buf)
Bram Moolenaar33a43be2017-08-06 21:36:22 +0200241
Bram Moolenaarc0870612017-08-14 22:01:16 +0200242 exe g:buf . 'bwipe'
243 unlet g:buf
Bram Moolenaar36783932017-08-14 23:07:30 +0200244 unlet g:line
Bram Moolenaar33a43be2017-08-06 21:36:22 +0200245 call delete('Xtext')
246endfunc
247
Bram Moolenaarf8d57a52017-08-07 20:38:42 +0200248func Test_terminal_scroll()
249 call writefile(range(1, 200), 'Xtext')
250 if has('win32')
251 let cmd = 'cmd /c "type Xtext"'
252 else
253 let cmd = "cat Xtext"
254 endif
255 let buf = term_start(cmd)
256
257 let g:job = term_getjob(buf)
258 call WaitFor('job_status(g:job) == "dead"')
259 call term_wait(buf)
260 if has('win32')
261 " TODO: this should not be needed
262 sleep 100m
263 endif
264
Bram Moolenaar82b9ca02017-08-08 23:06:46 +0200265 let scrolled = term_getscrolled(buf)
Bram Moolenaarf8d57a52017-08-07 20:38:42 +0200266 call assert_equal('1', getline(1))
Bram Moolenaar82b9ca02017-08-08 23:06:46 +0200267 call assert_equal('1', term_getline(buf, 1 - scrolled))
Bram Moolenaarf8d57a52017-08-07 20:38:42 +0200268 call assert_equal('49', getline(49))
Bram Moolenaar82b9ca02017-08-08 23:06:46 +0200269 call assert_equal('49', term_getline(buf, 49 - scrolled))
Bram Moolenaarf8d57a52017-08-07 20:38:42 +0200270 call assert_equal('200', getline(200))
Bram Moolenaar82b9ca02017-08-08 23:06:46 +0200271 call assert_equal('200', term_getline(buf, 200 - scrolled))
Bram Moolenaarf8d57a52017-08-07 20:38:42 +0200272
273 exe buf . 'bwipe'
274 call delete('Xtext')
275endfunc
276
Bram Moolenaarcfcc0222017-08-05 17:13:48 +0200277func Test_terminal_size()
Bram Moolenaar33a43be2017-08-06 21:36:22 +0200278 let cmd = Get_cat_123_cmd()
Bram Moolenaarcfcc0222017-08-05 17:13:48 +0200279
Bram Moolenaarb2412082017-08-20 18:09:14 +0200280 exe 'terminal ++rows=5 ' . cmd
Bram Moolenaarcfcc0222017-08-05 17:13:48 +0200281 let size = term_getsize('')
282 bwipe!
283 call assert_equal(5, size[0])
284
Bram Moolenaar08d384f2017-08-11 21:51:23 +0200285 call term_start(cmd, {'term_rows': 6})
286 let size = term_getsize('')
287 bwipe!
288 call assert_equal(6, size[0])
289
Bram Moolenaarcfcc0222017-08-05 17:13:48 +0200290 vsplit
Bram Moolenaarb2412082017-08-20 18:09:14 +0200291 exe 'terminal ++rows=5 ++cols=33 ' . cmd
Bram Moolenaarcfcc0222017-08-05 17:13:48 +0200292 let size = term_getsize('')
293 bwipe!
294 call assert_equal([5, 33], size)
295
Bram Moolenaar08d384f2017-08-11 21:51:23 +0200296 call term_start(cmd, {'term_rows': 6, 'term_cols': 36})
297 let size = term_getsize('')
298 bwipe!
299 call assert_equal([6, 36], size)
300
Bram Moolenaarb2412082017-08-20 18:09:14 +0200301 exe 'vertical terminal ++cols=20 ' . cmd
Bram Moolenaarcfcc0222017-08-05 17:13:48 +0200302 let size = term_getsize('')
303 bwipe!
304 call assert_equal(20, size[1])
305
Bram Moolenaar08d384f2017-08-11 21:51:23 +0200306 call term_start(cmd, {'vertical': 1, 'term_cols': 26})
307 let size = term_getsize('')
308 bwipe!
309 call assert_equal(26, size[1])
310
Bram Moolenaarcfcc0222017-08-05 17:13:48 +0200311 split
Bram Moolenaarb2412082017-08-20 18:09:14 +0200312 exe 'vertical terminal ++rows=6 ++cols=20 ' . cmd
Bram Moolenaarcfcc0222017-08-05 17:13:48 +0200313 let size = term_getsize('')
314 bwipe!
315 call assert_equal([6, 20], size)
Bram Moolenaar08d384f2017-08-11 21:51:23 +0200316
317 call term_start(cmd, {'vertical': 1, 'term_rows': 7, 'term_cols': 27})
318 let size = term_getsize('')
319 bwipe!
320 call assert_equal([7, 27], size)
Bram Moolenaar9d654a82017-09-03 19:52:17 +0200321
322 call delete('Xtext')
Bram Moolenaarda43b612017-08-11 22:27:50 +0200323endfunc
324
325func Test_terminal_curwin()
326 let cmd = Get_cat_123_cmd()
327 call assert_equal(1, winnr('$'))
328
329 split dummy
330 exe 'terminal ++curwin ' . cmd
331 call assert_equal(2, winnr('$'))
332 bwipe!
333
334 split dummy
335 call term_start(cmd, {'curwin': 1})
336 call assert_equal(2, winnr('$'))
337 bwipe!
338
339 split dummy
340 call setline(1, 'change')
341 call assert_fails('terminal ++curwin ' . cmd, 'E37:')
342 call assert_equal(2, winnr('$'))
343 exe 'terminal! ++curwin ' . cmd
344 call assert_equal(2, winnr('$'))
345 bwipe!
346
347 split dummy
348 call setline(1, 'change')
349 call assert_fails("call term_start(cmd, {'curwin': 1})", 'E37:')
350 call assert_equal(2, winnr('$'))
351 bwipe!
352
353 split dummy
354 bwipe!
Bram Moolenaar9d654a82017-09-03 19:52:17 +0200355 call delete('Xtext')
Bram Moolenaarcfcc0222017-08-05 17:13:48 +0200356endfunc
Bram Moolenaardd693ce2017-08-10 23:15:19 +0200357
Bram Moolenaar37c45832017-08-12 16:01:04 +0200358func Test_finish_open_close()
Bram Moolenaardd693ce2017-08-10 23:15:19 +0200359 call assert_equal(1, winnr('$'))
360
Bram Moolenaarb81bc772017-08-11 22:45:01 +0200361 if s:python != ''
362 let cmd = s:python . " test_short_sleep.py"
363 let waittime = 500
Bram Moolenaardd693ce2017-08-10 23:15:19 +0200364 else
Bram Moolenaarb81bc772017-08-11 22:45:01 +0200365 echo 'This will take five seconds...'
366 let waittime = 2000
367 if has('win32')
368 let cmd = $windir . '\system32\timeout.exe 1'
369 else
370 let cmd = 'sleep 1'
371 endif
Bram Moolenaardd693ce2017-08-10 23:15:19 +0200372 endif
Bram Moolenaarb81bc772017-08-11 22:45:01 +0200373
Bram Moolenaardd693ce2017-08-10 23:15:19 +0200374 exe 'terminal ++close ' . cmd
Bram Moolenaardd693ce2017-08-10 23:15:19 +0200375 call assert_equal(2, winnr('$'))
Bram Moolenaardd693ce2017-08-10 23:15:19 +0200376 wincmd p
Bram Moolenaarb81bc772017-08-11 22:45:01 +0200377 call WaitFor("winnr('$') == 1", waittime)
Bram Moolenaardd693ce2017-08-10 23:15:19 +0200378 call assert_equal(1, winnr('$'))
379
380 call term_start(cmd, {'term_finish': 'close'})
381 call assert_equal(2, winnr('$'))
Bram Moolenaardd693ce2017-08-10 23:15:19 +0200382 wincmd p
Bram Moolenaarb81bc772017-08-11 22:45:01 +0200383 call WaitFor("winnr('$') == 1", waittime)
Bram Moolenaardd693ce2017-08-10 23:15:19 +0200384 call assert_equal(1, winnr('$'))
385
386 exe 'terminal ++open ' . cmd
Bram Moolenaar97a80e42017-08-30 13:31:49 +0200387 close!
Bram Moolenaarb81bc772017-08-11 22:45:01 +0200388 call WaitFor("winnr('$') == 2", waittime)
Bram Moolenaardd693ce2017-08-10 23:15:19 +0200389 call assert_equal(2, winnr('$'))
390 bwipe
391
392 call term_start(cmd, {'term_finish': 'open'})
Bram Moolenaar97a80e42017-08-30 13:31:49 +0200393 close!
Bram Moolenaarb81bc772017-08-11 22:45:01 +0200394 call WaitFor("winnr('$') == 2", waittime)
Bram Moolenaardd693ce2017-08-10 23:15:19 +0200395 call assert_equal(2, winnr('$'))
Bram Moolenaar8cad9302017-08-12 14:32:32 +0200396 bwipe
Bram Moolenaardd693ce2017-08-10 23:15:19 +0200397
Bram Moolenaar8cad9302017-08-12 14:32:32 +0200398 exe 'terminal ++hidden ++open ' . cmd
399 call assert_equal(1, winnr('$'))
400 call WaitFor("winnr('$') == 2", waittime)
401 call assert_equal(2, winnr('$'))
402 bwipe
403
404 call term_start(cmd, {'term_finish': 'open', 'hidden': 1})
405 call assert_equal(1, winnr('$'))
406 call WaitFor("winnr('$') == 2", waittime)
407 call assert_equal(2, winnr('$'))
Bram Moolenaardd693ce2017-08-10 23:15:19 +0200408 bwipe
Bram Moolenaar37c45832017-08-12 16:01:04 +0200409
410 call assert_fails("call term_start(cmd, {'term_opencmd': 'open'})", 'E475:')
411 call assert_fails("call term_start(cmd, {'term_opencmd': 'split %x'})", 'E475:')
412 call assert_fails("call term_start(cmd, {'term_opencmd': 'split %d and %s'})", 'E475:')
413 call assert_fails("call term_start(cmd, {'term_opencmd': 'split % and %d'})", 'E475:')
414
415 call term_start(cmd, {'term_finish': 'open', 'term_opencmd': '4split | buffer %d'})
Bram Moolenaar97a80e42017-08-30 13:31:49 +0200416 close!
Bram Moolenaar37c45832017-08-12 16:01:04 +0200417 call WaitFor("winnr('$') == 2", waittime)
418 call assert_equal(2, winnr('$'))
419 call assert_equal(4, winheight(0))
420 bwipe
Bram Moolenaardd693ce2017-08-10 23:15:19 +0200421endfunc
Bram Moolenaar05aafed2017-08-11 19:12:11 +0200422
423func Test_terminal_cwd()
Bram Moolenaare9f6fd22017-09-10 14:25:49 +0200424 if !executable('pwd')
Bram Moolenaar05aafed2017-08-11 19:12:11 +0200425 return
426 endif
427 call mkdir('Xdir')
428 let buf = term_start('pwd', {'cwd': 'Xdir'})
Bram Moolenaare9f6fd22017-09-10 14:25:49 +0200429 call WaitFor('"Xdir" == fnamemodify(getline(1), ":t")')
430 call assert_equal('Xdir', fnamemodify(getline(1), ":t"))
Bram Moolenaar05aafed2017-08-11 19:12:11 +0200431
432 exe buf . 'bwipe'
433 call delete('Xdir', 'rf')
434endfunc
435
436func Test_terminal_env()
Bram Moolenaarc0870612017-08-14 22:01:16 +0200437 let g:buf = Run_shell_in_terminal({'env': {'TESTENV': 'correct'}})
Bram Moolenaar51c23682017-08-14 21:45:00 +0200438 " Wait for the shell to display a prompt
Bram Moolenaarc0870612017-08-14 22:01:16 +0200439 call WaitFor('term_getline(g:buf, 1) != ""')
Bram Moolenaarba6febd2017-10-30 21:56:23 +0100440 if has('win32')
441 call term_sendkeys(g:buf, "echo %TESTENV%\r")
442 else
443 call term_sendkeys(g:buf, "echo $TESTENV\r")
444 endif
Bram Moolenaarc0870612017-08-14 22:01:16 +0200445 call term_wait(g:buf)
446 call Stop_shell_in_terminal(g:buf)
Bram Moolenaar51c23682017-08-14 21:45:00 +0200447 call WaitFor('getline(2) == "correct"')
Bram Moolenaar05aafed2017-08-11 19:12:11 +0200448 call assert_equal('correct', getline(2))
449
Bram Moolenaarc0870612017-08-14 22:01:16 +0200450 exe g:buf . 'bwipe'
451 unlet g:buf
Bram Moolenaar05aafed2017-08-11 19:12:11 +0200452endfunc
Bram Moolenaar679653e2017-08-13 14:13:19 +0200453
454" must be last, we can't go back from GUI to terminal
455func Test_zz_terminal_in_gui()
Bram Moolenaar9f0139a2017-08-13 20:26:20 +0200456 if !CanRunGui()
Bram Moolenaar679653e2017-08-13 14:13:19 +0200457 return
458 endif
Bram Moolenaar97f65fa2017-08-29 20:42:07 +0200459
460 " Ignore the "failed to create input context" error.
461 call test_ignore_error('E285:')
462
Bram Moolenaar679653e2017-08-13 14:13:19 +0200463 gui -f
464
465 call assert_equal(1, winnr('$'))
466 let buf = Run_shell_in_terminal({'term_finish': 'close'})
467 call Stop_shell_in_terminal(buf)
468 call term_wait(buf)
469
470 " closing window wipes out the terminal buffer a with finished job
471 call WaitFor("winnr('$') == 1")
472 call assert_equal(1, winnr('$'))
473 call assert_equal("", bufname(buf))
474
475 unlet g:job
476endfunc
Bram Moolenaardcaa6132017-08-13 17:13:09 +0200477
478func Test_terminal_list_args()
479 let buf = term_start([&shell, &shellcmdflag, 'echo "123"'])
480 call assert_fails(buf . 'bwipe', 'E517')
481 exe buf . 'bwipe!'
482 call assert_equal("", bufname(buf))
483endfunction
Bram Moolenaar97bd5e62017-08-18 20:50:30 +0200484
485func Test_terminal_noblock()
Bram Moolenaard21f8b52017-08-19 15:40:01 +0200486 let g:buf = term_start(&shell)
Bram Moolenaard8d85bf2017-09-03 18:08:00 +0200487 if has('mac')
488 " The shell or something else has a problem dealing with more than 1000
489 " characters at the same time.
490 let len = 1000
491 else
492 let len = 5000
493 endif
Bram Moolenaar97bd5e62017-08-18 20:50:30 +0200494
495 for c in ['a','b','c','d','e','f','g','h','i','j','k']
Bram Moolenaard8d85bf2017-09-03 18:08:00 +0200496 call term_sendkeys(g:buf, 'echo ' . repeat(c, len) . "\<cr>")
Bram Moolenaar97bd5e62017-08-18 20:50:30 +0200497 endfor
Bram Moolenaard21f8b52017-08-19 15:40:01 +0200498 call term_sendkeys(g:buf, "echo done\<cr>")
Bram Moolenaareef05312017-08-20 20:21:23 +0200499
500 " On MS-Windows there is an extra empty line below "done". Find "done" in
501 " the last-but-one or the last-but-two line.
Bram Moolenaard21f8b52017-08-19 15:40:01 +0200502 let g:lnum = term_getsize(g:buf)[0] - 1
Bram Moolenaareef05312017-08-20 20:21:23 +0200503 call WaitFor('term_getline(g:buf, g:lnum) =~ "done" || term_getline(g:buf, g:lnum - 1) =~ "done"', 3000)
504 let line = term_getline(g:buf, g:lnum)
505 if line !~ 'done'
506 let line = term_getline(g:buf, g:lnum - 1)
507 endif
508 call assert_match('done', line)
Bram Moolenaar97bd5e62017-08-18 20:50:30 +0200509
Bram Moolenaard21f8b52017-08-19 15:40:01 +0200510 let g:job = term_getjob(g:buf)
511 call Stop_shell_in_terminal(g:buf)
512 call term_wait(g:buf)
513 unlet g:buf
514 unlet g:job
515 unlet g:lnum
Bram Moolenaar97bd5e62017-08-18 20:50:30 +0200516 bwipe
517endfunc
Bram Moolenaar37819ed2017-08-20 19:33:47 +0200518
519func Test_terminal_write_stdin()
Bram Moolenaar3346cc42017-09-02 14:54:21 +0200520 if !executable('wc')
Bram Moolenaardada6d22017-09-02 17:18:35 +0200521 throw 'skipped: wc command not available'
Bram Moolenaar37819ed2017-08-20 19:33:47 +0200522 endif
523 new
524 call setline(1, ['one', 'two', 'three'])
525 %term wc
Bram Moolenaardada6d22017-09-02 17:18:35 +0200526 call WaitFor('getline("$") =~ "3"')
Bram Moolenaar3346cc42017-09-02 14:54:21 +0200527 let nrs = split(getline('$'))
Bram Moolenaar37819ed2017-08-20 19:33:47 +0200528 call assert_equal(['3', '3', '14'], nrs)
529 bwipe
530
Bram Moolenaardada6d22017-09-02 17:18:35 +0200531 new
Bram Moolenaar37819ed2017-08-20 19:33:47 +0200532 call setline(1, ['one', 'two', 'three', 'four'])
533 2,3term wc
Bram Moolenaardada6d22017-09-02 17:18:35 +0200534 call WaitFor('getline("$") =~ "2"')
Bram Moolenaar3346cc42017-09-02 14:54:21 +0200535 let nrs = split(getline('$'))
Bram Moolenaar37819ed2017-08-20 19:33:47 +0200536 call assert_equal(['2', '2', '10'], nrs)
537 bwipe
538
Bram Moolenaardada6d22017-09-02 17:18:35 +0200539 if executable('python')
540 new
541 call setline(1, ['print("hello")'])
542 1term ++eof=exit() python
543 " MS-Windows echoes the input, Unix doesn't.
544 call WaitFor('getline("$") =~ "exit" || getline(1) =~ "hello"')
545 if getline(1) =~ 'hello'
546 call assert_equal('hello', getline(1))
547 else
548 call assert_equal('hello', getline(line('$') - 1))
549 endif
550 bwipe
551
552 if has('win32')
553 new
554 call setline(1, ['print("hello")'])
555 1term ++eof=<C-Z> python
556 call WaitFor('getline("$") =~ "Z"')
557 call assert_equal('hello', getline(line('$') - 1))
558 bwipe
559 endif
560 endif
561
Bram Moolenaar37819ed2017-08-20 19:33:47 +0200562 bwipe!
563endfunc
Bram Moolenaar13ebb032017-08-26 22:02:51 +0200564
565func Test_terminal_no_cmd()
Bram Moolenaar13ebb032017-08-26 22:02:51 +0200566 " Todo: make this work in the GUI
567 if !has('gui_running')
568 return
569 endif
570 let buf = term_start('NONE', {})
571 call assert_notequal(0, buf)
572
Bram Moolenaar2dc9d262017-09-08 14:39:30 +0200573 let pty = job_info(term_getjob(buf))['tty_out']
Bram Moolenaar13ebb032017-08-26 22:02:51 +0200574 call assert_notequal('', pty)
Bram Moolenaar2dc9d262017-09-08 14:39:30 +0200575 if has('win32')
Bram Moolenaare738a1a2017-09-16 17:42:41 +0200576 silent exe '!start cmd /c "echo look here > ' . pty . '"'
Bram Moolenaar2dc9d262017-09-08 14:39:30 +0200577 else
578 call system('echo "look here" > ' . pty)
579 endif
Bram Moolenaare738a1a2017-09-16 17:42:41 +0200580 let g:buf = buf
581 call WaitFor('term_getline(g:buf, 1) =~ "look here"')
Bram Moolenaar2dc9d262017-09-08 14:39:30 +0200582
Bram Moolenaare738a1a2017-09-16 17:42:41 +0200583 call assert_match('look here', term_getline(buf, 1))
Bram Moolenaar13ebb032017-08-26 22:02:51 +0200584 bwipe!
585endfunc
Bram Moolenaar9d654a82017-09-03 19:52:17 +0200586
587func Test_terminal_special_chars()
588 " this file name only works on Unix
589 if !has('unix')
590 return
591 endif
592 call mkdir('Xdir with spaces')
593 call writefile(['x'], 'Xdir with spaces/quoted"file')
594 term ls Xdir\ with\ spaces/quoted\"file
595 call WaitFor('term_getline("", 1) =~ "quoted"')
596 call assert_match('quoted"file', term_getline('', 1))
597 call term_wait('')
598
599 call delete('Xdir with spaces', 'rf')
600 bwipe
601endfunc
Bram Moolenaare88fc7a2017-09-03 20:59:40 +0200602
603func Test_terminal_wrong_options()
604 call assert_fails('call term_start(&shell, {
605 \ "in_io": "file",
606 \ "in_name": "xxx",
607 \ "out_io": "file",
608 \ "out_name": "xxx",
609 \ "err_io": "file",
610 \ "err_name": "xxx"
611 \ })', 'E474:')
612 call assert_fails('call term_start(&shell, {
613 \ "out_buf": bufnr("%")
614 \ })', 'E474:')
615 call assert_fails('call term_start(&shell, {
616 \ "err_buf": bufnr("%")
617 \ })', 'E474:')
618endfunc
619
620func Test_terminal_redir_file()
Bram Moolenaar17833372017-09-04 22:23:19 +0200621 " TODO: this should work on MS-Window
622 if has('unix')
623 let cmd = Get_cat_123_cmd()
624 let buf = term_start(cmd, {'out_io': 'file', 'out_name': 'Xfile'})
625 call term_wait(buf)
626 call WaitFor('len(readfile("Xfile")) > 0')
627 call assert_match('123', readfile('Xfile')[0])
Bram Moolenaare9f6fd22017-09-10 14:25:49 +0200628 let g:job = term_getjob(buf)
629 call WaitFor('job_status(g:job) == "dead"')
Bram Moolenaar17833372017-09-04 22:23:19 +0200630 call delete('Xfile')
Bram Moolenaar2dc9d262017-09-08 14:39:30 +0200631 bwipe
Bram Moolenaar17833372017-09-04 22:23:19 +0200632 endif
Bram Moolenaare88fc7a2017-09-03 20:59:40 +0200633
634 if has('unix')
Bram Moolenaare88fc7a2017-09-03 20:59:40 +0200635 call writefile(['one line'], 'Xfile')
636 let buf = term_start('cat', {'in_io': 'file', 'in_name': 'Xfile'})
637 call term_wait(buf)
638 call WaitFor('term_getline(' . buf . ', 1) == "one line"')
639 call assert_equal('one line', term_getline(buf, 1))
Bram Moolenaar8b53b792017-09-05 20:29:25 +0200640 let g:job = term_getjob(buf)
641 call WaitFor('job_status(g:job) == "dead"')
Bram Moolenaare88fc7a2017-09-03 20:59:40 +0200642 bwipe
643 call delete('Xfile')
644 endif
645endfunc
Bram Moolenaar69fbc9e2017-09-14 20:37:57 +0200646
647func TerminalTmap(remap)
648 let buf = Run_shell_in_terminal({})
649 call assert_equal('t', mode())
650
651 if a:remap
652 tmap 123 456
653 else
654 tnoremap 123 456
655 endif
656 tmap 456 abcde
657 call assert_equal('456', maparg('123', 't'))
658 call assert_equal('abcde', maparg('456', 't'))
659 call feedkeys("123", 'tx')
Bram Moolenaar1514e8f2017-09-16 17:35:13 +0200660 let g:buf = buf
661 call WaitFor("term_getline(g:buf,term_getcursor(g:buf)[0]) =~ 'abcde\\|456'")
Bram Moolenaar69fbc9e2017-09-14 20:37:57 +0200662 let lnum = term_getcursor(buf)[0]
663 if a:remap
664 call assert_match('abcde', term_getline(buf, lnum))
665 else
666 call assert_match('456', term_getline(buf, lnum))
667 endif
668
669 call term_sendkeys(buf, "\r")
670 call Stop_shell_in_terminal(buf)
671 call term_wait(buf)
672
673 tunmap 123
674 tunmap 456
675 call assert_equal('', maparg('123', 't'))
676 close
677 unlet g:job
678endfunc
679
680func Test_terminal_tmap()
681 call TerminalTmap(1)
682 call TerminalTmap(0)
683endfunc
Bram Moolenaar059db5c2017-10-15 22:42:23 +0200684
685func Test_terminal_wall()
686 let buf = Run_shell_in_terminal({})
687 wall
688 call Stop_shell_in_terminal(buf)
689 call term_wait(buf)
690 exe buf . 'bwipe'
691 unlet g:job
692endfunc
Bram Moolenaar6daeef12017-10-15 22:56:49 +0200693
694func Test_terminal_composing_unicode()
695 let save_enc = &encoding
696 set encoding=utf-8
697
698 if has('win32')
699 let cmd = "cmd /K chcp 65001"
700 let lnum = [3, 6, 9]
701 else
702 let cmd = &shell
703 let lnum = [1, 3, 5]
704 endif
705
706 enew
707 let buf = term_start(cmd, {'curwin': bufnr('')})
Bram Moolenaar3e1c6172017-11-02 16:58:00 +0100708 let g:job = term_getjob(buf)
Bram Moolenaar6daeef12017-10-15 22:56:49 +0200709 call term_wait(buf, 50)
710
711 " ascii + composing
712 let txt = "a\u0308bc"
713 call term_sendkeys(buf, "echo " . txt . "\r")
714 call term_wait(buf, 50)
715 call assert_match("echo " . txt, term_getline(buf, lnum[0]))
716 call assert_equal(txt, term_getline(buf, lnum[0] + 1))
717 let l = term_scrape(buf, lnum[0] + 1)
718 call assert_equal("a\u0308", l[0].chars)
719 call assert_equal("b", l[1].chars)
720 call assert_equal("c", l[2].chars)
721
722 " multibyte + composing
723 let txt = "\u304b\u3099\u304e\u304f\u3099\u3052\u3053\u3099"
724 call term_sendkeys(buf, "echo " . txt . "\r")
725 call term_wait(buf, 50)
726 call assert_match("echo " . txt, term_getline(buf, lnum[1]))
727 call assert_equal(txt, term_getline(buf, lnum[1] + 1))
728 let l = term_scrape(buf, lnum[1] + 1)
729 call assert_equal("\u304b\u3099", l[0].chars)
730 call assert_equal("\u304e", l[1].chars)
731 call assert_equal("\u304f\u3099", l[2].chars)
732 call assert_equal("\u3052", l[3].chars)
733 call assert_equal("\u3053\u3099", l[4].chars)
734
735 " \u00a0 + composing
736 let txt = "abc\u00a0\u0308"
737 call term_sendkeys(buf, "echo " . txt . "\r")
738 call term_wait(buf, 50)
739 call assert_match("echo " . txt, term_getline(buf, lnum[2]))
740 call assert_equal(txt, term_getline(buf, lnum[2] + 1))
741 let l = term_scrape(buf, lnum[2] + 1)
742 call assert_equal("\u00a0\u0308", l[3].chars)
743
744 call term_sendkeys(buf, "exit\r")
Bram Moolenaar3e1c6172017-11-02 16:58:00 +0100745 call WaitFor('job_status(g:job) == "dead"')
746 call assert_equal('dead', job_status(g:job))
Bram Moolenaar6daeef12017-10-15 22:56:49 +0200747 bwipe!
Bram Moolenaar3e1c6172017-11-02 16:58:00 +0100748 unlet g:job
Bram Moolenaar6daeef12017-10-15 22:56:49 +0200749 let &encoding = save_enc
750endfunc