2024-05-13 11:48:26 -06:00
|
|
|
import { expect } from 'chai';
|
|
|
|
import * as cp from 'node:child_process';
|
|
|
|
import * as nodeCrypto from 'node:crypto';
|
|
|
|
import * as originalFs from 'node:original-fs';
|
2024-06-19 08:10:16 -06:00
|
|
|
import * as fs from 'node:fs';
|
2024-05-13 11:48:26 -06:00
|
|
|
import * as os from 'node:os';
|
|
|
|
import * as path from 'node:path';
|
|
|
|
import { ifdescribe } from './lib/spec-helpers';
|
|
|
|
|
|
|
|
import { getRawHeader } from '@electron/asar';
|
|
|
|
import { resedit } from '@electron/packager/dist/resedit';
|
|
|
|
import { flipFuses, FuseV1Config, FuseV1Options, FuseVersion } from '@electron/fuses';
|
|
|
|
import { copyApp } from './lib/fs-helpers';
|
|
|
|
|
|
|
|
const bufferReplace = (haystack: Buffer, needle: string, replacement: string, throwOnMissing = true): Buffer => {
|
|
|
|
const needleBuffer = Buffer.from(needle);
|
|
|
|
const idx = haystack.indexOf(needleBuffer);
|
|
|
|
if (idx === -1) {
|
|
|
|
if (throwOnMissing) throw new Error(`Needle ${needle} not found in haystack`);
|
|
|
|
return haystack;
|
|
|
|
}
|
|
|
|
|
|
|
|
const before = haystack.slice(0, idx);
|
|
|
|
const after = bufferReplace(haystack.slice(idx + needleBuffer.length), needle, replacement, false);
|
|
|
|
const len = idx + replacement.length + after.length;
|
|
|
|
return Buffer.concat([before, Buffer.from(replacement), after], len);
|
|
|
|
};
|
|
|
|
|
|
|
|
type SpawnResult = { code: number | null, out: string, signal: NodeJS.Signals | null };
|
|
|
|
function spawn (cmd: string, args: string[], opts: any = {}) {
|
|
|
|
let out = '';
|
|
|
|
const child = cp.spawn(cmd, args, opts);
|
|
|
|
child.stdout.on('data', (chunk: Buffer) => {
|
|
|
|
out += chunk.toString();
|
|
|
|
});
|
|
|
|
child.stderr.on('data', (chunk: Buffer) => {
|
|
|
|
out += chunk.toString();
|
|
|
|
});
|
|
|
|
return new Promise<SpawnResult>((resolve) => {
|
|
|
|
child.on('exit', (code, signal) => {
|
|
|
|
resolve({
|
|
|
|
code,
|
|
|
|
signal,
|
|
|
|
out
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
const expectToHaveCrashed = (res: SpawnResult) => {
|
|
|
|
if (process.platform === 'win32') {
|
|
|
|
expect(res.code).to.not.equal(0);
|
|
|
|
expect(res.code).to.not.equal(null);
|
|
|
|
expect(res.signal).to.equal(null);
|
|
|
|
} else {
|
|
|
|
expect(res.code).to.equal(null);
|
|
|
|
expect(res.signal).to.equal('SIGTRAP');
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
describe('fuses', function () {
|
|
|
|
this.timeout(120000);
|
|
|
|
|
|
|
|
let tmpDir: string;
|
|
|
|
let appPath: string;
|
|
|
|
|
|
|
|
const launchApp = (args: string[] = []) => {
|
|
|
|
if (process.platform === 'darwin') {
|
|
|
|
return spawn(path.resolve(appPath, 'Contents/MacOS/Electron'), args);
|
|
|
|
}
|
|
|
|
return spawn(appPath, args);
|
|
|
|
};
|
|
|
|
|
|
|
|
const ensureFusesBeforeEach = (fuses: Omit<FuseV1Config<boolean>, 'version' | 'strictlyRequireAllFuses' | 'resetAdhocDarwinSignature'>) => {
|
|
|
|
beforeEach(async () => {
|
|
|
|
await flipFuses(appPath, {
|
|
|
|
version: FuseVersion.V1,
|
|
|
|
resetAdHocDarwinSignature: true,
|
|
|
|
...fuses
|
|
|
|
});
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
beforeEach(async () => {
|
2024-06-19 08:10:16 -06:00
|
|
|
tmpDir = await fs.promises.mkdtemp(path.resolve(os.tmpdir(), 'electron-asar-integrity-spec-'));
|
2024-05-13 11:48:26 -06:00
|
|
|
appPath = await copyApp(tmpDir);
|
|
|
|
});
|
|
|
|
|
|
|
|
afterEach(async () => {
|
|
|
|
for (let attempt = 0; attempt <= 3; attempt++) {
|
|
|
|
// Somtimes windows holds on to a DLL during the crash for a little bit, so we try a few times to delete it
|
|
|
|
if (attempt > 0) await new Promise((resolve) => setTimeout(resolve, 500 * attempt));
|
|
|
|
try {
|
|
|
|
await originalFs.promises.rm(tmpDir, { recursive: true });
|
|
|
|
break;
|
|
|
|
} catch {}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
ifdescribe((process.platform === 'win32' && process.arch !== 'arm64') || process.platform === 'darwin')('ASAR Integrity', () => {
|
|
|
|
let pathToAsar: string;
|
|
|
|
|
|
|
|
beforeEach(async () => {
|
|
|
|
if (process.platform === 'darwin') {
|
|
|
|
pathToAsar = path.resolve(appPath, 'Contents', 'Resources', 'default_app.asar');
|
|
|
|
} else {
|
|
|
|
pathToAsar = path.resolve(path.dirname(appPath), 'resources', 'default_app.asar');
|
|
|
|
}
|
|
|
|
|
|
|
|
if (process.platform === 'win32') {
|
|
|
|
await resedit(appPath, {
|
|
|
|
asarIntegrity: {
|
|
|
|
'resources\\default_app.asar': {
|
|
|
|
algorithm: 'SHA256',
|
|
|
|
hash: nodeCrypto.createHash('sha256').update(getRawHeader(pathToAsar).headerString).digest('hex')
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('when enabled', () => {
|
|
|
|
ensureFusesBeforeEach({
|
|
|
|
[FuseV1Options.EnableEmbeddedAsarIntegrityValidation]: true
|
|
|
|
});
|
|
|
|
|
|
|
|
it('opens normally when unmodified', async () => {
|
|
|
|
const res = await launchApp([path.resolve(__dirname, 'fixtures/apps/hello/hello.js')]);
|
|
|
|
expect(res.code).to.equal(0);
|
|
|
|
expect(res.signal).to.equal(null);
|
|
|
|
expect(res.out).to.include('alive');
|
|
|
|
});
|
|
|
|
|
|
|
|
it('fatals if the integrity header does not match', async () => {
|
|
|
|
const asar = await originalFs.promises.readFile(pathToAsar);
|
|
|
|
// Ensure that the header stil starts with the same thing, if build system
|
|
|
|
// things result in the header changing we should update this test
|
|
|
|
expect(asar.toString()).to.contain('{"files":{"default_app.js"');
|
|
|
|
await originalFs.promises.writeFile(pathToAsar, bufferReplace(asar, '{"files":{"default_app.js"', '{"files":{"default_oop.js"'));
|
|
|
|
|
|
|
|
const res = await launchApp(['--version']);
|
|
|
|
expectToHaveCrashed(res);
|
|
|
|
expect(res.out).to.include('Integrity check failed for asar archive');
|
|
|
|
});
|
|
|
|
|
|
|
|
it('fatals if a loaded main process JS file does not match', async () => {
|
|
|
|
const asar = await originalFs.promises.readFile(pathToAsar);
|
|
|
|
// Ensure that the header stil starts with the same thing, if build system
|
|
|
|
// things result in the header changing we should update this test
|
|
|
|
expect(asar.toString()).to.contain('Invalid Usage');
|
|
|
|
await originalFs.promises.writeFile(pathToAsar, bufferReplace(asar, 'Invalid Usage', 'VVValid Usage'));
|
|
|
|
|
|
|
|
const res = await launchApp(['--version']);
|
|
|
|
expect(res.code).to.equal(1);
|
|
|
|
expect(res.signal).to.equal(null);
|
|
|
|
expect(res.out).to.include('ASAR Integrity Violation: got a hash mismatch');
|
|
|
|
});
|
|
|
|
|
|
|
|
it('fatals if a renderer content file does not match', async () => {
|
|
|
|
const asar = await originalFs.promises.readFile(pathToAsar);
|
|
|
|
// Ensure that the header stil starts with the same thing, if build system
|
|
|
|
// things result in the header changing we should update this test
|
|
|
|
expect(asar.toString()).to.contain('require-trusted-types-for');
|
|
|
|
await originalFs.promises.writeFile(pathToAsar, bufferReplace(asar, 'require-trusted-types-for', 'require-trusted-types-not'));
|
|
|
|
|
|
|
|
const res = await launchApp();
|
|
|
|
expectToHaveCrashed(res);
|
|
|
|
expect(res.out).to.include('Failed to validate block while ending ASAR file stream');
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('when disabled', () => {
|
|
|
|
ensureFusesBeforeEach({
|
|
|
|
[FuseV1Options.EnableEmbeddedAsarIntegrityValidation]: false
|
|
|
|
});
|
|
|
|
|
|
|
|
it('does nothing if the integrity header does not match', async () => {
|
|
|
|
const asar = await originalFs.promises.readFile(pathToAsar);
|
|
|
|
// Ensure that the header stil starts with the same thing, if build system
|
|
|
|
// things result in the header changing we should update this test
|
|
|
|
expect(asar.toString()).to.contain('{"files":{"default_app.js"');
|
|
|
|
await originalFs.promises.writeFile(pathToAsar, bufferReplace(asar, '{"files":{"default_app.js"', '{"files":{"default_oop.js"'));
|
|
|
|
|
|
|
|
const res = await launchApp(['--version']);
|
|
|
|
expect(res.code).to.equal(0);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|