Bram Moolenaar | da65058 | 2018-02-20 15:51:40 +0100 | [diff] [blame] | 1 | " Functions shared by tests making screen dumps. |
| 2 | |
| 3 | " Only load this script once. |
Bram Moolenaar | 6bb2cdf | 2018-02-24 19:53:53 +0100 | [diff] [blame] | 4 | if exists('*CanRunVimInTerminal') |
Bram Moolenaar | da65058 | 2018-02-20 15:51:40 +0100 | [diff] [blame] | 5 | finish |
| 6 | endif |
| 7 | |
Bram Moolenaar | f273245 | 2018-06-03 14:47:35 +0200 | [diff] [blame] | 8 | " For most tests we need to be able to run terminal Vim with 256 colors. On |
| 9 | " MS-Windows the console only has 16 colors and the GUI can't run in a |
| 10 | " terminal. |
| 11 | func CanRunVimInTerminal() |
| 12 | return has('terminal') && !has('win32') |
| 13 | endfunc |
| 14 | |
| 15 | " Skip the rest if there is no terminal feature at all. |
| 16 | if !has('terminal') |
Bram Moolenaar | 6bb2cdf | 2018-02-24 19:53:53 +0100 | [diff] [blame] | 17 | finish |
| 18 | endif |
| 19 | |
Bram Moolenaar | da65058 | 2018-02-20 15:51:40 +0100 | [diff] [blame] | 20 | source shared.vim |
| 21 | |
| 22 | " Run Vim with "arguments" in a new terminal window. |
| 23 | " By default uses a size of 20 lines and 75 columns. |
| 24 | " Returns the buffer number of the terminal. |
| 25 | " |
Bram Moolenaar | cf67a50 | 2018-03-25 20:31:32 +0200 | [diff] [blame] | 26 | " Options is a dictionary, these items are recognized: |
| 27 | " "rows" - height of the terminal window (max. 20) |
| 28 | " "cols" - width of the terminal window (max. 78) |
Bram Moolenaar | 0fef0ae | 2019-05-01 20:30:40 +0200 | [diff] [blame] | 29 | " "statusoff" - number of lines the status is offset from default |
Bram Moolenaar | da65058 | 2018-02-20 15:51:40 +0100 | [diff] [blame] | 30 | func RunVimInTerminal(arguments, options) |
Bram Moolenaar | 948a796 | 2018-03-23 20:37:45 +0100 | [diff] [blame] | 31 | " If Vim doesn't exit a swap file remains, causing other tests to fail. |
| 32 | " Remove it here. |
| 33 | call delete(".swp") |
| 34 | |
Bram Moolenaar | e7499dd | 2018-03-24 17:56:13 +0100 | [diff] [blame] | 35 | if exists('$COLORFGBG') |
| 36 | " Clear $COLORFGBG to avoid 'background' being set to "dark", which will |
| 37 | " only be corrected if the response to t_RB is received, which may be too |
| 38 | " late. |
| 39 | let $COLORFGBG = '' |
| 40 | endif |
| 41 | |
Bram Moolenaar | da65058 | 2018-02-20 15:51:40 +0100 | [diff] [blame] | 42 | " Make a horizontal and vertical split, so that we can get exactly the right |
Bram Moolenaar | 8fbaeb1 | 2018-03-25 18:20:17 +0200 | [diff] [blame] | 43 | " size terminal window. Works only when the current window is full width. |
| 44 | call assert_equal(&columns, winwidth(0)) |
Bram Moolenaar | da65058 | 2018-02-20 15:51:40 +0100 | [diff] [blame] | 45 | split |
| 46 | vsplit |
| 47 | |
Bram Moolenaar | 6acadda | 2018-02-24 16:51:32 +0100 | [diff] [blame] | 48 | " Always do this with 256 colors and a light background. |
| 49 | set t_Co=256 background=light |
| 50 | hi Normal ctermfg=NONE ctermbg=NONE |
Bram Moolenaar | da65058 | 2018-02-20 15:51:40 +0100 | [diff] [blame] | 51 | |
Bram Moolenaar | cf67a50 | 2018-03-25 20:31:32 +0200 | [diff] [blame] | 52 | " Make the window 20 lines high and 75 columns, unless told otherwise. |
| 53 | let rows = get(a:options, 'rows', 20) |
| 54 | let cols = get(a:options, 'cols', 75) |
Bram Moolenaar | 0fef0ae | 2019-05-01 20:30:40 +0200 | [diff] [blame] | 55 | let statusoff = get(a:options, 'statusoff', 1) |
Bram Moolenaar | 15a1c3f | 2018-03-25 18:56:25 +0200 | [diff] [blame] | 56 | |
Bram Moolenaar | da65058 | 2018-02-20 15:51:40 +0100 | [diff] [blame] | 57 | let cmd = GetVimCommandClean() |
Bram Moolenaar | f273245 | 2018-06-03 14:47:35 +0200 | [diff] [blame] | 58 | |
Bram Moolenaar | b7ea7cb | 2018-02-24 14:38:51 +0100 | [diff] [blame] | 59 | " Add -v to have gvim run in the terminal (if possible) |
| 60 | let cmd .= ' -v ' . a:arguments |
Bram Moolenaar | f587ef3 | 2019-04-13 13:13:54 +0200 | [diff] [blame] | 61 | let buf = term_start(cmd, { |
| 62 | \ 'curwin': 1, |
| 63 | \ 'term_rows': rows, |
| 64 | \ 'term_cols': cols, |
Bram Moolenaar | f587ef3 | 2019-04-13 13:13:54 +0200 | [diff] [blame] | 65 | \ }) |
Bram Moolenaar | b833c1e | 2018-05-05 16:36:06 +0200 | [diff] [blame] | 66 | if &termwinsize == '' |
Bram Moolenaar | e40b9d4 | 2019-01-27 16:55:47 +0100 | [diff] [blame] | 67 | " in the GUI we may end up with a different size, try to set it. |
| 68 | if term_getsize(buf) != [rows, cols] |
| 69 | call term_setsize(buf, rows, cols) |
| 70 | endif |
Bram Moolenaar | a7eef3d | 2018-04-15 22:25:54 +0200 | [diff] [blame] | 71 | call assert_equal([rows, cols], term_getsize(buf)) |
| 72 | else |
| 73 | let rows = term_getsize(buf)[0] |
| 74 | let cols = term_getsize(buf)[1] |
| 75 | endif |
Bram Moolenaar | da65058 | 2018-02-20 15:51:40 +0100 | [diff] [blame] | 76 | |
Bram Moolenaar | f273245 | 2018-06-03 14:47:35 +0200 | [diff] [blame] | 77 | " Wait for "All" or "Top" of the ruler to be shown in the last line or in |
| 78 | " the status line of the last window. This can be quite slow (e.g. when |
| 79 | " using valgrind). |
Bram Moolenaar | 50182fa | 2018-04-28 21:34:40 +0200 | [diff] [blame] | 80 | " If it fails then show the terminal contents for debugging. |
| 81 | try |
Bram Moolenaar | 0fef0ae | 2019-05-01 20:30:40 +0200 | [diff] [blame] | 82 | call WaitFor({-> len(term_getline(buf, rows)) >= cols - 1 || len(term_getline(buf, rows - statusoff)) >= cols - 1}) |
Bram Moolenaar | 50182fa | 2018-04-28 21:34:40 +0200 | [diff] [blame] | 83 | catch /timed out after/ |
| 84 | let lines = map(range(1, rows), {key, val -> term_getline(buf, val)}) |
| 85 | call assert_report('RunVimInTerminal() failed, screen contents: ' . join(lines, "<NL>")) |
| 86 | endtry |
Bram Moolenaar | 1834d37 | 2018-03-29 17:40:46 +0200 | [diff] [blame] | 87 | |
Bram Moolenaar | da65058 | 2018-02-20 15:51:40 +0100 | [diff] [blame] | 88 | return buf |
| 89 | endfunc |
| 90 | |
| 91 | " Stop a Vim running in terminal buffer "buf". |
| 92 | func StopVimInTerminal(buf) |
| 93 | call assert_equal("running", term_getstatus(a:buf)) |
Bram Moolenaar | 3339d3d | 2018-06-03 17:10:40 +0200 | [diff] [blame] | 94 | |
| 95 | " CTRL-O : works both in Normal mode and Insert mode to start a command line. |
| 96 | " In Command-line it's inserted, the CTRL-U removes it again. |
Bram Moolenaar | acb9eff | 2018-06-04 19:11:11 +0200 | [diff] [blame] | 97 | call term_sendkeys(a:buf, "\<C-O>:\<C-U>qa!\<cr>") |
Bram Moolenaar | 3339d3d | 2018-06-03 17:10:40 +0200 | [diff] [blame] | 98 | |
Bram Moolenaar | 50182fa | 2018-04-28 21:34:40 +0200 | [diff] [blame] | 99 | call WaitForAssert({-> assert_equal("finished", term_getstatus(a:buf))}) |
Bram Moolenaar | da65058 | 2018-02-20 15:51:40 +0100 | [diff] [blame] | 100 | only! |
| 101 | endfunc |
| 102 | |
| 103 | " Verify that Vim running in terminal buffer "buf" matches the screen dump. |
Bram Moolenaar | 6bb2cdf | 2018-02-24 19:53:53 +0100 | [diff] [blame] | 104 | " "options" is passed to term_dumpwrite(). |
Bram Moolenaar | da65058 | 2018-02-20 15:51:40 +0100 | [diff] [blame] | 105 | " The file name used is "dumps/{filename}.dump". |
Bram Moolenaar | 6f8bdab | 2018-09-09 22:02:24 +0200 | [diff] [blame] | 106 | " Optionally an extra argument can be passed which is prepended to the error |
| 107 | " message. Use this when using the same dump file with different options. |
Bram Moolenaar | da65058 | 2018-02-20 15:51:40 +0100 | [diff] [blame] | 108 | " Will wait for up to a second for the screen dump to match. |
Bram Moolenaar | 6f8bdab | 2018-09-09 22:02:24 +0200 | [diff] [blame] | 109 | " Returns non-zero when verification fails. |
| 110 | func VerifyScreenDump(buf, filename, options, ...) |
Bram Moolenaar | da65058 | 2018-02-20 15:51:40 +0100 | [diff] [blame] | 111 | let reference = 'dumps/' . a:filename . '.dump' |
Bram Moolenaar | ef7f0e3 | 2019-03-30 15:59:51 +0100 | [diff] [blame] | 112 | let testfile = 'failed/' . a:filename . '.dump' |
| 113 | |
Bram Moolenaar | 87dcfd7 | 2019-04-13 22:35:29 +0200 | [diff] [blame] | 114 | " Redraw to execut the code that updates the screen. Otherwise we get the |
| 115 | " text and attributes only from the internal buffer. |
| 116 | redraw |
| 117 | |
Bram Moolenaar | ef7f0e3 | 2019-03-30 15:59:51 +0100 | [diff] [blame] | 118 | let did_mkdir = 0 |
| 119 | if !isdirectory('failed') |
| 120 | let did_mkdir = 1 |
| 121 | call mkdir('failed') |
| 122 | endif |
Bram Moolenaar | da65058 | 2018-02-20 15:51:40 +0100 | [diff] [blame] | 123 | |
| 124 | let i = 0 |
| 125 | while 1 |
Bram Moolenaar | b6fc728 | 2018-12-04 22:24:16 +0100 | [diff] [blame] | 126 | " leave some time for updating the original window |
| 127 | sleep 10m |
Bram Moolenaar | da65058 | 2018-02-20 15:51:40 +0100 | [diff] [blame] | 128 | call delete(testfile) |
Bram Moolenaar | 6bb2cdf | 2018-02-24 19:53:53 +0100 | [diff] [blame] | 129 | call term_dumpwrite(a:buf, testfile, a:options) |
Bram Moolenaar | 353aca1 | 2019-02-21 17:05:59 +0100 | [diff] [blame] | 130 | let testdump = readfile(testfile) |
Bram Moolenaar | 2d7260d | 2019-04-06 20:51:52 +0200 | [diff] [blame] | 131 | if filereadable(reference) |
| 132 | let refdump = readfile(reference) |
| 133 | else |
| 134 | " Must be a new screendump, always fail |
| 135 | let refdump = [] |
| 136 | endif |
Bram Moolenaar | 353aca1 | 2019-02-21 17:05:59 +0100 | [diff] [blame] | 137 | if refdump == testdump |
Bram Moolenaar | da65058 | 2018-02-20 15:51:40 +0100 | [diff] [blame] | 138 | call delete(testfile) |
Bram Moolenaar | ef7f0e3 | 2019-03-30 15:59:51 +0100 | [diff] [blame] | 139 | if did_mkdir |
| 140 | call delete('failed', 'd') |
| 141 | endif |
Bram Moolenaar | da65058 | 2018-02-20 15:51:40 +0100 | [diff] [blame] | 142 | break |
| 143 | endif |
| 144 | if i == 100 |
Bram Moolenaar | 2d7260d | 2019-04-06 20:51:52 +0200 | [diff] [blame] | 145 | " Leave the failed dump around for inspection. |
| 146 | if filereadable(reference) |
| 147 | let msg = 'See dump file difference: call term_dumpdiff("' . testfile . '", "' . reference . '")' |
| 148 | if a:0 == 1 |
| 149 | let msg = a:1 . ': ' . msg |
| 150 | endif |
| 151 | if len(testdump) != len(refdump) |
| 152 | let msg = msg . '; line count is ' . len(testdump) . ' instead of ' . len(refdump) |
| 153 | endif |
| 154 | else |
| 155 | let msg = 'See new dump file: call term_dumpload("' . testfile . '")' |
Bram Moolenaar | 353aca1 | 2019-02-21 17:05:59 +0100 | [diff] [blame] | 156 | endif |
| 157 | for i in range(len(refdump)) |
| 158 | if i >= len(testdump) |
| 159 | break |
| 160 | endif |
| 161 | if testdump[i] != refdump[i] |
| 162 | let msg = msg . '; difference in line ' . (i + 1) . ': "' . testdump[i] . '"' |
| 163 | endif |
| 164 | endfor |
Bram Moolenaar | 6f8bdab | 2018-09-09 22:02:24 +0200 | [diff] [blame] | 165 | call assert_report(msg) |
| 166 | return 1 |
Bram Moolenaar | da65058 | 2018-02-20 15:51:40 +0100 | [diff] [blame] | 167 | endif |
Bram Moolenaar | da65058 | 2018-02-20 15:51:40 +0100 | [diff] [blame] | 168 | let i += 1 |
| 169 | endwhile |
Bram Moolenaar | 6f8bdab | 2018-09-09 22:02:24 +0200 | [diff] [blame] | 170 | return 0 |
Bram Moolenaar | da65058 | 2018-02-20 15:51:40 +0100 | [diff] [blame] | 171 | endfunc |