/** * @license * Copyright 2021 Balena Ltd. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ import { promises as fs, constants } from 'fs'; import * as path from 'path'; export const { F_OK, R_OK, W_OK, X_OK } = constants; export async function exists(filename: string, mode = F_OK) { try { await fs.access(filename, mode); return true; } catch { return false; } } /** * Replace sequences of untowardly characters like /[<>:"/\\|?*\u0000-\u001F]/g * and '.' or '..' with an underscore, plus other rules enforced by the filenamify * package. See https://github.com/sindresorhus/filenamify/ */ export function sanitizePath(filepath: string) { const filenamify = require('filenamify') as typeof import('filenamify'); // normalize also converts forward slash to backslash on Windows return path .normalize(filepath) .split(path.sep) .map((f) => filenamify(f, { replacement: '_' })) .join(path.sep); } /** * Given a program name like 'mount', search for it in a pre-defined set of * folders ('/usr/bin', '/bin', '/usr/sbin', '/sbin') and return the full path if found. * * For executables, in some scenarios, this can be more secure than allowing * any folder in the PATH. Only relevant on Linux or macOS. */ export async function whichBin(programName: string): Promise { for (const dir of ['/usr/bin', '/bin', '/usr/sbin', '/sbin']) { const candidate = path.join(dir, programName); if (await exists(candidate, X_OK)) { return candidate; } } return ''; } /** * Error handling wrapper around the npm `which` package: * "Like the unix which utility. Finds the first instance of a specified * executable in the PATH environment variable. Does not cache the results, * so hash -r is not needed when the PATH changes." * * @param program Basename of a program, for example 'ssh' * @param rejectOnMissing If the program cannot be found, reject the promise * with an ExpectedError instead of fulfilling it with an empty string. * @returns The program's full path, e.g. 'C:\WINDOWS\System32\OpenSSH\ssh.EXE' */ export async function which( program: string, rejectOnMissing = true, ): Promise { const whichMod = await import('which'); let programPath: string; try { programPath = await whichMod(program); } catch (err) { if (err.code === 'ENOENT') { if (rejectOnMissing) { const { ExpectedError } = await import('../errors'); throw new ExpectedError( `'${program}' program not found. Is it installed?`, ); } else { return ''; } } throw err; } return programPath; } /** * Call which(programName) and spawn() with the given arguments. * * If returnExitCodeOrSignal is true, the returned promise will resolve to * an array [code, signal] with the child process exit code number or exit * signal string respectively (as provided by the spawn close event). * * If returnExitCodeOrSignal is false, the returned promise will reject with * a custom error if the child process returns a non-zero exit code or a * non-empty signal string (as reported by the spawn close event). * * In either case and if spawn itself emits an error event or fails synchronously, * the returned promise will reject with a custom error that includes the error * message of spawn's error. */ export async function whichSpawn( programName: string, args: string[], options: import('child_process').SpawnOptions = { stdio: 'inherit' }, returnExitCodeOrSignal = false, ): Promise<[number | undefined, string | undefined]> { const { spawn } = await import('child_process'); const program = await which(programName); if (process.env.DEBUG) { console.error(`[debug] [${program}, ${args.join(', ')}]`); } let error: Error | undefined; let exitCode: number | undefined; let exitSignal: string | undefined; try { [exitCode, exitSignal] = await new Promise((resolve, reject) => { spawn(program, args, options) .on('error', reject) .on('close', (code, signal) => resolve([code, signal])); }); } catch (err) { error = err; } if (error || (!returnExitCodeOrSignal && (exitCode || exitSignal))) { const msg = [ `${programName} failed with exit code=${exitCode} signal=${exitSignal}:`, `[${program}, ${args.join(', ')}]`, ...(error ? [`${error}`] : []), ]; throw new Error(msg.join('\n')); } return [exitCode, exitSignal]; }