Skip to content

Commit

Permalink
windows fixes
Browse files Browse the repository at this point in the history
  • Loading branch information
lukekarrys committed Oct 31, 2022
1 parent 6403d8a commit 2e8f34c
Show file tree
Hide file tree
Showing 3 changed files with 52 additions and 41 deletions.
12 changes: 6 additions & 6 deletions lib/index.js
Original file line number Diff line number Diff line change
@@ -1,9 +1,8 @@
const isexe = require('isexe')
const { join, posix, win32 } = require('path')
const { join, posix, win32, delimiter: platformDelimiter } = require('path')
const isWindows = require('./is-windows.js')

const delimiter = isWindows ? win32.delimiter : posix.delimiter
const slashes = `[${isWindows ? win32.sep.replace(/(.)/g, '\\$1') : ''}${posix.sep}]`
const slashes = `[\\\\/]`
const rSlash = new RegExp(slashes)
const rRel = new RegExp(`^\\.${slashes}`)

Expand All @@ -13,18 +12,19 @@ const getNotFoundError = (cmd) =>
const getPathInfo = (cmd, {
path: optPath = process.env.PATH,
pathExt: optPathExt = process.env.PATHEXT,
delimiter: optDelimiter = platformDelimiter
}) => {
// If it has a slash, then we don't bother searching the pathenv.
// just check the file itself, and that's it.
const pathEnv = cmd.match(rSlash) ? [''] : [
// windows always checks the cwd first
...(isWindows ? [process.cwd()] : []),
...(optPath || /* istanbul ignore next: very unusual */ '').split(delimiter),
...(optPath || /* istanbul ignore next: very unusual */ '').split(optDelimiter),
]

if (isWindows) {
const pathExtExe = optPathExt || ['.EXE', '.CMD', '.BAT', '.COM'].join(delimiter)
const pathExt = pathExtExe.split(delimiter)
const pathExtExe = optPathExt || ['.EXE', '.CMD', '.BAT', '.COM'].join(optDelimiter)
const pathExt = pathExtExe.split(optDelimiter)
if (cmd.includes('.') && pathExt[0] !== '') {
pathExt.unshift('')
}
Expand Down
1 change: 1 addition & 0 deletions lib/is-windows.js
Original file line number Diff line number Diff line change
@@ -1 +1,2 @@
// only separate so it can be more easily mocked in tests
module.exports = process.platform === 'win32'
80 changes: 45 additions & 35 deletions test/index.js
Original file line number Diff line number Diff line change
@@ -1,12 +1,17 @@

const t = require('tap')
const fs = require('fs')
const { basename, join, relative, sep, posix, win32 } = require('path')
const { basename, join, relative, sep, delimiter } = require('path')
const realWindows = process.platform === 'win32'

const DELIMITER = (p) => p === 'win32' ? win32.delimiter : posix.delimiter
const envVars = { PATH: process.env.PATH, PATHEXT: process.env.PATHEXT }

const runTest = async (t, exec, expect, { platforms = ['posix', 'win32'], ...opt } = {}) => {
const runTest = async (t, exec, expect, {
methods =['sync', 'async'],
platforms = ['posix', 'win32'],
mocks,
...opt
} = {}) => {
t.teardown(() => {
for (const [k, v] of Object.entries(envVars)) {
if (v) {
Expand All @@ -19,29 +24,30 @@ const runTest = async (t, exec, expect, { platforms = ['posix', 'win32'], ...opt

for (const platform of platforms) {
t.test(`${t.name} - ${platform}`, async t => {
const platformOpt = Object.keys(opt).length ? {
...opt,
...Array.isArray(opt.path) ? { path: opt.path.join(DELIMITER(platform)) } : {},
...Array.isArray(opt.pathExt) ? { pathExt: opt.pathExt.join(DELIMITER(platform)) } : {},
} : undefined

const which = t.mock('..', { '../lib/is-windows.js': platform === 'win32' })

if (expect?.code) {
await t.rejects(() => which(exec, platformOpt), expect, 'async rejects')
t.throws(() => which.sync(exec, platformOpt), expect, 'sync throws')
return
const platformOpt = Object.keys(opt).length ? { ...opt } : undefined
const mocks = { '../lib/is-windows.js': platform === 'win32' }
if (realWindows && platform === 'posix') {
const isexe = async (p) => [].concat(expect).includes(p)
isexe.sync = (p) => [].concat(expect).includes(p)
mocks.isexe = isexe
}
const which = t.mock('..', mocks)

for (const method of methods) {
if (expect?.code) {
console.log(method, platform, expect.code, t.name)
if (method === 'async') {
await t.rejects(() => which(exec, platformOpt), expect, 'async rejects')
} else {
t.throws(() => which.sync(exec, platformOpt), expect, 'sync throws')
}
} else {
const res = method === 'async' ? await which(exec, platformOpt) : which.sync(exec, platformOpt)
t.strictSame(JSON.stringify(res).toLowerCase(), JSON.stringify(expect).toLowerCase(), method)

}

const syncRes = which.sync(exec, platformOpt)
const res = await which(exec, platformOpt)

if (typeof expect === 'string') {
t.strictSame(syncRes.toLowerCase(), expect.toLowerCase(), 'sync')
t.strictSame(res.toLowerCase(), expect.toLowerCase(), 'async')
} else {
t.strictSame(syncRes, expect, 'sync')
t.strictSame(res, expect, 'async')
}
})
}
Expand Down Expand Up @@ -77,40 +83,42 @@ t.test('find when executable', async t => {
const foo = join(fixture, 'foo.sh')
fs.chmodSync(foo, '0755')

const opts = realWindows ? { pathExt: '.sh' } : {}

t.test('absolute', async (t) => {
await runTest(t, foo, foo)
await runTest(t, foo, foo, opts)
})

t.test('with process.env.PATH', async (t) => {
process.env.PATH = fixture
await runTest(t, basename(foo), foo)
await runTest(t, basename(foo), foo, opts)
})

t.test('with path opt', async (t) => {
await runTest(t, basename(foo), foo, { path: fixture })
await runTest(t, basename(foo), foo, { ...opts, path: fixture })
})

t.test('no ./', async (t) => {
const rel = relative(process.cwd(), foo)
await runTest(t, rel, rel)
await runTest(t, rel, rel, opts)
})

t.test('with ./', async (t) => {
const rel = `.${sep}${relative(process.cwd(), foo)}`
await runTest(t, rel, rel)
await runTest(t, rel, rel, opts)
})

t.test('with ../', async (t) => {
const dir = basename(process.cwd())
const rel = join('..', dir, relative(process.cwd(), foo))
await runTest(t, rel, rel)
await runTest(t, rel, rel, opts)
})
})

t.test('find all', async t => {
const cmdName = 'x.cmd'
const fixture = t.testdir({
all: {
all: {
a: { [cmdName]: 'exec me' },
b: { [cmdName]: 'exec me' },
},
Expand All @@ -126,7 +134,7 @@ t.test('find all', async t => {
})
await runTest(t, cmdName, cmds, {
all: true,
path: dirs.map((dir, index) => index % 2 ? dir : `"${dir}"`),
path: dirs.map((dir, index) => index % 2 ? dir : `"${dir}"`).join(delimiter),
})
})

Expand All @@ -135,10 +143,8 @@ t.test('pathExt', async (t) => {
const foo = join(fixture, 'foo.sh')
fs.chmodSync(foo, '0755')

const pathExt = '.SH;.sh'
const opts = {
platforms: ['win32'],
}
const pathExt = '.SH'
const opts = { platforms: ['win32'] }

t.test('foo.sh - env vars', async (t) => {
process.env.PATHEXT = pathExt
Expand All @@ -159,4 +165,8 @@ t.test('pathExt', async (t) => {
t.test('foo - opts', async (t) => {
await runTest(t, basename(foo, '.sh'), foo, { ...opts, path: fixture, pathExt })
})

t.test('foo - no pathext', async (t) => {
await runTest(t, basename(foo, '.sh'), { code: 'ENOENT'}, { ...opts, path: fixture, pathExt: '' })
})
})

0 comments on commit 2e8f34c

Please sign in to comment.