mirror of
https://github.com/balena-io/balena-cli.git
synced 2025-01-18 02:39:49 +00:00
ac96616e4e
* Replace old resin-image-fs with newer balena-image-fs * package.json: Remove resin-image-fs package * package: Install dependencies that work with node14 * Remove resin-image-fs typings * Fix etcher-sdk related types * local/flash: Add unmountOnSuccess, write, direct properties on flash Taken from https://github.com/balena-io-modules/etcher-sdk/blob/master/examples/multi-destination.ts * tests/utils/eol-conversion: Remove ext2fs sample binary Specifically ext2fs/build/Release/bindings.node I removed it because the file doesn't exist * tests/test-data/pkg: Add new expected warnings darwin/linux/windows * os/configure: Remove windows check * local/flash: Check if environment is WSL and show warning message * Get tests to pass with certain Node v14 warning messages * INSTALL-WINDOWS: Remove os configure warning Improve push and logs support for Node.js v14 (bump 'net-keepalive') Resolves: #2200 Resolves: #1990 Change-type: minor Signed-off-by: Marios Balamatsias <mbalamatsias@gmail.com>
219 lines
6.1 KiB
TypeScript
219 lines
6.1 KiB
TypeScript
/**
|
|
* @license
|
|
* Copyright 2019-2020 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 { spawn } from 'child_process';
|
|
import * as _ from 'lodash';
|
|
import * as path from 'path';
|
|
|
|
export const ROOT = path.join(__dirname, '..');
|
|
|
|
const nodeEngineWarn = `\
|
|
------------------------------------------------------------------------------
|
|
Warning: Node version "v14.x.x" does not match required versions ">=10.20.0 <13.0.0".
|
|
This may cause unexpected behavior. To upgrade Node, visit:
|
|
https://nodejs.org/en/download/
|
|
------------------------------------------------------------------------------
|
|
`;
|
|
const nodeEngineWarnArray = nodeEngineWarn.split('\n').filter((l) => l);
|
|
|
|
export function matchesNodeEngineVersionWarn(line: string) {
|
|
line = line.replace(/"v14\.\d{1,3}\.\d{1,3}"/, '"v14.x.x"');
|
|
return (
|
|
line === nodeEngineWarn || nodeEngineWarnArray.includes(line.trimEnd())
|
|
);
|
|
}
|
|
|
|
/** Tap and buffer this process' stdout and stderr */
|
|
export class StdOutTap {
|
|
public stdoutBuf: string[] = [];
|
|
public stderrBuf: string[] = [];
|
|
public allBuf: string[] = []; // both stdout and stderr
|
|
|
|
protected origStdoutWrite: typeof process.stdout.write;
|
|
protected origStderrWrite: typeof process.stdout.write;
|
|
|
|
constructor(protected printDots = false) {}
|
|
|
|
tap() {
|
|
this.origStdoutWrite = process.stdout.write;
|
|
this.origStderrWrite = process.stderr.write;
|
|
|
|
process.stdout.write = (chunk: string, ...args: any[]): boolean => {
|
|
this.stdoutBuf.push(chunk);
|
|
this.allBuf.push(chunk);
|
|
const str = this.printDots ? '.' : chunk;
|
|
return this.origStdoutWrite.call(process.stdout, str, ...args);
|
|
};
|
|
|
|
process.stderr.write = (chunk: string, ...args: any[]): boolean => {
|
|
this.stderrBuf.push(chunk);
|
|
this.allBuf.push(chunk);
|
|
const str = this.printDots ? '.' : chunk;
|
|
return this.origStderrWrite.call(process.stderr, str, ...args);
|
|
};
|
|
}
|
|
|
|
untap() {
|
|
process.stdout.write = this.origStdoutWrite;
|
|
process.stderr.write = this.origStderrWrite;
|
|
if (this.printDots) {
|
|
console.error('');
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Diff strings by line, using the 'diff' npm package:
|
|
* https://www.npmjs.com/package/diff
|
|
*/
|
|
export function diffLines(str1: string, str2: string): string {
|
|
const { diffTrimmedLines } = require('diff');
|
|
const diffObjs = diffTrimmedLines(str1, str2);
|
|
const prefix = (chunk: string, char: string) =>
|
|
chunk
|
|
.split('\n')
|
|
.map((line: string) => `${char} ${line}`)
|
|
.join('\n');
|
|
const diffStr = diffObjs
|
|
.map((part: any) => {
|
|
return part.added
|
|
? prefix(part.value, '+')
|
|
: part.removed
|
|
? prefix(part.value, '-')
|
|
: prefix(part.value, ' ');
|
|
})
|
|
.join('\n');
|
|
return diffStr;
|
|
}
|
|
|
|
export function loadPackageJson() {
|
|
return require(path.join(ROOT, 'package.json'));
|
|
}
|
|
|
|
/**
|
|
* Run the executable at execPath as a child process, and resolve a promise
|
|
* to the executable's stdout output as a string. Reject the promise if
|
|
* anything is printed to stderr, or if the child process exits with a
|
|
* non-zero exit code.
|
|
* @param execPath Executable path
|
|
* @param args Command-line argument for the executable
|
|
*/
|
|
export async function getSubprocessStdout(
|
|
execPath: string,
|
|
args: string[],
|
|
): Promise<string> {
|
|
const child = spawn(execPath, args);
|
|
return new Promise((resolve, reject) => {
|
|
let stdout = '';
|
|
child.stdout.on('error', reject);
|
|
child.stderr.on('error', reject);
|
|
child.stdout.on('data', (data: Buffer) => {
|
|
try {
|
|
stdout = data.toString();
|
|
} catch (err) {
|
|
reject(err);
|
|
}
|
|
});
|
|
child.stderr.on('data', (data: Buffer) => {
|
|
try {
|
|
const stderr = data.toString();
|
|
|
|
// ignore any debug lines, but ensure that we parse
|
|
// every line provided to the stderr stream
|
|
const lines = _.filter(
|
|
stderr.trim().split(/\r?\n/),
|
|
(line) =>
|
|
!line.startsWith('[debug]') && !matchesNodeEngineVersionWarn(line),
|
|
);
|
|
if (lines.length > 0) {
|
|
reject(
|
|
new Error(`"${execPath}": non-empty stderr "${lines.join('\n')}"`),
|
|
);
|
|
}
|
|
} catch (err) {
|
|
reject(err);
|
|
}
|
|
});
|
|
child.on('exit', (code: number) => {
|
|
if (code) {
|
|
reject(new Error(`"${execPath}": non-zero exit code "${code}"`));
|
|
} else {
|
|
resolve(stdout);
|
|
}
|
|
});
|
|
});
|
|
}
|
|
|
|
/**
|
|
* 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'
|
|
* @returns The program's full path, e.g. 'C:\WINDOWS\System32\OpenSSH\ssh.EXE'
|
|
*/
|
|
export async function which(program: string): Promise<string> {
|
|
const whichMod = await import('which');
|
|
let programPath: string;
|
|
try {
|
|
programPath = await whichMod(program);
|
|
} catch (err) {
|
|
if (err.code === 'ENOENT') {
|
|
throw new Error(`'${program}' program not found. Is it installed?`);
|
|
}
|
|
throw err;
|
|
}
|
|
return programPath;
|
|
}
|
|
|
|
/**
|
|
* Call which(programName) and spawn() with the given arguments. Throw an error
|
|
* if the process exit code is not zero.
|
|
*/
|
|
export async function whichSpawn(
|
|
programName: string,
|
|
args?: string[],
|
|
): Promise<void> {
|
|
const program = await which(programName);
|
|
let error: Error | undefined;
|
|
let exitCode: number | undefined;
|
|
try {
|
|
exitCode = await new Promise<number>((resolve, reject) => {
|
|
try {
|
|
spawn(program, args, { stdio: 'inherit' })
|
|
.on('error', reject)
|
|
.on('close', resolve);
|
|
} catch (err) {
|
|
reject(err);
|
|
}
|
|
});
|
|
} catch (err) {
|
|
error = err;
|
|
}
|
|
if (error || exitCode) {
|
|
const msg = [
|
|
`${programName} failed with exit code ${exitCode}:`,
|
|
`"${program}" [${args}]`,
|
|
];
|
|
if (error) {
|
|
msg.push(`${error}`);
|
|
}
|
|
throw new Error(msg.join('\n'));
|
|
}
|
|
}
|