aboutsummaryrefslogtreecommitdiff
path: root/test/functional/vimscript/executable_spec.lua
diff options
context:
space:
mode:
authorJustin M. Keyes <justinkz@gmail.com>2021-09-17 09:16:40 -0700
committerGitHub <noreply@github.com>2021-09-17 09:16:40 -0700
commitd8de4eb685e35646c7d541e9a75bdc296127b7e2 (patch)
tree4bb05ec713856715ac9ba57e5d116eed344511b9 /test/functional/vimscript/executable_spec.lua
parentd56002f7b722facd97b0958e141c8ed2d01495f7 (diff)
downloadrneovim-d8de4eb685e35646c7d541e9a75bdc296127b7e2.tar.gz
rneovim-d8de4eb685e35646c7d541e9a75bdc296127b7e2.tar.bz2
rneovim-d8de4eb685e35646c7d541e9a75bdc296127b7e2.zip
test: reorg #15698
Problem: Subdirectories like "visual", "insert", "normal" encourage people to separate *related* tests for no good reason. Typically the _mode_ is not the relevant topic of a test (and when it is, _then_ create an appropriate describe() or it()). Solution: - Delete the various `test/functional/<mode>/` subdirectories, move their tests to more meaningful topics. - Rename `…/normal/` to `…/editor/`. - Move or merge `…/visual/*` and `…/insert/*` tests into here where appropriate. - Rename `…/eval/` to `…/vimscript/`. - Move `…/viml/*` into here also. * test(reorg): insert/* => editor/mode_insert_spec.lua * test(reorg): cmdline/* => editor/mode_cmdline_spec.lua * test(reorg): eval core tests => eval_spec.lua
Diffstat (limited to 'test/functional/vimscript/executable_spec.lua')
-rw-r--r--test/functional/vimscript/executable_spec.lua218
1 files changed, 218 insertions, 0 deletions
diff --git a/test/functional/vimscript/executable_spec.lua b/test/functional/vimscript/executable_spec.lua
new file mode 100644
index 0000000000..28aefb72e5
--- /dev/null
+++ b/test/functional/vimscript/executable_spec.lua
@@ -0,0 +1,218 @@
+local helpers = require('test.functional.helpers')(after_each)
+local eq, clear, call, iswin, write_file, command =
+ helpers.eq, helpers.clear, helpers.call, helpers.iswin, helpers.write_file,
+ helpers.command
+local exc_exec = helpers.exc_exec
+local eval = helpers.eval
+
+describe('executable()', function()
+ before_each(clear)
+
+ it('returns 1 for commands in $PATH', function()
+ local exe = iswin() and 'ping' or 'ls'
+ eq(1, call('executable', exe))
+ command('let $PATH = fnamemodify("./test/functional/fixtures/bin", ":p")')
+ eq(1, call('executable', 'null'))
+ eq(1, call('executable', 'true'))
+ eq(1, call('executable', 'false'))
+ end)
+
+ it('fails for invalid values', function()
+ for _, input in ipairs({'""', 'v:null', 'v:true', 'v:false', '{}', '[]'}) do
+ eq('Vim(call):E928: String required', exc_exec('call executable('..input..')'))
+ end
+ command('let $PATH = fnamemodify("./test/functional/fixtures/bin", ":p")')
+ for _, input in ipairs({'v:null', 'v:true', 'v:false'}) do
+ eq('Vim(call):E928: String required', exc_exec('call executable('..input..')'))
+ end
+ end)
+
+ it('returns 0 for non-existent files', function()
+ eq(0, call('executable', 'no_such_file_exists_209ufq23f'))
+ end)
+
+ it('sibling to nvim binary', function()
+ -- Some executable in build/bin/, *not* in $PATH nor CWD.
+ local sibling_exe = 'printargs-test'
+ -- Windows: siblings are in Nvim's "pseudo-$PATH".
+ local expected = iswin() and 1 or 0
+ if iswin() then
+ eq('arg1=lemon;arg2=sky;arg3=tree;',
+ call('system', sibling_exe..' lemon sky tree'))
+ end
+ eq(expected, call('executable', sibling_exe))
+ end)
+
+ describe('exec-bit', function()
+ setup(function()
+ clear()
+ write_file('Xtest_not_executable', 'non-executable file')
+ write_file('Xtest_executable', 'executable file (exec-bit set)')
+ if not iswin() then -- N/A for Windows.
+ call('system', {'chmod', '-x', 'Xtest_not_executable'})
+ call('system', {'chmod', '+x', 'Xtest_executable'})
+ end
+ end)
+
+ teardown(function()
+ os.remove('Xtest_not_executable')
+ os.remove('Xtest_executable')
+ end)
+
+ it('not set', function()
+ eq(0, call('executable', 'Xtest_not_executable'))
+ eq(0, call('executable', './Xtest_not_executable'))
+ end)
+
+ it('set, unqualified and not in $PATH', function()
+ eq(0, call('executable', 'Xtest_executable'))
+ end)
+
+ it('set, qualified as a path', function()
+ local expected = iswin() and 0 or 1
+ eq(expected, call('executable', './Xtest_executable'))
+ end)
+ end)
+end)
+
+describe('executable() (Windows)', function()
+ if not iswin() then return end -- N/A for Unix.
+
+ local exts = {'bat', 'exe', 'com', 'cmd'}
+ setup(function()
+ for _, ext in ipairs(exts) do
+ write_file('test_executable_'..ext..'.'..ext, '')
+ end
+ write_file('test_executable_zzz.zzz', '')
+ end)
+
+ teardown(function()
+ for _, ext in ipairs(exts) do
+ os.remove('test_executable_'..ext..'.'..ext)
+ end
+ os.remove('test_executable_zzz.zzz')
+ end)
+
+ it('tries default extensions on a filename if $PATHEXT is empty', function()
+ -- Empty $PATHEXT defaults to ".com;.exe;.bat;.cmd".
+ clear({env={PATHEXT=''}})
+ for _,ext in ipairs(exts) do
+ eq(1, call('executable', 'test_executable_'..ext))
+ end
+ eq(0, call('executable', 'test_executable_zzz'))
+ end)
+
+ it('tries default extensions on a filepath if $PATHEXT is empty', function()
+ -- Empty $PATHEXT defaults to ".com;.exe;.bat;.cmd".
+ clear({env={PATHEXT=''}})
+ for _,ext in ipairs(exts) do
+ eq(1, call('executable', '.\\test_executable_'..ext))
+ end
+ eq(0, call('executable', '.\\test_executable_zzz'))
+ end)
+
+ it('system([…]), jobstart([…]) use $PATHEXT #9569', function()
+ -- Invoking `cmdscript` should find/execute `cmdscript.cmd`.
+ eq('much success\n', call('system', {'test/functional/fixtures/cmdscript'}))
+ assert(0 < call('jobstart', {'test/functional/fixtures/cmdscript'}))
+ end)
+
+ it('full path with extension', function()
+ -- Some executable we can expect in the test env.
+ local exe = 'printargs-test'
+ local exedir = eval("fnamemodify(v:progpath, ':h')")
+ local exepath = exedir..'/'..exe..'.exe'
+ eq(1, call('executable', exepath))
+ eq('arg1=lemon;arg2=sky;arg3=tree;',
+ call('system', exepath..' lemon sky tree'))
+ end)
+
+ it('full path without extension', function()
+ -- Some executable we can expect in the test env.
+ local exe = 'printargs-test'
+ local exedir = eval("fnamemodify(v:progpath, ':h')")
+ local exepath = exedir..'/'..exe
+ eq('arg1=lemon;arg2=sky;arg3=tree;',
+ call('system', exepath..' lemon sky tree'))
+ eq(1, call('executable', exepath))
+ end)
+
+ it('respects $PATHEXT when trying extensions on a filename', function()
+ clear({env={PATHEXT='.zzz'}})
+ for _,ext in ipairs(exts) do
+ eq(0, call('executable', 'test_executable_'..ext))
+ end
+ eq(1, call('executable', 'test_executable_zzz'))
+ end)
+
+ it('respects $PATHEXT when trying extensions on a filepath', function()
+ clear({env={PATHEXT='.zzz'}})
+ for _,ext in ipairs(exts) do
+ eq(0, call('executable', '.\\test_executable_'..ext))
+ end
+ eq(1, call('executable', '.\\test_executable_zzz'))
+ end)
+
+ it("with weird $PATHEXT", function()
+ clear({env={PATHEXT=';'}})
+ eq(0, call('executable', '.\\test_executable_zzz'))
+ clear({env={PATHEXT=';;;.zzz;;'}})
+ eq(1, call('executable', '.\\test_executable_zzz'))
+ end)
+
+ it("unqualified filename, Unix-style 'shell'", function()
+ clear({env={PATHEXT=''}})
+ command('set shell=sh')
+ for _,ext in ipairs(exts) do
+ eq(1, call('executable', 'test_executable_'..ext..'.'..ext))
+ end
+ eq(1, call('executable', 'test_executable_zzz.zzz'))
+ end)
+
+ it("relative path, Unix-style 'shell' (backslashes)", function()
+ clear({env={PATHEXT=''}})
+ command('set shell=bash.exe')
+ for _,ext in ipairs(exts) do
+ eq(1, call('executable', '.\\test_executable_'..ext..'.'..ext))
+ eq(1, call('executable', './test_executable_'..ext..'.'..ext))
+ end
+ eq(1, call('executable', '.\\test_executable_zzz.zzz'))
+ eq(1, call('executable', './test_executable_zzz.zzz'))
+ end)
+
+ it('unqualified filename, $PATHEXT contains dot', function()
+ clear({env={PATHEXT='.;.zzz'}})
+ for _,ext in ipairs(exts) do
+ eq(1, call('executable', 'test_executable_'..ext..'.'..ext))
+ end
+ eq(1, call('executable', 'test_executable_zzz.zzz'))
+ clear({env={PATHEXT='.zzz;.'}})
+ for _,ext in ipairs(exts) do
+ eq(1, call('executable', 'test_executable_'..ext..'.'..ext))
+ end
+ eq(1, call('executable', 'test_executable_zzz.zzz'))
+ end)
+
+ it('relative path, $PATHEXT contains dot (backslashes)', function()
+ clear({env={PATHEXT='.;.zzz'}})
+ for _,ext in ipairs(exts) do
+ eq(1, call('executable', '.\\test_executable_'..ext..'.'..ext))
+ eq(1, call('executable', './test_executable_'..ext..'.'..ext))
+ end
+ eq(1, call('executable', '.\\test_executable_zzz.zzz'))
+ eq(1, call('executable', './test_executable_zzz.zzz'))
+ end)
+
+ it('ignores case of extension', function()
+ clear({env={PATHEXT='.ZZZ'}})
+ eq(1, call('executable', 'test_executable_zzz.zzz'))
+ end)
+
+ it('relative path does not search $PATH', function()
+ clear({env={PATHEXT=''}})
+ eq(0, call('executable', './System32/notepad.exe'))
+ eq(0, call('executable', '.\\System32\\notepad.exe'))
+ eq(0, call('executable', '../notepad.exe'))
+ eq(0, call('executable', '..\\notepad.exe'))
+ end)
+end)