buffer_spec.lua 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457
  1. local helpers = require('test.functional.helpers')(after_each)
  2. local clear, nvim, buffer = helpers.clear, helpers.nvim, helpers.buffer
  3. local curbuf, curwin, eq = helpers.curbuf, helpers.curwin, helpers.eq
  4. local curbufmeths, ok = helpers.curbufmeths, helpers.ok
  5. local funcs = helpers.funcs
  6. local request = helpers.request
  7. local exc_exec = helpers.exc_exec
  8. local feed_command = helpers.feed_command
  9. local insert = helpers.insert
  10. local NIL = helpers.NIL
  11. local meth_pcall = helpers.meth_pcall
  12. local command = helpers.command
  13. local bufmeths = helpers.bufmeths
  14. describe('api/buf', function()
  15. before_each(clear)
  16. -- access deprecated functions
  17. local function curbuf_depr(method, ...)
  18. return request('buffer_'..method, 0, ...)
  19. end
  20. describe('line_count, insert and del_line', function()
  21. it('works', function()
  22. eq(1, curbuf_depr('line_count'))
  23. curbuf_depr('insert', -1, {'line'})
  24. eq(2, curbuf_depr('line_count'))
  25. curbuf_depr('insert', -1, {'line'})
  26. eq(3, curbuf_depr('line_count'))
  27. curbuf_depr('del_line', -1)
  28. eq(2, curbuf_depr('line_count'))
  29. curbuf_depr('del_line', -1)
  30. curbuf_depr('del_line', -1)
  31. -- There's always at least one line
  32. eq(1, curbuf_depr('line_count'))
  33. end)
  34. it('line_count has defined behaviour for unloaded buffers', function()
  35. -- we'll need to know our bufnr for when it gets unloaded
  36. local bufnr = curbuf('get_number')
  37. -- replace the buffer contents with these three lines
  38. request('nvim_buf_set_lines', bufnr, 0, -1, 1, {"line1", "line2", "line3", "line4"})
  39. -- check the line count is correct
  40. eq(4, request('nvim_buf_line_count', bufnr))
  41. -- force unload the buffer (this will discard changes)
  42. command('new')
  43. command('bunload! '..bufnr)
  44. -- line count for an unloaded buffer should always be 0
  45. eq(0, request('nvim_buf_line_count', bufnr))
  46. end)
  47. it('get_lines has defined behaviour for unloaded buffers', function()
  48. -- we'll need to know our bufnr for when it gets unloaded
  49. local bufnr = curbuf('get_number')
  50. -- replace the buffer contents with these three lines
  51. buffer('set_lines', bufnr, 0, -1, 1, {"line1", "line2", "line3", "line4"})
  52. -- confirm that getting lines works
  53. eq({"line2", "line3"}, buffer('get_lines', bufnr, 1, 3, 1))
  54. -- force unload the buffer (this will discard changes)
  55. command('new')
  56. command('bunload! '..bufnr)
  57. -- attempting to get lines now always gives empty list
  58. eq({}, buffer('get_lines', bufnr, 1, 3, 1))
  59. -- it's impossible to get out-of-bounds errors for an unloaded buffer
  60. eq({}, buffer('get_lines', bufnr, 8888, 9999, 1))
  61. end)
  62. end)
  63. describe('{get,set,del}_line', function()
  64. it('works', function()
  65. eq('', curbuf_depr('get_line', 0))
  66. curbuf_depr('set_line', 0, 'line1')
  67. eq('line1', curbuf_depr('get_line', 0))
  68. curbuf_depr('set_line', 0, 'line2')
  69. eq('line2', curbuf_depr('get_line', 0))
  70. curbuf_depr('del_line', 0)
  71. eq('', curbuf_depr('get_line', 0))
  72. end)
  73. it('get_line: out-of-bounds is an error', function()
  74. curbuf_depr('set_line', 0, 'line1.a')
  75. eq(1, curbuf_depr('line_count')) -- sanity
  76. eq(false, pcall(curbuf_depr, 'get_line', 1))
  77. eq(false, pcall(curbuf_depr, 'get_line', -2))
  78. end)
  79. it('set_line, del_line: out-of-bounds is an error', function()
  80. curbuf_depr('set_line', 0, 'line1.a')
  81. eq(false, pcall(curbuf_depr, 'set_line', 1, 'line1.b'))
  82. eq(false, pcall(curbuf_depr, 'set_line', -2, 'line1.b'))
  83. eq(false, pcall(curbuf_depr, 'del_line', 2))
  84. eq(false, pcall(curbuf_depr, 'del_line', -3))
  85. end)
  86. it('can handle NULs', function()
  87. curbuf_depr('set_line', 0, 'ab\0cd')
  88. eq('ab\0cd', curbuf_depr('get_line', 0))
  89. end)
  90. end)
  91. describe('{get,set}_line_slice', function()
  92. it('get_line_slice: out-of-bounds returns empty array', function()
  93. curbuf_depr('set_line_slice', 0, 0, true, true, {'a', 'b', 'c'})
  94. eq({'a', 'b', 'c'}, curbuf_depr('get_line_slice', 0, 2, true, true)) --sanity
  95. eq({}, curbuf_depr('get_line_slice', 2, 3, false, true))
  96. eq({}, curbuf_depr('get_line_slice', 3, 9, true, true))
  97. eq({}, curbuf_depr('get_line_slice', 3, -1, true, true))
  98. eq({}, curbuf_depr('get_line_slice', -3, -4, false, true))
  99. eq({}, curbuf_depr('get_line_slice', -4, -5, true, true))
  100. end)
  101. it('set_line_slice: out-of-bounds extends past end', function()
  102. curbuf_depr('set_line_slice', 0, 0, true, true, {'a', 'b', 'c'})
  103. eq({'a', 'b', 'c'}, curbuf_depr('get_line_slice', 0, 2, true, true)) --sanity
  104. eq({'c'}, curbuf_depr('get_line_slice', -1, 4, true, true))
  105. eq({'a', 'b', 'c'}, curbuf_depr('get_line_slice', 0, 5, true, true))
  106. curbuf_depr('set_line_slice', 4, 5, true, true, {'d'})
  107. eq({'a', 'b', 'c', 'd'}, curbuf_depr('get_line_slice', 0, 5, true, true))
  108. curbuf_depr('set_line_slice', -4, -5, true, true, {'e'})
  109. eq({'e', 'a', 'b', 'c', 'd'}, curbuf_depr('get_line_slice', 0, 5, true, true))
  110. end)
  111. it('works', function()
  112. eq({''}, curbuf_depr('get_line_slice', 0, -1, true, true))
  113. -- Replace buffer
  114. curbuf_depr('set_line_slice', 0, -1, true, true, {'a', 'b', 'c'})
  115. eq({'a', 'b', 'c'}, curbuf_depr('get_line_slice', 0, -1, true, true))
  116. eq({'b', 'c'}, curbuf_depr('get_line_slice', 1, -1, true, true))
  117. eq({'b'}, curbuf_depr('get_line_slice', 1, 2, true, false))
  118. eq({}, curbuf_depr('get_line_slice', 1, 1, true, false))
  119. eq({'a', 'b'}, curbuf_depr('get_line_slice', 0, -1, true, false))
  120. eq({'b'}, curbuf_depr('get_line_slice', 1, -1, true, false))
  121. eq({'b', 'c'}, curbuf_depr('get_line_slice', -2, -1, true, true))
  122. curbuf_depr('set_line_slice', 1, 2, true, false, {'a', 'b', 'c'})
  123. eq({'a', 'a', 'b', 'c', 'c'}, curbuf_depr('get_line_slice', 0, -1, true, true))
  124. curbuf_depr('set_line_slice', -1, -1, true, true, {'a', 'b', 'c'})
  125. eq({'a', 'a', 'b', 'c', 'a', 'b', 'c'},
  126. curbuf_depr('get_line_slice', 0, -1, true, true))
  127. curbuf_depr('set_line_slice', 0, -3, true, false, {})
  128. eq({'a', 'b', 'c'}, curbuf_depr('get_line_slice', 0, -1, true, true))
  129. curbuf_depr('set_line_slice', 0, -1, true, true, {})
  130. eq({''}, curbuf_depr('get_line_slice', 0, -1, true, true))
  131. end)
  132. end)
  133. describe('{get,set}_lines', function()
  134. local get_lines, set_lines = curbufmeths.get_lines, curbufmeths.set_lines
  135. local line_count = curbufmeths.line_count
  136. it('fails correctly when input is not valid', function()
  137. eq(1, curbufmeths.get_number())
  138. local err, emsg = pcall(bufmeths.set_lines, 1, 1, 2, false, {'b\na'})
  139. eq(false, err)
  140. local exp_emsg = 'String cannot contain newlines'
  141. -- Expected {filename}:{lnum}: {exp_emsg}
  142. eq(': ' .. exp_emsg, emsg:sub(-#exp_emsg - 2))
  143. end)
  144. it('has correct line_count when inserting and deleting', function()
  145. eq(1, line_count())
  146. set_lines(-1, -1, true, {'line'})
  147. eq(2, line_count())
  148. set_lines(-1, -1, true, {'line'})
  149. eq(3, line_count())
  150. set_lines(-2, -1, true, {})
  151. eq(2, line_count())
  152. set_lines(-2, -1, true, {})
  153. set_lines(-2, -1, true, {})
  154. -- There's always at least one line
  155. eq(1, line_count())
  156. end)
  157. it('can get, set and delete a single line', function()
  158. eq({''}, get_lines(0, 1, true))
  159. set_lines(0, 1, true, {'line1'})
  160. eq({'line1'}, get_lines(0, 1, true))
  161. set_lines(0, 1, true, {'line2'})
  162. eq({'line2'}, get_lines(0, 1, true))
  163. set_lines(0, 1, true, {})
  164. eq({''}, get_lines(0, 1, true))
  165. end)
  166. it('can get a single line with strict indexing', function()
  167. set_lines(0, 1, true, {'line1.a'})
  168. eq(1, line_count()) -- sanity
  169. eq(false, pcall(get_lines, 1, 2, true))
  170. eq(false, pcall(get_lines, -3, -2, true))
  171. end)
  172. it('can get a single line with non-strict indexing', function()
  173. set_lines(0, 1, true, {'line1.a'})
  174. eq(1, line_count()) -- sanity
  175. eq({}, get_lines(1, 2, false))
  176. eq({}, get_lines(-3, -2, false))
  177. end)
  178. it('can set and delete a single line with strict indexing', function()
  179. set_lines(0, 1, true, {'line1.a'})
  180. eq(false, pcall(set_lines, 1, 2, true, {'line1.b'}))
  181. eq(false, pcall(set_lines, -3, -2, true, {'line1.c'}))
  182. eq({'line1.a'}, get_lines(0, -1, true))
  183. eq(false, pcall(set_lines, 1, 2, true, {}))
  184. eq(false, pcall(set_lines, -3, -2, true, {}))
  185. eq({'line1.a'}, get_lines(0, -1, true))
  186. end)
  187. it('can set and delete a single line with non-strict indexing', function()
  188. set_lines(0, 1, true, {'line1.a'})
  189. set_lines(1, 2, false, {'line1.b'})
  190. set_lines(-4, -3, false, {'line1.c'})
  191. eq({'line1.c', 'line1.a', 'line1.b'}, get_lines(0, -1, true))
  192. set_lines(3, 4, false, {})
  193. set_lines(-5, -4, false, {})
  194. eq({'line1.c', 'line1.a', 'line1.b'}, get_lines(0, -1, true))
  195. end)
  196. it('can handle NULs', function()
  197. set_lines(0, 1, true, {'ab\0cd'})
  198. eq({'ab\0cd'}, get_lines(0, -1, true))
  199. end)
  200. it('works with multiple lines', function()
  201. eq({''}, get_lines(0, -1, true))
  202. -- Replace buffer
  203. for _, mode in pairs({false, true}) do
  204. set_lines(0, -1, mode, {'a', 'b', 'c'})
  205. eq({'a', 'b', 'c'}, get_lines(0, -1, mode))
  206. eq({'b', 'c'}, get_lines(1, -1, mode))
  207. eq({'b'}, get_lines(1, 2, mode))
  208. eq({}, get_lines(1, 1, mode))
  209. eq({'a', 'b'}, get_lines(0, -2, mode))
  210. eq({'b'}, get_lines(1, -2, mode))
  211. eq({'b', 'c'}, get_lines(-3, -1, mode))
  212. set_lines(1, 2, mode, {'a', 'b', 'c'})
  213. eq({'a', 'a', 'b', 'c', 'c'}, get_lines(0, -1, mode))
  214. set_lines(-2, -1, mode, {'a', 'b', 'c'})
  215. eq({'a', 'a', 'b', 'c', 'a', 'b', 'c'},
  216. get_lines(0, -1, mode))
  217. set_lines(0, -4, mode, {})
  218. eq({'a', 'b', 'c'}, get_lines(0, -1, mode))
  219. set_lines(0, -1, mode, {})
  220. eq({''}, get_lines(0, -1, mode))
  221. end
  222. end)
  223. it('can get line ranges with non-strict indexing', function()
  224. set_lines(0, -1, true, {'a', 'b', 'c'})
  225. eq({'a', 'b', 'c'}, get_lines(0, -1, true)) --sanity
  226. eq({}, get_lines(3, 4, false))
  227. eq({}, get_lines(3, 10, false))
  228. eq({}, get_lines(-5, -5, false))
  229. eq({}, get_lines(3, -1, false))
  230. eq({}, get_lines(-3, -4, false))
  231. end)
  232. it('can get line ranges with strict indexing', function()
  233. set_lines(0, -1, true, {'a', 'b', 'c'})
  234. eq({'a', 'b', 'c'}, get_lines(0, -1, true)) --sanity
  235. eq(false, pcall(get_lines, 3, 4, true))
  236. eq(false, pcall(get_lines, 3, 10, true))
  237. eq(false, pcall(get_lines, -5, -5, true))
  238. -- empty or inverted ranges are not errors
  239. eq({}, get_lines(3, -1, true))
  240. eq({}, get_lines(-3, -4, true))
  241. end)
  242. it('set_line_slice: out-of-bounds can extend past end', function()
  243. set_lines(0, -1, true, {'a', 'b', 'c'})
  244. eq({'a', 'b', 'c'}, get_lines(0, -1, true)) --sanity
  245. eq({'c'}, get_lines(-2, 5, false))
  246. eq({'a', 'b', 'c'}, get_lines(0, 6, false))
  247. eq(false, pcall(set_lines, 4, 6, true, {'d'}))
  248. set_lines(4, 6, false, {'d'})
  249. eq({'a', 'b', 'c', 'd'}, get_lines(0, -1, true))
  250. eq(false, pcall(set_lines, -6, -6, true, {'e'}))
  251. set_lines(-6, -6, false, {'e'})
  252. eq({'e', 'a', 'b', 'c', 'd'}, get_lines(0, -1, true))
  253. end)
  254. it("set_line on alternate buffer does not access invalid line (E315)", function()
  255. feed_command('set hidden')
  256. insert('Initial file')
  257. command('enew')
  258. insert([[
  259. More
  260. Lines
  261. Than
  262. In
  263. The
  264. Other
  265. Buffer]])
  266. feed_command('$')
  267. local retval = exc_exec("call nvim_buf_set_lines(1, 0, 1, v:false, ['test'])")
  268. eq(0, retval)
  269. end)
  270. end)
  271. describe('get_offset', function()
  272. local get_offset = curbufmeths.get_offset
  273. it('works', function()
  274. curbufmeths.set_lines(0,-1,true,{'Some\r','exa\000mple', '', 'buf\rfer', 'text'})
  275. eq(5, curbufmeths.line_count())
  276. eq(0, get_offset(0))
  277. eq(6, get_offset(1))
  278. eq(15, get_offset(2))
  279. eq(16, get_offset(3))
  280. eq(24, get_offset(4))
  281. eq(29, get_offset(5))
  282. eq({false,'Index out of bounds'}, meth_pcall(get_offset, 6))
  283. eq({false,'Index out of bounds'}, meth_pcall(get_offset, -1))
  284. curbufmeths.set_option('eol', false)
  285. curbufmeths.set_option('fixeol', false)
  286. eq(28, get_offset(5))
  287. -- fileformat is ignored
  288. curbufmeths.set_option('fileformat', 'dos')
  289. eq(0, get_offset(0))
  290. eq(6, get_offset(1))
  291. eq(15, get_offset(2))
  292. eq(16, get_offset(3))
  293. eq(24, get_offset(4))
  294. eq(28, get_offset(5))
  295. curbufmeths.set_option('eol', true)
  296. eq(29, get_offset(5))
  297. command("set hidden")
  298. command("enew")
  299. eq(6, bufmeths.get_offset(1,1))
  300. command("bunload! 1")
  301. eq(-1, bufmeths.get_offset(1,1))
  302. end)
  303. end)
  304. describe('{get,set,del}_var', function()
  305. it('works', function()
  306. curbuf('set_var', 'lua', {1, 2, {['3'] = 1}})
  307. eq({1, 2, {['3'] = 1}}, curbuf('get_var', 'lua'))
  308. eq({1, 2, {['3'] = 1}}, nvim('eval', 'b:lua'))
  309. eq(1, funcs.exists('b:lua'))
  310. curbufmeths.del_var('lua')
  311. eq(0, funcs.exists('b:lua'))
  312. eq({false, 'Key not found: lua'}, meth_pcall(curbufmeths.del_var, 'lua'))
  313. curbufmeths.set_var('lua', 1)
  314. command('lockvar b:lua')
  315. eq({false, 'Key is locked: lua'}, meth_pcall(curbufmeths.del_var, 'lua'))
  316. eq({false, 'Key is locked: lua'}, meth_pcall(curbufmeths.set_var, 'lua', 1))
  317. eq({false, 'Key is read-only: changedtick'},
  318. meth_pcall(curbufmeths.del_var, 'changedtick'))
  319. eq({false, 'Key is read-only: changedtick'},
  320. meth_pcall(curbufmeths.set_var, 'changedtick', 1))
  321. end)
  322. end)
  323. describe('get_changedtick', function()
  324. it('works', function()
  325. eq(2, curbufmeths.get_changedtick())
  326. curbufmeths.set_lines(0, 1, false, {'abc\0', '\0def', 'ghi'})
  327. eq(3, curbufmeths.get_changedtick())
  328. eq(3, curbufmeths.get_var('changedtick'))
  329. end)
  330. it('buffer_set_var returns the old value', function()
  331. local val1 = {1, 2, {['3'] = 1}}
  332. local val2 = {4, 7}
  333. eq(NIL, request('buffer_set_var', 0, 'lua', val1))
  334. eq(val1, request('buffer_set_var', 0, 'lua', val2))
  335. end)
  336. it('buffer_del_var returns the old value', function()
  337. local val1 = {1, 2, {['3'] = 1}}
  338. local val2 = {4, 7}
  339. eq(NIL, request('buffer_set_var', 0, 'lua', val1))
  340. eq(val1, request('buffer_set_var', 0, 'lua', val2))
  341. eq(val2, request('buffer_del_var', 0, 'lua'))
  342. end)
  343. end)
  344. describe('{get,set}_option', function()
  345. it('works', function()
  346. eq(8, curbuf('get_option', 'shiftwidth'))
  347. curbuf('set_option', 'shiftwidth', 4)
  348. eq(4, curbuf('get_option', 'shiftwidth'))
  349. -- global-local option
  350. curbuf('set_option', 'define', 'test')
  351. eq('test', curbuf('get_option', 'define'))
  352. -- Doesn't change the global value
  353. eq([[^\s*#\s*define]], nvim('get_option', 'define'))
  354. end)
  355. end)
  356. describe('{get,set}_name', function()
  357. it('works', function()
  358. nvim('command', 'new')
  359. eq('', curbuf('get_name'))
  360. local new_name = nvim('eval', 'resolve(tempname())')
  361. curbuf('set_name', new_name)
  362. eq(new_name, curbuf('get_name'))
  363. nvim('command', 'w!')
  364. local f = io.open(new_name)
  365. ok(f ~= nil)
  366. f:close()
  367. os.remove(new_name)
  368. end)
  369. end)
  370. describe('is_loaded', function()
  371. it('works', function()
  372. -- record our buffer number for when we unload it
  373. local bufnr = curbuf('get_number')
  374. -- api should report that the buffer is loaded
  375. ok(buffer('is_loaded', bufnr))
  376. -- hide the current buffer by switching to a new empty buffer
  377. -- Careful! we need to modify the buffer first or vim will just reuse it
  378. buffer('set_lines', bufnr, 0, -1, 1, {'line1'})
  379. command('hide enew')
  380. -- confirm the buffer is hidden, but still loaded
  381. local infolist = nvim('eval', 'getbufinfo('..bufnr..')')
  382. eq(1, #infolist)
  383. eq(1, infolist[1].hidden)
  384. eq(1, infolist[1].loaded)
  385. -- now force unload the buffer
  386. command('bunload! '..bufnr)
  387. -- confirm the buffer is unloaded
  388. infolist = nvim('eval', 'getbufinfo('..bufnr..')')
  389. eq(0, infolist[1].loaded)
  390. -- nvim_buf_is_loaded() should also report the buffer as unloaded
  391. eq(false, buffer('is_loaded', bufnr))
  392. end)
  393. end)
  394. describe('is_valid', function()
  395. it('works', function()
  396. nvim('command', 'new')
  397. local b = nvim('get_current_buf')
  398. ok(buffer('is_valid', b))
  399. nvim('command', 'bw!')
  400. ok(not buffer('is_valid', b))
  401. end)
  402. end)
  403. describe('get_mark', function()
  404. it('works', function()
  405. curbuf('set_lines', -1, -1, true, {'a', 'bit of', 'text'})
  406. curwin('set_cursor', {3, 4})
  407. nvim('command', 'mark V')
  408. eq({3, 0}, curbuf('get_mark', 'V'))
  409. end)
  410. end)
  411. end)