diff --git a/spec-main/asar-spec.ts b/spec-main/asar-spec.ts index babfcdb8c1a2..3b8aacc6a4a1 100644 --- a/spec-main/asar-spec.ts +++ b/spec-main/asar-spec.ts @@ -5,6 +5,10 @@ import { Worker } from 'worker_threads'; import { BrowserWindow, ipcMain } from 'electron/main'; import { closeAllWindows } from './window-helpers'; import { emittedOnce } from './events-helpers'; +import { getRemoteContext, ifdescribe, itremote, useRemoteContext } from './spec-helpers'; +import * as importedFs from 'fs'; + +const features = process._linkedBinding('electron_common_features'); describe('asar package', () => { const fixtures = path.join(__dirname, '..', 'spec', 'fixtures'); @@ -125,3 +129,1434 @@ describe('asar package', () => { }); }); }); + +// eslint-disable-next-line @typescript-eslint/no-unused-vars +async function expectToThrowErrorWithCode (_func: Function, _code: string) { + /* dummy for typescript */ +} + +// eslint-disable-next-line @typescript-eslint/no-unused-vars +function promisify (_f: Function): any { + /* dummy for typescript */ +} + +describe('asar package', function () { + const fixtures = path.join(__dirname, '..', 'spec', 'fixtures'); + const asarDir = path.join(fixtures, 'test.asar'); + const fs = require('fs') as typeof importedFs; // dummy, to fool typescript + + useRemoteContext({ + url: url.pathToFileURL(path.join(fixtures, 'pages', 'blank.html')), + setup: ` + async function expectToThrowErrorWithCode (func, code) { + let error; + try { + await func(); + } catch (e) { + error = e; + } + + const chai = require('chai') + chai.expect(error).to.have.property('code').which.equals(code); + } + + fs = require('fs') + path = require('path') + asarDir = ${JSON.stringify(asarDir)} + + // This is used instead of util.promisify for some tests to dodge the + // util.promisify.custom behavior. + promisify = (f) => { + return (...args) => new Promise((resolve, reject) => { + f(...args, (err, result) => { + if (err) reject(err) + else resolve(result) + }) + }) + } + + null + ` + }); + + describe('node api', function () { + itremote('supports paths specified as a Buffer', function () { + const file = Buffer.from(path.join(asarDir, 'a.asar', 'file1')); + expect(fs.existsSync(file)).to.be.true(); + }); + + describe('fs.readFileSync', function () { + itremote('does not leak fd', function () { + let readCalls = 1; + while (readCalls <= 10000) { + fs.readFileSync(path.join(process.resourcesPath, 'default_app.asar', 'main.js')); + readCalls++; + } + }); + + itremote('reads a normal file', function () { + const file1 = path.join(asarDir, 'a.asar', 'file1'); + expect(fs.readFileSync(file1).toString().trim()).to.equal('file1'); + const file2 = path.join(asarDir, 'a.asar', 'file2'); + expect(fs.readFileSync(file2).toString().trim()).to.equal('file2'); + const file3 = path.join(asarDir, 'a.asar', 'file3'); + expect(fs.readFileSync(file3).toString().trim()).to.equal('file3'); + }); + + itremote('reads from a empty file', function () { + const file = path.join(asarDir, 'empty.asar', 'file1'); + const buffer = fs.readFileSync(file); + expect(buffer).to.be.empty(); + expect(buffer.toString()).to.equal(''); + }); + + itremote('reads a linked file', function () { + const p = path.join(asarDir, 'a.asar', 'link1'); + expect(fs.readFileSync(p).toString().trim()).to.equal('file1'); + }); + + itremote('reads a file from linked directory', function () { + const p1 = path.join(asarDir, 'a.asar', 'link2', 'file1'); + expect(fs.readFileSync(p1).toString().trim()).to.equal('file1'); + const p2 = path.join(asarDir, 'a.asar', 'link2', 'link2', 'file1'); + expect(fs.readFileSync(p2).toString().trim()).to.equal('file1'); + }); + + itremote('throws ENOENT error when can not find file', function () { + const p = path.join(asarDir, 'a.asar', 'not-exist'); + expect(() => { + fs.readFileSync(p); + }).to.throw(/ENOENT/); + }); + + itremote('passes ENOENT error to callback when can not find file', function () { + const p = path.join(asarDir, 'a.asar', 'not-exist'); + let async = false; + fs.readFile(p, function (error) { + expect(async).to.be.true(); + expect(error).to.match(/ENOENT/); + }); + async = true; + }); + + itremote('reads a normal file with unpacked files', function () { + const p = path.join(asarDir, 'unpack.asar', 'a.txt'); + expect(fs.readFileSync(p).toString().trim()).to.equal('a'); + }); + + itremote('reads a file in filesystem', function () { + const p = path.resolve(asarDir, 'file'); + expect(fs.readFileSync(p).toString().trim()).to.equal('file'); + }); + }); + + describe('fs.readFile', function () { + itremote('reads a normal file', async function () { + const p = path.join(asarDir, 'a.asar', 'file1'); + const content = await new Promise((resolve, reject) => fs.readFile(p, (err, content) => { + if (err) return reject(err); + resolve(content); + })); + expect(String(content).trim()).to.equal('file1'); + }); + + itremote('reads from a empty file', async function () { + const p = path.join(asarDir, 'empty.asar', 'file1'); + const content = await new Promise((resolve, reject) => fs.readFile(p, (err, content) => { + if (err) return reject(err); + resolve(content); + })); + expect(String(content)).to.equal(''); + }); + + itremote('reads from a empty file with encoding', async function () { + const p = path.join(asarDir, 'empty.asar', 'file1'); + const content = await new Promise((resolve, reject) => fs.readFile(p, (err, content) => { + if (err) return reject(err); + resolve(content); + })); + expect(String(content)).to.equal(''); + }); + + itremote('reads a linked file', async function () { + const p = path.join(asarDir, 'a.asar', 'link1'); + const content = await new Promise((resolve, reject) => fs.readFile(p, (err, content) => { + if (err) return reject(err); + resolve(content); + })); + expect(String(content).trim()).to.equal('file1'); + }); + + itremote('reads a file from linked directory', async function () { + const p = path.join(asarDir, 'a.asar', 'link2', 'link2', 'file1'); + const content = await new Promise((resolve, reject) => fs.readFile(p, (err, content) => { + if (err) return reject(err); + resolve(content); + })); + expect(String(content).trim()).to.equal('file1'); + }); + + itremote('throws ENOENT error when can not find file', async function () { + const p = path.join(asarDir, 'a.asar', 'not-exist'); + const err = await new Promise((resolve) => fs.readFile(p, resolve)); + expect(err.code).to.equal('ENOENT'); + }); + }); + + describe('fs.promises.readFile', function () { + itremote('reads a normal file', async function () { + const p = path.join(asarDir, 'a.asar', 'file1'); + const content = await fs.promises.readFile(p); + expect(String(content).trim()).to.equal('file1'); + }); + + itremote('reads from a empty file', async function () { + const p = path.join(asarDir, 'empty.asar', 'file1'); + const content = await fs.promises.readFile(p); + expect(String(content)).to.equal(''); + }); + + itremote('reads from a empty file with encoding', async function () { + const p = path.join(asarDir, 'empty.asar', 'file1'); + const content = await fs.promises.readFile(p, 'utf8'); + expect(content).to.equal(''); + }); + + itremote('reads a linked file', async function () { + const p = path.join(asarDir, 'a.asar', 'link1'); + const content = await fs.promises.readFile(p); + expect(String(content).trim()).to.equal('file1'); + }); + + itremote('reads a file from linked directory', async function () { + const p = path.join(asarDir, 'a.asar', 'link2', 'link2', 'file1'); + const content = await fs.promises.readFile(p); + expect(String(content).trim()).to.equal('file1'); + }); + + itremote('throws ENOENT error when can not find file', async function () { + const p = path.join(asarDir, 'a.asar', 'not-exist'); + await expectToThrowErrorWithCode(() => fs.promises.readFile(p), 'ENOENT'); + }); + }); + + describe('fs.copyFile', function () { + itremote('copies a normal file', async function () { + const p = path.join(asarDir, 'a.asar', 'file1'); + const temp = require('temp').track(); + const dest = temp.path(); + await new Promise((resolve, reject) => { + fs.copyFile(p, dest, (err) => { + if (err) reject(err); + else resolve(); + }); + }); + expect(fs.readFileSync(p).equals(fs.readFileSync(dest))).to.be.true(); + }); + + itremote('copies a unpacked file', async function () { + const p = path.join(asarDir, 'unpack.asar', 'a.txt'); + const temp = require('temp').track(); + const dest = temp.path(); + await new Promise((resolve, reject) => { + fs.copyFile(p, dest, (err) => { + if (err) reject(err); + else resolve(); + }); + }); + expect(fs.readFileSync(p).equals(fs.readFileSync(dest))).to.be.true(); + }); + }); + + describe('fs.promises.copyFile', function () { + itremote('copies a normal file', async function () { + const p = path.join(asarDir, 'a.asar', 'file1'); + const temp = require('temp').track(); + const dest = temp.path(); + await fs.promises.copyFile(p, dest); + expect(fs.readFileSync(p).equals(fs.readFileSync(dest))).to.be.true(); + }); + + itremote('copies a unpacked file', async function () { + const p = path.join(asarDir, 'unpack.asar', 'a.txt'); + const temp = require('temp').track(); + const dest = temp.path(); + await fs.promises.copyFile(p, dest); + expect(fs.readFileSync(p).equals(fs.readFileSync(dest))).to.be.true(); + }); + }); + + describe('fs.copyFileSync', function () { + itremote('copies a normal file', function () { + const p = path.join(asarDir, 'a.asar', 'file1'); + const temp = require('temp').track(); + const dest = temp.path(); + fs.copyFileSync(p, dest); + expect(fs.readFileSync(p).equals(fs.readFileSync(dest))).to.be.true(); + }); + + itremote('copies a unpacked file', function () { + const p = path.join(asarDir, 'unpack.asar', 'a.txt'); + const temp = require('temp').track(); + const dest = temp.path(); + fs.copyFileSync(p, dest); + expect(fs.readFileSync(p).equals(fs.readFileSync(dest))).to.be.true(); + }); + }); + + describe('fs.lstatSync', function () { + itremote('handles path with trailing slash correctly', function () { + const p = path.join(asarDir, 'a.asar', 'link2', 'link2', 'file1'); + fs.lstatSync(p); + fs.lstatSync(p + '/'); + }); + + itremote('returns information of root', function () { + const p = path.join(asarDir, 'a.asar'); + const stats = fs.lstatSync(p); + expect(stats.isFile()).to.be.false(); + expect(stats.isDirectory()).to.be.true(); + expect(stats.isSymbolicLink()).to.be.false(); + expect(stats.size).to.equal(0); + }); + + itremote('returns information of root with stats as bigint', function () { + const p = path.join(asarDir, 'a.asar'); + const stats = fs.lstatSync(p, { bigint: false }); + expect(stats.isFile()).to.be.false(); + expect(stats.isDirectory()).to.be.true(); + expect(stats.isSymbolicLink()).to.be.false(); + expect(stats.size).to.equal(0); + }); + + itremote('returns information of a normal file', function () { + const ref2 = ['file1', 'file2', 'file3', path.join('dir1', 'file1'), path.join('link2', 'file1')]; + for (let j = 0, len = ref2.length; j < len; j++) { + const file = ref2[j]; + const p = path.join(asarDir, 'a.asar', file); + const stats = fs.lstatSync(p); + expect(stats.isFile()).to.be.true(); + expect(stats.isDirectory()).to.be.false(); + expect(stats.isSymbolicLink()).to.be.false(); + expect(stats.size).to.equal(6); + } + }); + + itremote('returns information of a normal directory', function () { + const ref2 = ['dir1', 'dir2', 'dir3']; + for (let j = 0, len = ref2.length; j < len; j++) { + const file = ref2[j]; + const p = path.join(asarDir, 'a.asar', file); + const stats = fs.lstatSync(p); + expect(stats.isFile()).to.be.false(); + expect(stats.isDirectory()).to.be.true(); + expect(stats.isSymbolicLink()).to.be.false(); + expect(stats.size).to.equal(0); + } + }); + + itremote('returns information of a linked file', function () { + const ref2 = ['link1', path.join('dir1', 'link1'), path.join('link2', 'link2')]; + for (let j = 0, len = ref2.length; j < len; j++) { + const file = ref2[j]; + const p = path.join(asarDir, 'a.asar', file); + const stats = fs.lstatSync(p); + expect(stats.isFile()).to.be.false(); + expect(stats.isDirectory()).to.be.false(); + expect(stats.isSymbolicLink()).to.be.true(); + expect(stats.size).to.equal(0); + } + }); + + itremote('returns information of a linked directory', function () { + const ref2 = ['link2', path.join('dir1', 'link2'), path.join('link2', 'link2')]; + for (let j = 0, len = ref2.length; j < len; j++) { + const file = ref2[j]; + const p = path.join(asarDir, 'a.asar', file); + const stats = fs.lstatSync(p); + expect(stats.isFile()).to.be.false(); + expect(stats.isDirectory()).to.be.false(); + expect(stats.isSymbolicLink()).to.be.true(); + expect(stats.size).to.equal(0); + } + }); + + itremote('throws ENOENT error when can not find file', function () { + const ref2 = ['file4', 'file5', path.join('dir1', 'file4')]; + for (let j = 0, len = ref2.length; j < len; j++) { + const file = ref2[j]; + const p = path.join(asarDir, 'a.asar', file); + expect(() => { + fs.lstatSync(p); + }).to.throw(/ENOENT/); + } + }); + }); + + describe('fs.lstat', function () { + itremote('handles path with trailing slash correctly', async function () { + const p = path.join(asarDir, 'a.asar', 'link2', 'link2', 'file1'); + await promisify(fs.lstat)(p + '/'); + }); + + itremote('returns information of root', async function () { + const p = path.join(asarDir, 'a.asar'); + const stats = await promisify(fs.lstat)(p); + expect(stats.isFile()).to.be.false(); + expect(stats.isDirectory()).to.be.true(); + expect(stats.isSymbolicLink()).to.be.false(); + expect(stats.size).to.equal(0); + }); + + itremote('returns information of root with stats as bigint', async function () { + const p = path.join(asarDir, 'a.asar'); + const stats = await promisify(fs.lstat)(p, { bigint: false }); + expect(stats.isFile()).to.be.false(); + expect(stats.isDirectory()).to.be.true(); + expect(stats.isSymbolicLink()).to.be.false(); + expect(stats.size).to.equal(0); + }); + + itremote('returns information of a normal file', async function () { + const p = path.join(asarDir, 'a.asar', 'link2', 'file1'); + const stats = await promisify(fs.lstat)(p); + expect(stats.isFile()).to.be.true(); + expect(stats.isDirectory()).to.be.false(); + expect(stats.isSymbolicLink()).to.be.false(); + expect(stats.size).to.equal(6); + }); + + itremote('returns information of a normal directory', async function () { + const p = path.join(asarDir, 'a.asar', 'dir1'); + const stats = await promisify(fs.lstat)(p); + expect(stats.isFile()).to.be.false(); + expect(stats.isDirectory()).to.be.true(); + expect(stats.isSymbolicLink()).to.be.false(); + expect(stats.size).to.equal(0); + }); + + itremote('returns information of a linked file', async function () { + const p = path.join(asarDir, 'a.asar', 'link2', 'link1'); + const stats = await promisify(fs.lstat)(p); + expect(stats.isFile()).to.be.false(); + expect(stats.isDirectory()).to.be.false(); + expect(stats.isSymbolicLink()).to.be.true(); + expect(stats.size).to.equal(0); + }); + + itremote('returns information of a linked directory', async function () { + const p = path.join(asarDir, 'a.asar', 'link2', 'link2'); + const stats = await promisify(fs.lstat)(p); + expect(stats.isFile()).to.be.false(); + expect(stats.isDirectory()).to.be.false(); + expect(stats.isSymbolicLink()).to.be.true(); + expect(stats.size).to.equal(0); + }); + + itremote('throws ENOENT error when can not find file', async function () { + const p = path.join(asarDir, 'a.asar', 'file4'); + const err = await new Promise(resolve => fs.lstat(p, resolve)); + expect(err.code).to.equal('ENOENT'); + }); + }); + + describe('fs.promises.lstat', function () { + itremote('handles path with trailing slash correctly', async function () { + const p = path.join(asarDir, 'a.asar', 'link2', 'link2', 'file1'); + await fs.promises.lstat(p + '/'); + }); + + itremote('returns information of root', async function () { + const p = path.join(asarDir, 'a.asar'); + const stats = await fs.promises.lstat(p); + expect(stats.isFile()).to.be.false(); + expect(stats.isDirectory()).to.be.true(); + expect(stats.isSymbolicLink()).to.be.false(); + expect(stats.size).to.equal(0); + }); + + itremote('returns information of root with stats as bigint', async function () { + const p = path.join(asarDir, 'a.asar'); + const stats = await fs.promises.lstat(p, { bigint: false }); + expect(stats.isFile()).to.be.false(); + expect(stats.isDirectory()).to.be.true(); + expect(stats.isSymbolicLink()).to.be.false(); + expect(stats.size).to.equal(0); + }); + + itremote('returns information of a normal file', async function () { + const p = path.join(asarDir, 'a.asar', 'link2', 'file1'); + const stats = await fs.promises.lstat(p); + expect(stats.isFile()).to.be.true(); + expect(stats.isDirectory()).to.be.false(); + expect(stats.isSymbolicLink()).to.be.false(); + expect(stats.size).to.equal(6); + }); + + itremote('returns information of a normal directory', async function () { + const p = path.join(asarDir, 'a.asar', 'dir1'); + const stats = await fs.promises.lstat(p); + expect(stats.isFile()).to.be.false(); + expect(stats.isDirectory()).to.be.true(); + expect(stats.isSymbolicLink()).to.be.false(); + expect(stats.size).to.equal(0); + }); + + itremote('returns information of a linked file', async function () { + const p = path.join(asarDir, 'a.asar', 'link2', 'link1'); + const stats = await fs.promises.lstat(p); + expect(stats.isFile()).to.be.false(); + expect(stats.isDirectory()).to.be.false(); + expect(stats.isSymbolicLink()).to.be.true(); + expect(stats.size).to.equal(0); + }); + + itremote('returns information of a linked directory', async function () { + const p = path.join(asarDir, 'a.asar', 'link2', 'link2'); + const stats = await fs.promises.lstat(p); + expect(stats.isFile()).to.be.false(); + expect(stats.isDirectory()).to.be.false(); + expect(stats.isSymbolicLink()).to.be.true(); + expect(stats.size).to.equal(0); + }); + + itremote('throws ENOENT error when can not find file', async function () { + const p = path.join(asarDir, 'a.asar', 'file4'); + await expectToThrowErrorWithCode(() => fs.promises.lstat(p), 'ENOENT'); + }); + }); + + describe('fs.realpathSync', () => { + itremote('returns real path root', () => { + const parent = fs.realpathSync(asarDir); + const p = 'a.asar'; + const r = fs.realpathSync(path.join(parent, p)); + expect(r).to.equal(path.join(parent, p)); + }); + + itremote('returns real path of a normal file', () => { + const parent = fs.realpathSync(asarDir); + const p = path.join('a.asar', 'file1'); + const r = fs.realpathSync(path.join(parent, p)); + expect(r).to.equal(path.join(parent, p)); + }); + + itremote('returns real path of a normal directory', () => { + const parent = fs.realpathSync(asarDir); + const p = path.join('a.asar', 'dir1'); + const r = fs.realpathSync(path.join(parent, p)); + expect(r).to.equal(path.join(parent, p)); + }); + + itremote('returns real path of a linked file', () => { + const parent = fs.realpathSync(asarDir); + const p = path.join('a.asar', 'link2', 'link1'); + const r = fs.realpathSync(path.join(parent, p)); + expect(r).to.equal(path.join(parent, 'a.asar', 'file1')); + }); + + itremote('returns real path of a linked directory', () => { + const parent = fs.realpathSync(asarDir); + const p = path.join('a.asar', 'link2', 'link2'); + const r = fs.realpathSync(path.join(parent, p)); + expect(r).to.equal(path.join(parent, 'a.asar', 'dir1')); + }); + + itremote('returns real path of an unpacked file', () => { + const parent = fs.realpathSync(asarDir); + const p = path.join('unpack.asar', 'a.txt'); + const r = fs.realpathSync(path.join(parent, p)); + expect(r).to.equal(path.join(parent, p)); + }); + + itremote('throws ENOENT error when can not find file', () => { + const parent = fs.realpathSync(asarDir); + const p = path.join('a.asar', 'not-exist'); + expect(() => { + fs.realpathSync(path.join(parent, p)); + }).to.throw(/ENOENT/); + }); + }); + + describe('fs.realpathSync.native', () => { + itremote('returns real path root', () => { + const parent = fs.realpathSync.native(asarDir); + const p = 'a.asar'; + const r = fs.realpathSync.native(path.join(parent, p)); + expect(r).to.equal(path.join(parent, p)); + }); + + itremote('returns real path of a normal file', () => { + const parent = fs.realpathSync.native(asarDir); + const p = path.join('a.asar', 'file1'); + const r = fs.realpathSync.native(path.join(parent, p)); + expect(r).to.equal(path.join(parent, p)); + }); + + itremote('returns real path of a normal directory', () => { + const parent = fs.realpathSync.native(asarDir); + const p = path.join('a.asar', 'dir1'); + const r = fs.realpathSync.native(path.join(parent, p)); + expect(r).to.equal(path.join(parent, p)); + }); + + itremote('returns real path of a linked file', () => { + const parent = fs.realpathSync.native(asarDir); + const p = path.join('a.asar', 'link2', 'link1'); + const r = fs.realpathSync.native(path.join(parent, p)); + expect(r).to.equal(path.join(parent, 'a.asar', 'file1')); + }); + + itremote('returns real path of a linked directory', () => { + const parent = fs.realpathSync.native(asarDir); + const p = path.join('a.asar', 'link2', 'link2'); + const r = fs.realpathSync.native(path.join(parent, p)); + expect(r).to.equal(path.join(parent, 'a.asar', 'dir1')); + }); + + itremote('returns real path of an unpacked file', () => { + const parent = fs.realpathSync.native(asarDir); + const p = path.join('unpack.asar', 'a.txt'); + const r = fs.realpathSync.native(path.join(parent, p)); + expect(r).to.equal(path.join(parent, p)); + }); + + itremote('throws ENOENT error when can not find file', () => { + const parent = fs.realpathSync.native(asarDir); + const p = path.join('a.asar', 'not-exist'); + expect(() => { + fs.realpathSync.native(path.join(parent, p)); + }).to.throw(/ENOENT/); + }); + }); + + describe('fs.realpath', () => { + itremote('returns real path root', async () => { + const parent = fs.realpathSync(asarDir); + const p = 'a.asar'; + const r = await promisify(fs.realpath)(path.join(parent, p)); + expect(r).to.equal(path.join(parent, p)); + }); + + itremote('returns real path of a normal file', async () => { + const parent = fs.realpathSync(asarDir); + const p = path.join('a.asar', 'file1'); + const r = await promisify(fs.realpath)(path.join(parent, p)); + expect(r).to.equal(path.join(parent, p)); + }); + + itremote('returns real path of a normal directory', async () => { + const parent = fs.realpathSync(asarDir); + const p = path.join('a.asar', 'dir1'); + const r = await promisify(fs.realpath)(path.join(parent, p)); + expect(r).to.equal(path.join(parent, p)); + }); + + itremote('returns real path of a linked file', async () => { + const parent = fs.realpathSync(asarDir); + const p = path.join('a.asar', 'link2', 'link1'); + const r = await promisify(fs.realpath)(path.join(parent, p)); + expect(r).to.equal(path.join(parent, 'a.asar', 'file1')); + }); + + itremote('returns real path of a linked directory', async () => { + const parent = fs.realpathSync(asarDir); + const p = path.join('a.asar', 'link2', 'link2'); + const r = await promisify(fs.realpath)(path.join(parent, p)); + expect(r).to.equal(path.join(parent, 'a.asar', 'dir1')); + }); + + itremote('returns real path of an unpacked file', async () => { + const parent = fs.realpathSync(asarDir); + const p = path.join('unpack.asar', 'a.txt'); + const r = await promisify(fs.realpath)(path.join(parent, p)); + expect(r).to.equal(path.join(parent, p)); + }); + + itremote('throws ENOENT error when can not find file', async () => { + const parent = fs.realpathSync(asarDir); + const p = path.join('a.asar', 'not-exist'); + const err = await new Promise(resolve => fs.realpath(path.join(parent, p), resolve)); + expect(err.code).to.equal('ENOENT'); + }); + }); + + describe('fs.promises.realpath', () => { + itremote('returns real path root', async () => { + const parent = fs.realpathSync(asarDir); + const p = 'a.asar'; + const r = await fs.promises.realpath(path.join(parent, p)); + expect(r).to.equal(path.join(parent, p)); + }); + + itremote('returns real path of a normal file', async () => { + const parent = fs.realpathSync(asarDir); + const p = path.join('a.asar', 'file1'); + const r = await fs.promises.realpath(path.join(parent, p)); + expect(r).to.equal(path.join(parent, p)); + }); + + itremote('returns real path of a normal directory', async () => { + const parent = fs.realpathSync(asarDir); + const p = path.join('a.asar', 'dir1'); + const r = await fs.promises.realpath(path.join(parent, p)); + expect(r).to.equal(path.join(parent, p)); + }); + + itremote('returns real path of a linked file', async () => { + const parent = fs.realpathSync(asarDir); + const p = path.join('a.asar', 'link2', 'link1'); + const r = await fs.promises.realpath(path.join(parent, p)); + expect(r).to.equal(path.join(parent, 'a.asar', 'file1')); + }); + + itremote('returns real path of a linked directory', async () => { + const parent = fs.realpathSync(asarDir); + const p = path.join('a.asar', 'link2', 'link2'); + const r = await fs.promises.realpath(path.join(parent, p)); + expect(r).to.equal(path.join(parent, 'a.asar', 'dir1')); + }); + + itremote('returns real path of an unpacked file', async () => { + const parent = fs.realpathSync(asarDir); + const p = path.join('unpack.asar', 'a.txt'); + const r = await fs.promises.realpath(path.join(parent, p)); + expect(r).to.equal(path.join(parent, p)); + }); + + itremote('throws ENOENT error when can not find file', async () => { + const parent = fs.realpathSync(asarDir); + const p = path.join('a.asar', 'not-exist'); + await expectToThrowErrorWithCode(() => fs.promises.realpath(path.join(parent, p)), 'ENOENT'); + }); + }); + + describe('fs.realpath.native', () => { + itremote('returns real path root', async () => { + const parent = fs.realpathSync.native(asarDir); + const p = 'a.asar'; + const r = await promisify(fs.realpath.native)(path.join(parent, p)); + expect(r).to.equal(path.join(parent, p)); + }); + + itremote('returns real path of a normal file', async () => { + const parent = fs.realpathSync.native(asarDir); + const p = path.join('a.asar', 'file1'); + const r = await promisify(fs.realpath.native)(path.join(parent, p)); + expect(r).to.equal(path.join(parent, p)); + }); + + itremote('returns real path of a normal directory', async () => { + const parent = fs.realpathSync.native(asarDir); + const p = path.join('a.asar', 'dir1'); + const r = await promisify(fs.realpath.native)(path.join(parent, p)); + expect(r).to.equal(path.join(parent, p)); + }); + + itremote('returns real path of a linked file', async () => { + const parent = fs.realpathSync.native(asarDir); + const p = path.join('a.asar', 'link2', 'link1'); + const r = await promisify(fs.realpath.native)(path.join(parent, p)); + expect(r).to.equal(path.join(parent, 'a.asar', 'file1')); + }); + + itremote('returns real path of a linked directory', async () => { + const parent = fs.realpathSync.native(asarDir); + const p = path.join('a.asar', 'link2', 'link2'); + const r = await promisify(fs.realpath.native)(path.join(parent, p)); + expect(r).to.equal(path.join(parent, 'a.asar', 'dir1')); + }); + + itremote('returns real path of an unpacked file', async () => { + const parent = fs.realpathSync.native(asarDir); + const p = path.join('unpack.asar', 'a.txt'); + const r = await promisify(fs.realpath.native)(path.join(parent, p)); + expect(r).to.equal(path.join(parent, p)); + }); + + itremote('throws ENOENT error when can not find file', async () => { + const parent = fs.realpathSync.native(asarDir); + const p = path.join('a.asar', 'not-exist'); + const err = await new Promise(resolve => fs.realpath.native(path.join(parent, p), resolve)); + expect(err.code).to.equal('ENOENT'); + }); + }); + + describe('fs.readdirSync', function () { + itremote('reads dirs from root', function () { + const p = path.join(asarDir, 'a.asar'); + const dirs = fs.readdirSync(p); + expect(dirs).to.deep.equal(['dir1', 'dir2', 'dir3', 'file1', 'file2', 'file3', 'link1', 'link2', 'ping.js']); + }); + + itremote('reads dirs from a normal dir', function () { + const p = path.join(asarDir, 'a.asar', 'dir1'); + const dirs = fs.readdirSync(p); + expect(dirs).to.deep.equal(['file1', 'file2', 'file3', 'link1', 'link2']); + }); + + itremote('supports withFileTypes', function () { + const p = path.join(asarDir, 'a.asar'); + const dirs = fs.readdirSync(p, { withFileTypes: true }); + for (const dir of dirs) { + expect(dir instanceof fs.Dirent).to.be.true(); + } + const names = dirs.map(a => a.name); + expect(names).to.deep.equal(['dir1', 'dir2', 'dir3', 'file1', 'file2', 'file3', 'link1', 'link2', 'ping.js']); + }); + + itremote('supports withFileTypes for a deep directory', function () { + const p = path.join(asarDir, 'a.asar', 'dir3'); + const dirs = fs.readdirSync(p, { withFileTypes: true }); + for (const dir of dirs) { + expect(dir instanceof fs.Dirent).to.be.true(); + } + const names = dirs.map(a => a.name); + expect(names).to.deep.equal(['file1', 'file2', 'file3']); + }); + + itremote('reads dirs from a linked dir', function () { + const p = path.join(asarDir, 'a.asar', 'link2', 'link2'); + const dirs = fs.readdirSync(p); + expect(dirs).to.deep.equal(['file1', 'file2', 'file3', 'link1', 'link2']); + }); + + itremote('throws ENOENT error when can not find file', function () { + const p = path.join(asarDir, 'a.asar', 'not-exist'); + expect(() => { + fs.readdirSync(p); + }).to.throw(/ENOENT/); + }); + }); + + describe('fs.readdir', function () { + itremote('reads dirs from root', async () => { + const p = path.join(asarDir, 'a.asar'); + const dirs = await promisify(fs.readdir)(p); + expect(dirs).to.deep.equal(['dir1', 'dir2', 'dir3', 'file1', 'file2', 'file3', 'link1', 'link2', 'ping.js']); + }); + + itremote('supports withFileTypes', async () => { + const p = path.join(asarDir, 'a.asar'); + + const dirs = await promisify(fs.readdir)(p, { withFileTypes: true }); + for (const dir of dirs) { + expect(dir instanceof fs.Dirent).to.be.true(); + } + + const names = dirs.map((a: any) => a.name); + expect(names).to.deep.equal(['dir1', 'dir2', 'dir3', 'file1', 'file2', 'file3', 'link1', 'link2', 'ping.js']); + }); + + itremote('reads dirs from a normal dir', async () => { + const p = path.join(asarDir, 'a.asar', 'dir1'); + const dirs = await promisify(fs.readdir)(p); + expect(dirs).to.deep.equal(['file1', 'file2', 'file3', 'link1', 'link2']); + }); + + itremote('reads dirs from a linked dir', async () => { + const p = path.join(asarDir, 'a.asar', 'link2', 'link2'); + const dirs = await promisify(fs.readdir)(p); + expect(dirs).to.deep.equal(['file1', 'file2', 'file3', 'link1', 'link2']); + }); + + itremote('throws ENOENT error when can not find file', async () => { + const p = path.join(asarDir, 'a.asar', 'not-exist'); + const err = await new Promise(resolve => fs.readdir(p, resolve)); + expect(err.code).to.equal('ENOENT'); + }); + }); + + describe('fs.promises.readdir', function () { + itremote('reads dirs from root', async function () { + const p = path.join(asarDir, 'a.asar'); + const dirs = await fs.promises.readdir(p); + expect(dirs).to.deep.equal(['dir1', 'dir2', 'dir3', 'file1', 'file2', 'file3', 'link1', 'link2', 'ping.js']); + }); + + itremote('supports withFileTypes', async function () { + const p = path.join(asarDir, 'a.asar'); + const dirs = await fs.promises.readdir(p, { withFileTypes: true }); + for (const dir of dirs) { + expect(dir instanceof fs.Dirent).to.be.true(); + } + const names = dirs.map(a => a.name); + expect(names).to.deep.equal(['dir1', 'dir2', 'dir3', 'file1', 'file2', 'file3', 'link1', 'link2', 'ping.js']); + }); + + itremote('reads dirs from a normal dir', async function () { + const p = path.join(asarDir, 'a.asar', 'dir1'); + const dirs = await fs.promises.readdir(p); + expect(dirs).to.deep.equal(['file1', 'file2', 'file3', 'link1', 'link2']); + }); + + itremote('reads dirs from a linked dir', async function () { + const p = path.join(asarDir, 'a.asar', 'link2', 'link2'); + const dirs = await fs.promises.readdir(p); + expect(dirs).to.deep.equal(['file1', 'file2', 'file3', 'link1', 'link2']); + }); + + itremote('throws ENOENT error when can not find file', async function () { + const p = path.join(asarDir, 'a.asar', 'not-exist'); + await expectToThrowErrorWithCode(() => fs.promises.readdir(p), 'ENOENT'); + }); + }); + + describe('fs.openSync', function () { + itremote('opens a normal/linked/under-linked-directory file', function () { + const ref2 = ['file1', 'link1', path.join('link2', 'file1')]; + for (let j = 0, len = ref2.length; j < len; j++) { + const file = ref2[j]; + const p = path.join(asarDir, 'a.asar', file); + const fd = fs.openSync(p, 'r'); + const buffer = Buffer.alloc(6); + fs.readSync(fd, buffer, 0, 6, 0); + expect(String(buffer).trim()).to.equal('file1'); + fs.closeSync(fd); + } + }); + + itremote('throws ENOENT error when can not find file', function () { + const p = path.join(asarDir, 'a.asar', 'not-exist'); + expect(() => { + (fs.openSync as any)(p); + }).to.throw(/ENOENT/); + }); + }); + + describe('fs.open', function () { + itremote('opens a normal file', async function () { + const p = path.join(asarDir, 'a.asar', 'file1'); + const fd = await promisify(fs.open)(p, 'r'); + const buffer = Buffer.alloc(6); + await promisify(fs.read)(fd, buffer, 0, 6, 0); + expect(String(buffer).trim()).to.equal('file1'); + await promisify(fs.close)(fd); + }); + + itremote('throws ENOENT error when can not find file', async function () { + const p = path.join(asarDir, 'a.asar', 'not-exist'); + const err = await new Promise(resolve => fs.open(p, 'r', resolve)); + expect(err.code).to.equal('ENOENT'); + }); + }); + + describe('fs.promises.open', function () { + itremote('opens a normal file', async function () { + const p = path.join(asarDir, 'a.asar', 'file1'); + const fh = await fs.promises.open(p, 'r'); + const buffer = Buffer.alloc(6); + await fh.read(buffer, 0, 6, 0); + expect(String(buffer).trim()).to.equal('file1'); + await fh.close(); + }); + + itremote('throws ENOENT error when can not find file', async function () { + const p = path.join(asarDir, 'a.asar', 'not-exist'); + await expectToThrowErrorWithCode(() => fs.promises.open(p, 'r'), 'ENOENT'); + }); + }); + + describe('fs.mkdir', function () { + itremote('throws error when calling inside asar archive', async function () { + const p = path.join(asarDir, 'a.asar', 'not-exist'); + const err = await new Promise(resolve => fs.mkdir(p, resolve)); + expect(err.code).to.equal('ENOTDIR'); + }); + }); + + describe('fs.promises.mkdir', function () { + itremote('throws error when calling inside asar archive', async function () { + const p = path.join(asarDir, 'a.asar', 'not-exist'); + await expectToThrowErrorWithCode(() => fs.promises.mkdir(p), 'ENOTDIR'); + }); + }); + + describe('fs.mkdirSync', function () { + itremote('throws error when calling inside asar archive', function () { + const p = path.join(asarDir, 'a.asar', 'not-exist'); + expect(() => { + fs.mkdirSync(p); + }).to.throw(/ENOTDIR/); + }); + }); + + describe('fs.exists', function () { + itremote('handles an existing file', async function () { + const p = path.join(asarDir, 'a.asar', 'file1'); + // eslint-disable-next-line + const exists = await new Promise(resolve => fs.exists(p, resolve)) + expect(exists).to.be.true(); + }); + + itremote('handles a non-existent file', async function () { + const p = path.join(asarDir, 'a.asar', 'not-exist'); + // eslint-disable-next-line + const exists = await new Promise(resolve => fs.exists(p, resolve)) + expect(exists).to.be.false(); + }); + + itremote('promisified version handles an existing file', async () => { + const p = path.join(asarDir, 'a.asar', 'file1'); + // eslint-disable-next-line + const exists = await require('util').promisify(fs.exists)(p) + expect(exists).to.be.true(); + }); + + itremote('promisified version handles a non-existent file', async function () { + const p = path.join(asarDir, 'a.asar', 'not-exist'); + // eslint-disable-next-line + const exists = await require('util').promisify(fs.exists)(p) + expect(exists).to.be.false(); + }); + }); + + describe('fs.existsSync', function () { + itremote('handles an existing file', function () { + const p = path.join(asarDir, 'a.asar', 'file1'); + expect(fs.existsSync(p)).to.be.true(); + }); + + itremote('handles a non-existent file', function () { + const p = path.join(asarDir, 'a.asar', 'not-exist'); + expect(fs.existsSync(p)).to.be.false(); + }); + }); + + describe('fs.access', function () { + itremote('accesses a normal file', async function () { + const p = path.join(asarDir, 'a.asar', 'file1'); + await promisify(fs.access)(p); + }); + + itremote('throws an error when called with write mode', async function () { + const p = path.join(asarDir, 'a.asar', 'file1'); + const err = await new Promise(resolve => fs.access(p, fs.constants.R_OK | fs.constants.W_OK, resolve)); + expect(err.code).to.equal('EACCES'); + }); + + itremote('throws an error when called on non-existent file', async function () { + const p = path.join(asarDir, 'a.asar', 'not-exist'); + const err = await new Promise(resolve => fs.access(p, fs.constants.R_OK | fs.constants.W_OK, resolve)); + expect(err.code).to.equal('ENOENT'); + }); + + itremote('allows write mode for unpacked files', async function () { + const p = path.join(asarDir, 'unpack.asar', 'a.txt'); + await promisify(fs.access)(p, fs.constants.R_OK | fs.constants.W_OK); + }); + }); + + describe('fs.promises.access', function () { + itremote('accesses a normal file', async function () { + const p = path.join(asarDir, 'a.asar', 'file1'); + await fs.promises.access(p); + }); + + itremote('throws an error when called with write mode', async function () { + const p = path.join(asarDir, 'a.asar', 'file1'); + await expectToThrowErrorWithCode(() => fs.promises.access(p, fs.constants.R_OK | fs.constants.W_OK), 'EACCES'); + }); + + itremote('throws an error when called on non-existent file', async function () { + const p = path.join(asarDir, 'a.asar', 'not-exist'); + await expectToThrowErrorWithCode(() => fs.promises.access(p), 'ENOENT'); + }); + + itremote('allows write mode for unpacked files', async function () { + const p = path.join(asarDir, 'unpack.asar', 'a.txt'); + await fs.promises.access(p, fs.constants.R_OK | fs.constants.W_OK); + }); + }); + + describe('fs.accessSync', function () { + itremote('accesses a normal file', function () { + const p = path.join(asarDir, 'a.asar', 'file1'); + expect(() => { + fs.accessSync(p); + }).to.not.throw(); + }); + + itremote('throws an error when called with write mode', function () { + const p = path.join(asarDir, 'a.asar', 'file1'); + expect(() => { + fs.accessSync(p, fs.constants.R_OK | fs.constants.W_OK); + }).to.throw(/EACCES/); + }); + + itremote('throws an error when called on non-existent file', function () { + const p = path.join(asarDir, 'a.asar', 'not-exist'); + expect(() => { + fs.accessSync(p); + }).to.throw(/ENOENT/); + }); + + itremote('allows write mode for unpacked files', function () { + const p = path.join(asarDir, 'unpack.asar', 'a.txt'); + expect(() => { + fs.accessSync(p, fs.constants.R_OK | fs.constants.W_OK); + }).to.not.throw(); + }); + }); + + ifdescribe(features.isRunAsNodeEnabled())('child_process.fork', function () { + itremote('opens a normal js file', async function () { + const child = require('child_process').fork(path.join(asarDir, 'a.asar', 'ping.js')); + child.send('message'); + const msg = await new Promise(resolve => child.once('message', resolve)); + expect(msg).to.equal('message'); + }); + + itremote('supports asar in the forked js', async function (fixtures: string) { + const file = path.join(asarDir, 'a.asar', 'file1'); + const child = require('child_process').fork(path.join(fixtures, 'module', 'asar.js')); + child.send(file); + const content = await new Promise(resolve => child.once('message', resolve)); + expect(content).to.equal(fs.readFileSync(file).toString()); + }, [fixtures]); + }); + + describe('child_process.exec', function () { + itremote('should not try to extract the command if there is a reference to a file inside an .asar', async function () { + const echo = path.join(asarDir, 'echo.asar', 'echo'); + + const stdout = await promisify(require('child_process').exec)('echo ' + echo + ' foo bar'); + expect(stdout.toString().replace(/\r/g, '')).to.equal(echo + ' foo bar\n'); + }); + }); + + describe('child_process.execSync', function () { + itremote('should not try to extract the command if there is a reference to a file inside an .asar', async function () { + const echo = path.join(asarDir, 'echo.asar', 'echo'); + + const stdout = require('child_process').execSync('echo ' + echo + ' foo bar'); + expect(stdout.toString().replace(/\r/g, '')).to.equal(echo + ' foo bar\n'); + }); + }); + + ifdescribe(process.platform === 'darwin' && process.arch !== 'arm64')('child_process.execFile', function () { + itremote('executes binaries', async function () { + const echo = path.join(asarDir, 'echo.asar', 'echo'); + const stdout = await promisify(require('child_process').execFile)(echo, ['test']); + expect(stdout).to.equal('test\n'); + }); + + itremote('executes binaries without callback', async function () { + const echo = path.join(asarDir, 'echo.asar', 'echo'); + const process = require('child_process').execFile(echo, ['test']); + const code = await new Promise(resolve => process.once('close', resolve)); + expect(code).to.equal(0); + process.on('error', function () { + throw new Error('error'); + }); + }); + + itremote('execFileSync executes binaries', function () { + const echo = path.join(asarDir, 'echo.asar', 'echo'); + const output = require('child_process').execFileSync(echo, ['test']); + expect(String(output)).to.equal('test\n'); + }); + }); + + describe('internalModuleReadJSON', function () { + itremote('reads a normal file', function () { + const { internalModuleReadJSON } = (process as any).binding('fs'); + const file1 = path.join(asarDir, 'a.asar', 'file1'); + const [s1, c1] = internalModuleReadJSON(file1); + expect([s1.toString().trim(), c1]).to.eql(['file1', true]); + + const file2 = path.join(asarDir, 'a.asar', 'file2'); + const [s2, c2] = internalModuleReadJSON(file2); + expect([s2.toString().trim(), c2]).to.eql(['file2', true]); + + const file3 = path.join(asarDir, 'a.asar', 'file3'); + const [s3, c3] = internalModuleReadJSON(file3); + expect([s3.toString().trim(), c3]).to.eql(['file3', true]); + }); + + itremote('reads a normal file with unpacked files', function () { + const { internalModuleReadJSON } = (process as any).binding('fs'); + const p = path.join(asarDir, 'unpack.asar', 'a.txt'); + const [s, c] = internalModuleReadJSON(p); + expect([s.toString().trim(), c]).to.eql(['a', true]); + }); + }); + + describe('util.promisify', function () { + itremote('can promisify all fs functions', function () { + const originalFs = require('original-fs'); + const util = require('util'); + const { hasOwnProperty } = Object.prototype; + + for (const [propertyName, originalValue] of Object.entries(originalFs)) { + // Some properties exist but have a value of `undefined` on some platforms. + // E.g. `fs.lchmod`, which in only available on MacOS, see + // https://nodejs.org/docs/latest-v10.x/api/fs.html#fs_fs_lchmod_path_mode_callback + // Also check for `null`s, `hasOwnProperty()` can't handle them. + if (typeof originalValue === 'undefined' || originalValue === null) continue; + + if (hasOwnProperty.call(originalValue, util.promisify.custom)) { + expect(fs).to.have.own.property(propertyName) + .that.has.own.property(util.promisify.custom); + } + } + }); + }); + + describe('process.noAsar', function () { + const errorName = process.platform === 'win32' ? 'ENOENT' : 'ENOTDIR'; + + beforeEach(async function () { + return (await getRemoteContext()).webContents.executeJavaScript(` + process.noAsar = true; + `); + }); + + afterEach(async function () { + return (await getRemoteContext()).webContents.executeJavaScript(` + process.noAsar = false; + `); + }); + + itremote('disables asar support in sync API', function (errorName: string) { + const file = path.join(asarDir, 'a.asar', 'file1'); + const dir = path.join(asarDir, 'a.asar', 'dir1'); + console.log(1); + expect(() => { + fs.readFileSync(file); + }).to.throw(new RegExp(errorName)); + expect(() => { + fs.lstatSync(file); + }).to.throw(new RegExp(errorName)); + expect(() => { + fs.realpathSync(file); + }).to.throw(new RegExp(errorName)); + expect(() => { + fs.readdirSync(dir); + }).to.throw(new RegExp(errorName)); + }, [errorName]); + + itremote('disables asar support in async API', async function (errorName: string) { + const file = path.join(asarDir, 'a.asar', 'file1'); + const dir = path.join(asarDir, 'a.asar', 'dir1'); + await new Promise(resolve => { + fs.readFile(file, function (error) { + expect(error?.code).to.equal(errorName); + fs.lstat(file, function (error) { + expect(error?.code).to.equal(errorName); + fs.realpath(file, function (error) { + expect(error?.code).to.equal(errorName); + fs.readdir(dir, function (error) { + expect(error?.code).to.equal(errorName); + resolve(); + }); + }); + }); + }); + }); + }, [errorName]); + + itremote('disables asar support in promises API', async function (errorName: string) { + const file = path.join(asarDir, 'a.asar', 'file1'); + const dir = path.join(asarDir, 'a.asar', 'dir1'); + await expect(fs.promises.readFile(file)).to.be.eventually.rejectedWith(Error, new RegExp(errorName)); + await expect(fs.promises.lstat(file)).to.be.eventually.rejectedWith(Error, new RegExp(errorName)); + await expect(fs.promises.realpath(file)).to.be.eventually.rejectedWith(Error, new RegExp(errorName)); + await expect(fs.promises.readdir(dir)).to.be.eventually.rejectedWith(Error, new RegExp(errorName)); + }, [errorName]); + + itremote('treats *.asar as normal file', function () { + const originalFs = require('original-fs'); + const asar = path.join(asarDir, 'a.asar'); + const content1 = fs.readFileSync(asar); + const content2 = originalFs.readFileSync(asar); + expect(content1.compare(content2)).to.equal(0); + expect(() => { + fs.readdirSync(asar); + }).to.throw(/ENOTDIR/); + }); + + itremote('is reset to its original value when execSync throws an error', function () { + process.noAsar = false; + expect(() => { + require('child_process').execSync(path.join(__dirname, 'does-not-exist.txt')); + }).to.throw(); + expect(process.noAsar).to.be.false(); + }); + }); + /* + + describe('process.env.ELECTRON_NO_ASAR', function () { + before(function () { + if (!features.isRunAsNodeEnabled()) { + this.skip(); + } + }); + + itremote('disables asar support in forked processes', function (done) { + const forked = ChildProcess.fork(path.join(__dirname, 'fixtures', 'module', 'no-asar.js'), [], { + env: { + ELECTRON_NO_ASAR: true + } + }); + forked.on('message', function (stats) { + try { + expect(stats.isFile).to.be.true(); + expect(stats.size).to.equal(3458); + done(); + } catch (e) { + done(e); + } + }); + }); + + itremote('disables asar support in spawned processes', function (done) { + const spawned = ChildProcess.spawn(process.execPath, [path.join(__dirname, 'fixtures', 'module', 'no-asar.js')], { + env: { + ELECTRON_NO_ASAR: true, + ELECTRON_RUN_AS_NODE: true + } + }); + + let output = ''; + spawned.stdout.on('data', function (data) { + output += data; + }); + spawned.stdout.on('close', function () { + try { + const stats = JSON.parse(output); + expect(stats.isFile).to.be.true(); + expect(stats.size).to.equal(3458); + done(); + } catch (e) { + done(e); + } + }); + }); + }); + */ + }); + + describe('asar protocol', function () { + itremote('can request a file in package', async function () { + const p = path.resolve(asarDir, 'a.asar', 'file1'); + const response = await fetch('file://' + p); + const data = await response.text(); + expect(data.trim()).to.equal('file1'); + }); + + itremote('can request a file in package with unpacked files', async function () { + const p = path.resolve(asarDir, 'unpack.asar', 'a.txt'); + const response = await fetch('file://' + p); + const data = await response.text(); + expect(data.trim()).to.equal('a'); + }); + + itremote('can request a linked file in package', async function () { + const p = path.resolve(asarDir, 'a.asar', 'link2', 'link1'); + const response = await fetch('file://' + p); + const data = await response.text(); + expect(data.trim()).to.equal('file1'); + }); + + itremote('can request a file in filesystem', async function () { + const p = path.resolve(asarDir, 'file'); + const response = await fetch('file://' + p); + const data = await response.text(); + expect(data.trim()).to.equal('file'); + }); + + itremote('gets error when file is not found', async function () { + const p = path.resolve(asarDir, 'a.asar', 'no-exist'); + try { + const response = await fetch('file://' + p); + expect(response.status).to.equal(404); + } catch (error: any) { + expect(error.message).to.equal('Failed to fetch'); + } + }); + }); + + describe('original-fs module', function () { + itremote('treats .asar as file', function () { + const file = path.join(asarDir, 'a.asar'); + const originalFs = require('original-fs'); + const stats = originalFs.statSync(file); + expect(stats.isFile()).to.be.true(); + }); + + /* + ifit(features.isRunAsNodeEnabled())('is available in forked scripts', async function () { + const child = ChildProcess.fork(path.join(fixtures, 'module', 'original-fs.js')); + const message = emittedOnce(child, 'message'); + child.send('message'); + const [msg] = await message; + expect(msg).to.equal('object'); + }); + */ + + itremote('can be used with streams', () => { + const originalFs = require('original-fs'); + originalFs.createReadStream(path.join(asarDir, 'a.asar')); + }); + + itremote('can recursively delete a directory with an asar file in itremote', () => { + const deleteDir = path.join(asarDir, 'deleteme'); + fs.mkdirSync(deleteDir); + + const originalFs = require('original-fs'); + originalFs.rmdirSync(deleteDir, { recursive: true }); + + expect(fs.existsSync(deleteDir)).to.be.false(); + }); + + itremote('has the same APIs as fs', function () { + expect(Object.keys(require('fs'))).to.deep.equal(Object.keys(require('original-fs'))); + expect(Object.keys(require('fs').promises)).to.deep.equal(Object.keys(require('original-fs').promises)); + }); + }); + + describe('graceful-fs module', function () { + itremote('recognize asar archives', function () { + const gfs = require('graceful-fs'); + + const p = path.join(asarDir, 'a.asar', 'link1'); + expect(gfs.readFileSync(p).toString().trim()).to.equal('file1'); + }); + itremote('does not touch global fs object', function () { + const gfs = require('graceful-fs'); + expect(fs.readdir).to.not.equal(gfs.readdir); + }); + }); + + describe('mkdirp module', function () { + itremote('throws error when calling inside asar archive', function () { + const mkdirp = require('mkdirp'); + + const p = path.join(asarDir, 'a.asar', 'not-exist'); + expect(() => { + mkdirp.sync(p); + }).to.throw(/ENOTDIR/); + }); + }); + + describe('native-image', function () { + itremote('reads image from asar archive', function () { + const p = path.join(asarDir, 'logo.asar', 'logo.png'); + const logo = require('electron').nativeImage.createFromPath(p); + expect(logo.getSize()).to.deep.equal({ + width: 55, + height: 55 + }); + }); + + itremote('reads image from asar archive with unpacked files', function () { + const p = path.join(asarDir, 'unpack.asar', 'atom.png'); + const logo = require('electron').nativeImage.createFromPath(p); + expect(logo.getSize()).to.deep.equal({ + width: 1024, + height: 1024 + }); + }); + }); +}); diff --git a/spec-main/spec-helpers.ts b/spec-main/spec-helpers.ts index 50f15102ccdc..99254ba585c5 100644 --- a/spec-main/spec-helpers.ts +++ b/spec-main/spec-helpers.ts @@ -149,9 +149,10 @@ export async function repeatedly ( } async function makeRemoteContext (opts?: any) { - const { webPreferences, ...rest } = opts ?? {}; + const { webPreferences, setup, url = 'about:blank', ...rest } = opts ?? {}; const w = new BrowserWindow({ show: false, webPreferences: { nodeIntegration: true, contextIsolation: false, ...webPreferences }, ...rest }); - await w.loadURL('about:blank'); + await w.loadURL(url.toString()); + if (setup) await w.webContents.executeJavaScript(setup); return w; } @@ -179,6 +180,7 @@ export async function itremote (name: string, fn: Function, args?: any[]) { const { ok, message } = await w.webContents.executeJavaScript(`(async () => { try { const chai_1 = require('chai') + chai_1.use(require('chai-as-promised')) chai_1.use(require('dirty-chai')) await (${fn})(...${JSON.stringify(args ?? [])}) return {ok: true}; diff --git a/spec/asar-spec.js b/spec/asar-spec.js deleted file mode 100644 index d95dcb4711c8..000000000000 --- a/spec/asar-spec.js +++ /dev/null @@ -1,1719 +0,0 @@ -const { expect } = require('chai'); -const ChildProcess = require('child_process'); -const fs = require('fs'); -const path = require('path'); -const temp = require('temp').track(); -const util = require('util'); -const { emittedOnce } = require('./events-helpers'); -const { ifit, ifdescribe } = require('./spec-helpers'); -const nativeImage = require('electron').nativeImage; - -const features = process._linkedBinding('electron_common_features'); - -async function expectToThrowErrorWithCode (func, code) { - let error; - try { - await func(); - } catch (e) { - error = e; - } - - expect(error).is.an('Error'); - expect(error).to.have.property('code').which.equals(code); -} - -describe('asar package', function () { - const fixtures = path.join(__dirname, 'fixtures'); - const asarDir = path.join(fixtures, 'test.asar'); - - describe('node api', function () { - it('supports paths specified as a Buffer', function () { - const file = Buffer.from(path.join(asarDir, 'a.asar', 'file1')); - expect(fs.existsSync(file)).to.be.true(); - }); - - describe('fs.readFileSync', function () { - it('does not leak fd', function () { - let readCalls = 1; - while (readCalls <= 10000) { - fs.readFileSync(path.join(process.resourcesPath, 'default_app.asar', 'main.js')); - readCalls++; - } - }); - - it('reads a normal file', function () { - const file1 = path.join(asarDir, 'a.asar', 'file1'); - expect(fs.readFileSync(file1).toString().trim()).to.equal('file1'); - const file2 = path.join(asarDir, 'a.asar', 'file2'); - expect(fs.readFileSync(file2).toString().trim()).to.equal('file2'); - const file3 = path.join(asarDir, 'a.asar', 'file3'); - expect(fs.readFileSync(file3).toString().trim()).to.equal('file3'); - }); - - it('reads from a empty file', function () { - const file = path.join(asarDir, 'empty.asar', 'file1'); - const buffer = fs.readFileSync(file); - expect(buffer).to.be.empty(); - expect(buffer.toString()).to.equal(''); - }); - - it('reads a linked file', function () { - const p = path.join(asarDir, 'a.asar', 'link1'); - expect(fs.readFileSync(p).toString().trim()).to.equal('file1'); - }); - - it('reads a file from linked directory', function () { - const p1 = path.join(asarDir, 'a.asar', 'link2', 'file1'); - expect(fs.readFileSync(p1).toString().trim()).to.equal('file1'); - const p2 = path.join(asarDir, 'a.asar', 'link2', 'link2', 'file1'); - expect(fs.readFileSync(p2).toString().trim()).to.equal('file1'); - }); - - it('throws ENOENT error when can not find file', function () { - const p = path.join(asarDir, 'a.asar', 'not-exist'); - expect(() => { - fs.readFileSync(p); - }).to.throw(/ENOENT/); - }); - - it('passes ENOENT error to callback when can not find file', function () { - const p = path.join(asarDir, 'a.asar', 'not-exist'); - let async = false; - fs.readFile(p, function (error) { - expect(async).to.be.true(); - expect(error).to.match(/ENOENT/); - }); - async = true; - }); - - it('reads a normal file with unpacked files', function () { - const p = path.join(asarDir, 'unpack.asar', 'a.txt'); - expect(fs.readFileSync(p).toString().trim()).to.equal('a'); - }); - - it('reads a file in filesystem', function () { - const p = path.resolve(asarDir, 'file'); - expect(fs.readFileSync(p).toString().trim()).to.equal('file'); - }); - }); - - describe('fs.readFile', function () { - it('reads a normal file', function (done) { - const p = path.join(asarDir, 'a.asar', 'file1'); - fs.readFile(p, function (err, content) { - try { - expect(err).to.be.null(); - expect(String(content).trim()).to.equal('file1'); - done(); - } catch (e) { - done(e); - } - }); - }); - - it('reads from a empty file', function (done) { - const p = path.join(asarDir, 'empty.asar', 'file1'); - fs.readFile(p, function (err, content) { - try { - expect(err).to.be.null(); - expect(String(content)).to.equal(''); - done(); - } catch (e) { - done(e); - } - }); - }); - - it('reads from a empty file with encoding', function (done) { - const p = path.join(asarDir, 'empty.asar', 'file1'); - fs.readFile(p, 'utf8', function (err, content) { - try { - expect(err).to.be.null(); - expect(content).to.equal(''); - done(); - } catch (e) { - done(e); - } - }); - }); - - it('reads a linked file', function (done) { - const p = path.join(asarDir, 'a.asar', 'link1'); - fs.readFile(p, function (err, content) { - try { - expect(err).to.be.null(); - expect(String(content).trim()).to.equal('file1'); - done(); - } catch (e) { - done(e); - } - }); - }); - - it('reads a file from linked directory', function (done) { - const p = path.join(asarDir, 'a.asar', 'link2', 'link2', 'file1'); - fs.readFile(p, function (err, content) { - try { - expect(err).to.be.null(); - expect(String(content).trim()).to.equal('file1'); - done(); - } catch (e) { - done(e); - } - }); - }); - - it('throws ENOENT error when can not find file', function (done) { - const p = path.join(asarDir, 'a.asar', 'not-exist'); - fs.readFile(p, function (err) { - try { - expect(err.code).to.equal('ENOENT'); - done(); - } catch (e) { - done(e); - } - }); - }); - }); - - describe('fs.promises.readFile', function () { - it('reads a normal file', async function () { - const p = path.join(asarDir, 'a.asar', 'file1'); - const content = await fs.promises.readFile(p); - expect(String(content).trim()).to.equal('file1'); - }); - - it('reads from a empty file', async function () { - const p = path.join(asarDir, 'empty.asar', 'file1'); - const content = await fs.promises.readFile(p); - expect(String(content)).to.equal(''); - }); - - it('reads from a empty file with encoding', async function () { - const p = path.join(asarDir, 'empty.asar', 'file1'); - const content = await fs.promises.readFile(p, 'utf8'); - expect(content).to.equal(''); - }); - - it('reads a linked file', async function () { - const p = path.join(asarDir, 'a.asar', 'link1'); - const content = await fs.promises.readFile(p); - expect(String(content).trim()).to.equal('file1'); - }); - - it('reads a file from linked directory', async function () { - const p = path.join(asarDir, 'a.asar', 'link2', 'link2', 'file1'); - const content = await fs.promises.readFile(p); - expect(String(content).trim()).to.equal('file1'); - }); - - it('throws ENOENT error when can not find file', async function () { - const p = path.join(asarDir, 'a.asar', 'not-exist'); - await expectToThrowErrorWithCode(() => fs.promises.readFile(p), 'ENOENT'); - }); - }); - - describe('fs.copyFile', function () { - it('copies a normal file', function (done) { - const p = path.join(asarDir, 'a.asar', 'file1'); - const dest = temp.path(); - fs.copyFile(p, dest, function (err) { - try { - expect(err).to.be.null(); - expect(fs.readFileSync(p).equals(fs.readFileSync(dest))).to.be.true(); - done(); - } catch (e) { - done(e); - } - }); - }); - - it('copies a unpacked file', function (done) { - const p = path.join(asarDir, 'unpack.asar', 'a.txt'); - const dest = temp.path(); - fs.copyFile(p, dest, function (err) { - try { - expect(err).to.be.null(); - expect(fs.readFileSync(p).equals(fs.readFileSync(dest))).to.be.true(); - done(); - } catch (e) { - done(e); - } - }); - }); - }); - - describe('fs.promises.copyFile', function () { - it('copies a normal file', async function () { - const p = path.join(asarDir, 'a.asar', 'file1'); - const dest = temp.path(); - await fs.promises.copyFile(p, dest); - expect(fs.readFileSync(p).equals(fs.readFileSync(dest))).to.be.true(); - }); - - it('copies a unpacked file', async function () { - const p = path.join(asarDir, 'unpack.asar', 'a.txt'); - const dest = temp.path(); - await fs.promises.copyFile(p, dest); - expect(fs.readFileSync(p).equals(fs.readFileSync(dest))).to.be.true(); - }); - }); - - describe('fs.copyFileSync', function () { - it('copies a normal file', function () { - const p = path.join(asarDir, 'a.asar', 'file1'); - const dest = temp.path(); - fs.copyFileSync(p, dest); - expect(fs.readFileSync(p).equals(fs.readFileSync(dest))).to.be.true(); - }); - - it('copies a unpacked file', function () { - const p = path.join(asarDir, 'unpack.asar', 'a.txt'); - const dest = temp.path(); - fs.copyFileSync(p, dest); - expect(fs.readFileSync(p).equals(fs.readFileSync(dest))).to.be.true(); - }); - }); - - describe('fs.lstatSync', function () { - it('handles path with trailing slash correctly', function () { - const p = path.join(asarDir, 'a.asar', 'link2', 'link2', 'file1'); - fs.lstatSync(p); - fs.lstatSync(p + '/'); - }); - - it('returns information of root', function () { - const p = path.join(asarDir, 'a.asar'); - const stats = fs.lstatSync(p); - expect(stats.isFile()).to.be.false(); - expect(stats.isDirectory()).to.be.true(); - expect(stats.isSymbolicLink()).to.be.false(); - expect(stats.size).to.equal(0); - }); - - it('returns information of root with stats as bigint', function () { - const p = path.join(asarDir, 'a.asar'); - const stats = fs.lstatSync(p, { bigint: false }); - expect(stats.isFile()).to.be.false(); - expect(stats.isDirectory()).to.be.true(); - expect(stats.isSymbolicLink()).to.be.false(); - expect(stats.size).to.equal(0); - }); - - it('returns information of a normal file', function () { - const ref2 = ['file1', 'file2', 'file3', path.join('dir1', 'file1'), path.join('link2', 'file1')]; - for (let j = 0, len = ref2.length; j < len; j++) { - const file = ref2[j]; - const p = path.join(asarDir, 'a.asar', file); - const stats = fs.lstatSync(p); - expect(stats.isFile()).to.be.true(); - expect(stats.isDirectory()).to.be.false(); - expect(stats.isSymbolicLink()).to.be.false(); - expect(stats.size).to.equal(6); - } - }); - - it('returns information of a normal directory', function () { - const ref2 = ['dir1', 'dir2', 'dir3']; - for (let j = 0, len = ref2.length; j < len; j++) { - const file = ref2[j]; - const p = path.join(asarDir, 'a.asar', file); - const stats = fs.lstatSync(p); - expect(stats.isFile()).to.be.false(); - expect(stats.isDirectory()).to.be.true(); - expect(stats.isSymbolicLink()).to.be.false(); - expect(stats.size).to.equal(0); - } - }); - - it('returns information of a linked file', function () { - const ref2 = ['link1', path.join('dir1', 'link1'), path.join('link2', 'link2')]; - for (let j = 0, len = ref2.length; j < len; j++) { - const file = ref2[j]; - const p = path.join(asarDir, 'a.asar', file); - const stats = fs.lstatSync(p); - expect(stats.isFile()).to.be.false(); - expect(stats.isDirectory()).to.be.false(); - expect(stats.isSymbolicLink()).to.be.true(); - expect(stats.size).to.equal(0); - } - }); - - it('returns information of a linked directory', function () { - const ref2 = ['link2', path.join('dir1', 'link2'), path.join('link2', 'link2')]; - for (let j = 0, len = ref2.length; j < len; j++) { - const file = ref2[j]; - const p = path.join(asarDir, 'a.asar', file); - const stats = fs.lstatSync(p); - expect(stats.isFile()).to.be.false(); - expect(stats.isDirectory()).to.be.false(); - expect(stats.isSymbolicLink()).to.be.true(); - expect(stats.size).to.equal(0); - } - }); - - it('throws ENOENT error when can not find file', function () { - const ref2 = ['file4', 'file5', path.join('dir1', 'file4')]; - for (let j = 0, len = ref2.length; j < len; j++) { - const file = ref2[j]; - const p = path.join(asarDir, 'a.asar', file); - expect(() => { - fs.lstatSync(p); - }).to.throw(/ENOENT/); - } - }); - }); - - describe('fs.lstat', function () { - it('handles path with trailing slash correctly', function (done) { - const p = path.join(asarDir, 'a.asar', 'link2', 'link2', 'file1'); - fs.lstat(p + '/', done); - }); - - it('returns information of root', function (done) { - const p = path.join(asarDir, 'a.asar'); - fs.lstat(p, function (err, stats) { - try { - expect(err).to.be.null(); - expect(stats.isFile()).to.be.false(); - expect(stats.isDirectory()).to.be.true(); - expect(stats.isSymbolicLink()).to.be.false(); - expect(stats.size).to.equal(0); - done(); - } catch (e) { - done(e); - } - }); - }); - - it('returns information of root with stats as bigint', function (done) { - const p = path.join(asarDir, 'a.asar'); - fs.lstat(p, { bigint: false }, function (err, stats) { - try { - expect(err).to.be.null(); - expect(stats.isFile()).to.be.false(); - expect(stats.isDirectory()).to.be.true(); - expect(stats.isSymbolicLink()).to.be.false(); - expect(stats.size).to.equal(0); - done(); - } catch (e) { - done(e); - } - }); - }); - - it('returns information of a normal file', function (done) { - const p = path.join(asarDir, 'a.asar', 'link2', 'file1'); - fs.lstat(p, function (err, stats) { - try { - expect(err).to.be.null(); - expect(stats.isFile()).to.be.true(); - expect(stats.isDirectory()).to.be.false(); - expect(stats.isSymbolicLink()).to.be.false(); - expect(stats.size).to.equal(6); - done(); - } catch (e) { - done(e); - } - }); - }); - - it('returns information of a normal directory', function (done) { - const p = path.join(asarDir, 'a.asar', 'dir1'); - fs.lstat(p, function (err, stats) { - try { - expect(err).to.be.null(); - expect(stats.isFile()).to.be.false(); - expect(stats.isDirectory()).to.be.true(); - expect(stats.isSymbolicLink()).to.be.false(); - expect(stats.size).to.equal(0); - done(); - } catch (e) { - done(e); - } - }); - }); - - it('returns information of a linked file', function (done) { - const p = path.join(asarDir, 'a.asar', 'link2', 'link1'); - fs.lstat(p, function (err, stats) { - try { - expect(err).to.be.null(); - expect(stats.isFile()).to.be.false(); - expect(stats.isDirectory()).to.be.false(); - expect(stats.isSymbolicLink()).to.be.true(); - expect(stats.size).to.equal(0); - done(); - } catch (e) { - done(e); - } - }); - }); - - it('returns information of a linked directory', function (done) { - const p = path.join(asarDir, 'a.asar', 'link2', 'link2'); - fs.lstat(p, function (err, stats) { - try { - expect(err).to.be.null(); - expect(stats.isFile()).to.be.false(); - expect(stats.isDirectory()).to.be.false(); - expect(stats.isSymbolicLink()).to.be.true(); - expect(stats.size).to.equal(0); - done(); - } catch (e) { - done(e); - } - }); - }); - - it('throws ENOENT error when can not find file', function (done) { - const p = path.join(asarDir, 'a.asar', 'file4'); - fs.lstat(p, function (err) { - try { - expect(err.code).to.equal('ENOENT'); - done(); - } catch (e) { - done(e); - } - }); - }); - }); - - describe('fs.promises.lstat', function () { - it('handles path with trailing slash correctly', async function () { - const p = path.join(asarDir, 'a.asar', 'link2', 'link2', 'file1'); - await fs.promises.lstat(p + '/'); - }); - - it('returns information of root', async function () { - const p = path.join(asarDir, 'a.asar'); - const stats = await fs.promises.lstat(p); - expect(stats.isFile()).to.be.false(); - expect(stats.isDirectory()).to.be.true(); - expect(stats.isSymbolicLink()).to.be.false(); - expect(stats.size).to.equal(0); - }); - - it('returns information of root with stats as bigint', async function () { - const p = path.join(asarDir, 'a.asar'); - const stats = await fs.promises.lstat(p, { bigint: false }); - expect(stats.isFile()).to.be.false(); - expect(stats.isDirectory()).to.be.true(); - expect(stats.isSymbolicLink()).to.be.false(); - expect(stats.size).to.equal(0); - }); - - it('returns information of a normal file', async function () { - const p = path.join(asarDir, 'a.asar', 'link2', 'file1'); - const stats = await fs.promises.lstat(p); - expect(stats.isFile()).to.be.true(); - expect(stats.isDirectory()).to.be.false(); - expect(stats.isSymbolicLink()).to.be.false(); - expect(stats.size).to.equal(6); - }); - - it('returns information of a normal directory', async function () { - const p = path.join(asarDir, 'a.asar', 'dir1'); - const stats = await fs.promises.lstat(p); - expect(stats.isFile()).to.be.false(); - expect(stats.isDirectory()).to.be.true(); - expect(stats.isSymbolicLink()).to.be.false(); - expect(stats.size).to.equal(0); - }); - - it('returns information of a linked file', async function () { - const p = path.join(asarDir, 'a.asar', 'link2', 'link1'); - const stats = await fs.promises.lstat(p); - expect(stats.isFile()).to.be.false(); - expect(stats.isDirectory()).to.be.false(); - expect(stats.isSymbolicLink()).to.be.true(); - expect(stats.size).to.equal(0); - }); - - it('returns information of a linked directory', async function () { - const p = path.join(asarDir, 'a.asar', 'link2', 'link2'); - const stats = await fs.promises.lstat(p); - expect(stats.isFile()).to.be.false(); - expect(stats.isDirectory()).to.be.false(); - expect(stats.isSymbolicLink()).to.be.true(); - expect(stats.size).to.equal(0); - }); - - it('throws ENOENT error when can not find file', async function () { - const p = path.join(asarDir, 'a.asar', 'file4'); - await expectToThrowErrorWithCode(() => fs.promises.lstat(p), 'ENOENT'); - }); - }); - - describe('fs.realpathSync', () => { - it('returns real path root', () => { - const parent = fs.realpathSync(asarDir); - const p = 'a.asar'; - const r = fs.realpathSync(path.join(parent, p)); - expect(r).to.equal(path.join(parent, p)); - }); - - it('returns real path of a normal file', () => { - const parent = fs.realpathSync(asarDir); - const p = path.join('a.asar', 'file1'); - const r = fs.realpathSync(path.join(parent, p)); - expect(r).to.equal(path.join(parent, p)); - }); - - it('returns real path of a normal directory', () => { - const parent = fs.realpathSync(asarDir); - const p = path.join('a.asar', 'dir1'); - const r = fs.realpathSync(path.join(parent, p)); - expect(r).to.equal(path.join(parent, p)); - }); - - it('returns real path of a linked file', () => { - const parent = fs.realpathSync(asarDir); - const p = path.join('a.asar', 'link2', 'link1'); - const r = fs.realpathSync(path.join(parent, p)); - expect(r).to.equal(path.join(parent, 'a.asar', 'file1')); - }); - - it('returns real path of a linked directory', () => { - const parent = fs.realpathSync(asarDir); - const p = path.join('a.asar', 'link2', 'link2'); - const r = fs.realpathSync(path.join(parent, p)); - expect(r).to.equal(path.join(parent, 'a.asar', 'dir1')); - }); - - it('returns real path of an unpacked file', () => { - const parent = fs.realpathSync(asarDir); - const p = path.join('unpack.asar', 'a.txt'); - const r = fs.realpathSync(path.join(parent, p)); - expect(r).to.equal(path.join(parent, p)); - }); - - it('throws ENOENT error when can not find file', () => { - const parent = fs.realpathSync(asarDir); - const p = path.join('a.asar', 'not-exist'); - expect(() => { - fs.realpathSync(path.join(parent, p)); - }).to.throw(/ENOENT/); - }); - }); - - describe('fs.realpathSync.native', () => { - it('returns real path root', () => { - const parent = fs.realpathSync.native(asarDir); - const p = 'a.asar'; - const r = fs.realpathSync.native(path.join(parent, p)); - expect(r).to.equal(path.join(parent, p)); - }); - - it('returns real path of a normal file', () => { - const parent = fs.realpathSync.native(asarDir); - const p = path.join('a.asar', 'file1'); - const r = fs.realpathSync.native(path.join(parent, p)); - expect(r).to.equal(path.join(parent, p)); - }); - - it('returns real path of a normal directory', () => { - const parent = fs.realpathSync.native(asarDir); - const p = path.join('a.asar', 'dir1'); - const r = fs.realpathSync.native(path.join(parent, p)); - expect(r).to.equal(path.join(parent, p)); - }); - - it('returns real path of a linked file', () => { - const parent = fs.realpathSync.native(asarDir); - const p = path.join('a.asar', 'link2', 'link1'); - const r = fs.realpathSync.native(path.join(parent, p)); - expect(r).to.equal(path.join(parent, 'a.asar', 'file1')); - }); - - it('returns real path of a linked directory', () => { - const parent = fs.realpathSync.native(asarDir); - const p = path.join('a.asar', 'link2', 'link2'); - const r = fs.realpathSync.native(path.join(parent, p)); - expect(r).to.equal(path.join(parent, 'a.asar', 'dir1')); - }); - - it('returns real path of an unpacked file', () => { - const parent = fs.realpathSync.native(asarDir); - const p = path.join('unpack.asar', 'a.txt'); - const r = fs.realpathSync.native(path.join(parent, p)); - expect(r).to.equal(path.join(parent, p)); - }); - - it('throws ENOENT error when can not find file', () => { - const parent = fs.realpathSync.native(asarDir); - const p = path.join('a.asar', 'not-exist'); - expect(() => { - fs.realpathSync.native(path.join(parent, p)); - }).to.throw(/ENOENT/); - }); - }); - - describe('fs.realpath', () => { - it('returns real path root', done => { - const parent = fs.realpathSync(asarDir); - const p = 'a.asar'; - fs.realpath(path.join(parent, p), (err, r) => { - try { - expect(err).to.be.null(); - expect(r).to.equal(path.join(parent, p)); - done(); - } catch (e) { - done(e); - } - }); - }); - - it('returns real path of a normal file', done => { - const parent = fs.realpathSync(asarDir); - const p = path.join('a.asar', 'file1'); - fs.realpath(path.join(parent, p), (err, r) => { - try { - expect(err).to.be.null(); - expect(r).to.equal(path.join(parent, p)); - done(); - } catch (e) { - done(e); - } - }); - }); - - it('returns real path of a normal directory', done => { - const parent = fs.realpathSync(asarDir); - const p = path.join('a.asar', 'dir1'); - fs.realpath(path.join(parent, p), (err, r) => { - try { - expect(err).to.be.null(); - expect(r).to.equal(path.join(parent, p)); - done(); - } catch (e) { - done(e); - } - }); - }); - - it('returns real path of a linked file', done => { - const parent = fs.realpathSync(asarDir); - const p = path.join('a.asar', 'link2', 'link1'); - fs.realpath(path.join(parent, p), (err, r) => { - try { - expect(err).to.be.null(); - expect(r).to.equal(path.join(parent, 'a.asar', 'file1')); - done(); - } catch (e) { - done(e); - } - }); - }); - - it('returns real path of a linked directory', done => { - const parent = fs.realpathSync(asarDir); - const p = path.join('a.asar', 'link2', 'link2'); - fs.realpath(path.join(parent, p), (err, r) => { - try { - expect(err).to.be.null(); - expect(r).to.equal(path.join(parent, 'a.asar', 'dir1')); - done(); - } catch (e) { - done(e); - } - }); - }); - - it('returns real path of an unpacked file', done => { - const parent = fs.realpathSync(asarDir); - const p = path.join('unpack.asar', 'a.txt'); - fs.realpath(path.join(parent, p), (err, r) => { - try { - expect(err).to.be.null(); - expect(r).to.equal(path.join(parent, p)); - done(); - } catch (e) { - done(e); - } - }); - }); - - it('throws ENOENT error when can not find file', done => { - const parent = fs.realpathSync(asarDir); - const p = path.join('a.asar', 'not-exist'); - fs.realpath(path.join(parent, p), err => { - try { - expect(err.code).to.equal('ENOENT'); - done(); - } catch (e) { - done(e); - } - }); - }); - }); - - describe('fs.promises.realpath', () => { - it('returns real path root', async () => { - const parent = fs.realpathSync(asarDir); - const p = 'a.asar'; - const r = await fs.promises.realpath(path.join(parent, p)); - expect(r).to.equal(path.join(parent, p)); - }); - - it('returns real path of a normal file', async () => { - const parent = fs.realpathSync(asarDir); - const p = path.join('a.asar', 'file1'); - const r = await fs.promises.realpath(path.join(parent, p)); - expect(r).to.equal(path.join(parent, p)); - }); - - it('returns real path of a normal directory', async () => { - const parent = fs.realpathSync(asarDir); - const p = path.join('a.asar', 'dir1'); - const r = await fs.promises.realpath(path.join(parent, p)); - expect(r).to.equal(path.join(parent, p)); - }); - - it('returns real path of a linked file', async () => { - const parent = fs.realpathSync(asarDir); - const p = path.join('a.asar', 'link2', 'link1'); - const r = await fs.promises.realpath(path.join(parent, p)); - expect(r).to.equal(path.join(parent, 'a.asar', 'file1')); - }); - - it('returns real path of a linked directory', async () => { - const parent = fs.realpathSync(asarDir); - const p = path.join('a.asar', 'link2', 'link2'); - const r = await fs.promises.realpath(path.join(parent, p)); - expect(r).to.equal(path.join(parent, 'a.asar', 'dir1')); - }); - - it('returns real path of an unpacked file', async () => { - const parent = fs.realpathSync(asarDir); - const p = path.join('unpack.asar', 'a.txt'); - const r = await fs.promises.realpath(path.join(parent, p)); - expect(r).to.equal(path.join(parent, p)); - }); - - it('throws ENOENT error when can not find file', async () => { - const parent = fs.realpathSync(asarDir); - const p = path.join('a.asar', 'not-exist'); - await expectToThrowErrorWithCode(() => fs.promises.realpath(path.join(parent, p)), 'ENOENT'); - }); - }); - - describe('fs.realpath.native', () => { - it('returns real path root', done => { - const parent = fs.realpathSync.native(asarDir); - const p = 'a.asar'; - fs.realpath.native(path.join(parent, p), (err, r) => { - try { - expect(err).to.be.null(); - expect(r).to.equal(path.join(parent, p)); - done(); - } catch (e) { - done(e); - } - }); - }); - - it('returns real path of a normal file', done => { - const parent = fs.realpathSync.native(asarDir); - const p = path.join('a.asar', 'file1'); - fs.realpath.native(path.join(parent, p), (err, r) => { - try { - expect(err).to.be.null(); - expect(r).to.equal(path.join(parent, p)); - done(); - } catch (e) { - done(e); - } - }); - }); - - it('returns real path of a normal directory', done => { - const parent = fs.realpathSync.native(asarDir); - const p = path.join('a.asar', 'dir1'); - fs.realpath.native(path.join(parent, p), (err, r) => { - try { - expect(err).to.be.null(); - expect(r).to.equal(path.join(parent, p)); - done(); - } catch (e) { - done(e); - } - }); - }); - - it('returns real path of a linked file', done => { - const parent = fs.realpathSync.native(asarDir); - const p = path.join('a.asar', 'link2', 'link1'); - fs.realpath.native(path.join(parent, p), (err, r) => { - try { - expect(err).to.be.null(); - expect(r).to.equal(path.join(parent, 'a.asar', 'file1')); - done(); - } catch (e) { - done(e); - } - }); - }); - - it('returns real path of a linked directory', done => { - const parent = fs.realpathSync.native(asarDir); - const p = path.join('a.asar', 'link2', 'link2'); - fs.realpath.native(path.join(parent, p), (err, r) => { - try { - expect(err).to.be.null(); - expect(r).to.equal(path.join(parent, 'a.asar', 'dir1')); - done(); - } catch (e) { - done(e); - } - }); - }); - - it('returns real path of an unpacked file', done => { - const parent = fs.realpathSync.native(asarDir); - const p = path.join('unpack.asar', 'a.txt'); - fs.realpath.native(path.join(parent, p), (err, r) => { - try { - expect(err).to.be.null(); - expect(r).to.equal(path.join(parent, p)); - done(); - } catch (e) { - done(e); - } - }); - }); - - it('throws ENOENT error when can not find file', done => { - const parent = fs.realpathSync.native(asarDir); - const p = path.join('a.asar', 'not-exist'); - fs.realpath.native(path.join(parent, p), err => { - try { - expect(err.code).to.equal('ENOENT'); - done(); - } catch (e) { - done(e); - } - }); - }); - }); - - describe('fs.readdirSync', function () { - it('reads dirs from root', function () { - const p = path.join(asarDir, 'a.asar'); - const dirs = fs.readdirSync(p); - expect(dirs).to.deep.equal(['dir1', 'dir2', 'dir3', 'file1', 'file2', 'file3', 'link1', 'link2', 'ping.js']); - }); - - it('reads dirs from a normal dir', function () { - const p = path.join(asarDir, 'a.asar', 'dir1'); - const dirs = fs.readdirSync(p); - expect(dirs).to.deep.equal(['file1', 'file2', 'file3', 'link1', 'link2']); - }); - - it('supports withFileTypes', function () { - const p = path.join(asarDir, 'a.asar'); - const dirs = fs.readdirSync(p, { withFileTypes: true }); - for (const dir of dirs) { - expect(dir instanceof fs.Dirent).to.be.true(); - } - const names = dirs.map(a => a.name); - expect(names).to.deep.equal(['dir1', 'dir2', 'dir3', 'file1', 'file2', 'file3', 'link1', 'link2', 'ping.js']); - }); - - it('supports withFileTypes for a deep directory', function () { - const p = path.join(asarDir, 'a.asar', 'dir3'); - const dirs = fs.readdirSync(p, { withFileTypes: true }); - for (const dir of dirs) { - expect(dir instanceof fs.Dirent).to.be.true(); - } - const names = dirs.map(a => a.name); - expect(names).to.deep.equal(['file1', 'file2', 'file3']); - }); - - it('reads dirs from a linked dir', function () { - const p = path.join(asarDir, 'a.asar', 'link2', 'link2'); - const dirs = fs.readdirSync(p); - expect(dirs).to.deep.equal(['file1', 'file2', 'file3', 'link1', 'link2']); - }); - - it('throws ENOENT error when can not find file', function () { - const p = path.join(asarDir, 'a.asar', 'not-exist'); - expect(() => { - fs.readdirSync(p); - }).to.throw(/ENOENT/); - }); - }); - - describe('fs.readdir', function () { - it('reads dirs from root', function (done) { - const p = path.join(asarDir, 'a.asar'); - fs.readdir(p, function (err, dirs) { - try { - expect(err).to.be.null(); - expect(dirs).to.deep.equal(['dir1', 'dir2', 'dir3', 'file1', 'file2', 'file3', 'link1', 'link2', 'ping.js']); - done(); - } catch (e) { - done(e); - } - }); - }); - - it('supports withFileTypes', function (done) { - const p = path.join(asarDir, 'a.asar'); - - fs.readdir(p, { withFileTypes: true }, (err, dirs) => { - try { - expect(err).to.be.null(); - for (const dir of dirs) { - expect(dir instanceof fs.Dirent).to.be.true(); - } - - const names = dirs.map(a => a.name); - expect(names).to.deep.equal(['dir1', 'dir2', 'dir3', 'file1', 'file2', 'file3', 'link1', 'link2', 'ping.js']); - done(); - } catch (e) { - done(e); - } - }); - }); - - it('reads dirs from a normal dir', function (done) { - const p = path.join(asarDir, 'a.asar', 'dir1'); - fs.readdir(p, function (err, dirs) { - try { - expect(err).to.be.null(); - expect(dirs).to.deep.equal(['file1', 'file2', 'file3', 'link1', 'link2']); - done(); - } catch (e) { - done(e); - } - }); - }); - - it('reads dirs from a linked dir', function (done) { - const p = path.join(asarDir, 'a.asar', 'link2', 'link2'); - fs.readdir(p, function (err, dirs) { - try { - expect(err).to.be.null(); - expect(dirs).to.deep.equal(['file1', 'file2', 'file3', 'link1', 'link2']); - done(); - } catch (e) { - done(e); - } - }); - }); - - it('throws ENOENT error when can not find file', function (done) { - const p = path.join(asarDir, 'a.asar', 'not-exist'); - fs.readdir(p, function (err) { - try { - expect(err.code).to.equal('ENOENT'); - done(); - } catch (e) { - done(e); - } - }); - }); - }); - - describe('fs.promises.readdir', function () { - it('reads dirs from root', async function () { - const p = path.join(asarDir, 'a.asar'); - const dirs = await fs.promises.readdir(p); - expect(dirs).to.deep.equal(['dir1', 'dir2', 'dir3', 'file1', 'file2', 'file3', 'link1', 'link2', 'ping.js']); - }); - - it('supports withFileTypes', async function () { - const p = path.join(asarDir, 'a.asar'); - const dirs = await fs.promises.readdir(p, { withFileTypes: true }); - for (const dir of dirs) { - expect(dir instanceof fs.Dirent).to.be.true(); - } - const names = dirs.map(a => a.name); - expect(names).to.deep.equal(['dir1', 'dir2', 'dir3', 'file1', 'file2', 'file3', 'link1', 'link2', 'ping.js']); - }); - - it('reads dirs from a normal dir', async function () { - const p = path.join(asarDir, 'a.asar', 'dir1'); - const dirs = await fs.promises.readdir(p); - expect(dirs).to.deep.equal(['file1', 'file2', 'file3', 'link1', 'link2']); - }); - - it('reads dirs from a linked dir', async function () { - const p = path.join(asarDir, 'a.asar', 'link2', 'link2'); - const dirs = await fs.promises.readdir(p); - expect(dirs).to.deep.equal(['file1', 'file2', 'file3', 'link1', 'link2']); - }); - - it('throws ENOENT error when can not find file', async function () { - const p = path.join(asarDir, 'a.asar', 'not-exist'); - await expectToThrowErrorWithCode(() => fs.promises.readdir(p), 'ENOENT'); - }); - }); - - describe('fs.openSync', function () { - it('opens a normal/linked/under-linked-directory file', function () { - const ref2 = ['file1', 'link1', path.join('link2', 'file1')]; - for (let j = 0, len = ref2.length; j < len; j++) { - const file = ref2[j]; - const p = path.join(asarDir, 'a.asar', file); - const fd = fs.openSync(p, 'r'); - const buffer = Buffer.alloc(6); - fs.readSync(fd, buffer, 0, 6, 0); - expect(String(buffer).trim()).to.equal('file1'); - fs.closeSync(fd); - } - }); - - it('throws ENOENT error when can not find file', function () { - const p = path.join(asarDir, 'a.asar', 'not-exist'); - expect(() => { - fs.openSync(p); - }).to.throw(/ENOENT/); - }); - }); - - describe('fs.open', function () { - it('opens a normal file', function (done) { - const p = path.join(asarDir, 'a.asar', 'file1'); - fs.open(p, 'r', function (err, fd) { - expect(err).to.be.null(); - const buffer = Buffer.alloc(6); - fs.read(fd, buffer, 0, 6, 0, function (err) { - expect(err).to.be.null(); - expect(String(buffer).trim()).to.equal('file1'); - fs.close(fd, done); - }); - }); - }); - - it('throws ENOENT error when can not find file', function (done) { - const p = path.join(asarDir, 'a.asar', 'not-exist'); - fs.open(p, 'r', function (err) { - try { - expect(err.code).to.equal('ENOENT'); - done(); - } catch (e) { - done(e); - } - }); - }); - }); - - describe('fs.promises.open', function () { - it('opens a normal file', async function () { - const p = path.join(asarDir, 'a.asar', 'file1'); - const fh = await fs.promises.open(p, 'r'); - const buffer = Buffer.alloc(6); - await fh.read(buffer, 0, 6, 0); - expect(String(buffer).trim()).to.equal('file1'); - await fh.close(); - }); - - it('throws ENOENT error when can not find file', async function () { - const p = path.join(asarDir, 'a.asar', 'not-exist'); - await expectToThrowErrorWithCode(() => fs.promises.open(p, 'r'), 'ENOENT'); - }); - }); - - describe('fs.mkdir', function () { - it('throws error when calling inside asar archive', function (done) { - const p = path.join(asarDir, 'a.asar', 'not-exist'); - fs.mkdir(p, function (err) { - try { - expect(err.code).to.equal('ENOTDIR'); - done(); - } catch (e) { - done(e); - } - }); - }); - }); - - describe('fs.promises.mkdir', function () { - it('throws error when calling inside asar archive', async function () { - const p = path.join(asarDir, 'a.asar', 'not-exist'); - await expectToThrowErrorWithCode(() => fs.promises.mkdir(p), 'ENOTDIR'); - }); - }); - - describe('fs.mkdirSync', function () { - it('throws error when calling inside asar archive', function () { - const p = path.join(asarDir, 'a.asar', 'not-exist'); - expect(() => { - fs.mkdirSync(p); - }).to.throw(/ENOTDIR/); - }); - }); - - describe('fs.exists', function () { - it('handles an existing file', function (done) { - const p = path.join(asarDir, 'a.asar', 'file1'); - // eslint-disable-next-line - fs.exists(p, function (exists) { - try { - expect(exists).to.be.true(); - done(); - } catch (e) { - done(e); - } - }); - }); - - it('handles a non-existent file', function (done) { - const p = path.join(asarDir, 'a.asar', 'not-exist'); - // eslint-disable-next-line - fs.exists(p, function (exists) { - try { - expect(exists).to.be.false(); - done(); - } catch (e) { - done(e); - } - }); - }); - - it('promisified version handles an existing file', (done) => { - const p = path.join(asarDir, 'a.asar', 'file1'); - // eslint-disable-next-line - util.promisify(fs.exists)(p).then(exists => { - try { - expect(exists).to.be.true(); - done(); - } catch (e) { - done(e); - } - }); - }); - - it('promisified version handles a non-existent file', function (done) { - const p = path.join(asarDir, 'a.asar', 'not-exist'); - // eslint-disable-next-line - util.promisify(fs.exists)(p).then(exists => { - try { - expect(exists).to.be.false(); - done(); - } catch (e) { - done(e); - } - }); - }); - }); - - describe('fs.existsSync', function () { - it('handles an existing file', function () { - const p = path.join(asarDir, 'a.asar', 'file1'); - expect(fs.existsSync(p)).to.be.true(); - }); - - it('handles a non-existent file', function () { - const p = path.join(asarDir, 'a.asar', 'not-exist'); - expect(fs.existsSync(p)).to.be.false(); - }); - }); - - describe('fs.access', function () { - it('accesses a normal file', function (done) { - const p = path.join(asarDir, 'a.asar', 'file1'); - fs.access(p, function (err) { - try { - expect(err).to.be.undefined(); - done(); - } catch (e) { - done(e); - } - }); - }); - - it('throws an error when called with write mode', function (done) { - const p = path.join(asarDir, 'a.asar', 'file1'); - fs.access(p, fs.constants.R_OK | fs.constants.W_OK, function (err) { - try { - expect(err.code).to.equal('EACCES'); - done(); - } catch (e) { - done(e); - } - }); - }); - - it('throws an error when called on non-existent file', function (done) { - const p = path.join(asarDir, 'a.asar', 'not-exist'); - fs.access(p, function (err) { - try { - expect(err.code).to.equal('ENOENT'); - done(); - } catch (e) { - done(e); - } - }); - }); - - it('allows write mode for unpacked files', function (done) { - const p = path.join(asarDir, 'unpack.asar', 'a.txt'); - fs.access(p, fs.constants.R_OK | fs.constants.W_OK, function (err) { - try { - expect(err).to.be.null(); - done(); - } catch (e) { - done(e); - } - }); - }); - }); - - describe('fs.promises.access', function () { - it('accesses a normal file', async function () { - const p = path.join(asarDir, 'a.asar', 'file1'); - await fs.promises.access(p); - }); - - it('throws an error when called with write mode', async function () { - const p = path.join(asarDir, 'a.asar', 'file1'); - await expectToThrowErrorWithCode(() => fs.promises.access(p, fs.constants.R_OK | fs.constants.W_OK), 'EACCES'); - }); - - it('throws an error when called on non-existent file', async function () { - const p = path.join(asarDir, 'a.asar', 'not-exist'); - await expectToThrowErrorWithCode(() => fs.promises.access(p), 'ENOENT'); - }); - - it('allows write mode for unpacked files', async function () { - const p = path.join(asarDir, 'unpack.asar', 'a.txt'); - await fs.promises.access(p, fs.constants.R_OK | fs.constants.W_OK); - }); - }); - - describe('fs.accessSync', function () { - it('accesses a normal file', function () { - const p = path.join(asarDir, 'a.asar', 'file1'); - expect(() => { - fs.accessSync(p); - }).to.not.throw(); - }); - - it('throws an error when called with write mode', function () { - const p = path.join(asarDir, 'a.asar', 'file1'); - expect(() => { - fs.accessSync(p, fs.constants.R_OK | fs.constants.W_OK); - }).to.throw(/EACCES/); - }); - - it('throws an error when called on non-existent file', function () { - const p = path.join(asarDir, 'a.asar', 'not-exist'); - expect(() => { - fs.accessSync(p); - }).to.throw(/ENOENT/); - }); - - it('allows write mode for unpacked files', function () { - const p = path.join(asarDir, 'unpack.asar', 'a.txt'); - expect(() => { - fs.accessSync(p, fs.constants.R_OK | fs.constants.W_OK); - }).to.not.throw(); - }); - }); - - describe('child_process.fork', function () { - before(function () { - if (!features.isRunAsNodeEnabled()) { - this.skip(); - } - }); - - it('opens a normal js file', function (done) { - const child = ChildProcess.fork(path.join(asarDir, 'a.asar', 'ping.js')); - child.on('message', function (msg) { - try { - expect(msg).to.equal('message'); - done(); - } catch (e) { - done(e); - } - }); - child.send('message'); - }); - - it('supports asar in the forked js', function (done) { - const file = path.join(asarDir, 'a.asar', 'file1'); - const child = ChildProcess.fork(path.join(fixtures, 'module', 'asar.js')); - child.on('message', function (content) { - try { - expect(content).to.equal(fs.readFileSync(file).toString()); - done(); - } catch (e) { - done(e); - } - }); - child.send(file); - }); - }); - - describe('child_process.exec', function () { - const echo = path.join(asarDir, 'echo.asar', 'echo'); - - it('should not try to extract the command if there is a reference to a file inside an .asar', function (done) { - ChildProcess.exec('echo ' + echo + ' foo bar', function (error, stdout) { - try { - expect(error).to.be.null(); - expect(stdout.toString().replace(/\r/g, '')).to.equal(echo + ' foo bar\n'); - done(); - } catch (e) { - done(e); - } - }); - }); - - it('can be promisified', () => { - return util.promisify(ChildProcess.exec)('echo ' + echo + ' foo bar').then(({ stdout }) => { - expect(stdout.toString().replace(/\r/g, '')).to.equal(echo + ' foo bar\n'); - }); - }); - }); - - describe('child_process.execSync', function () { - const echo = path.join(asarDir, 'echo.asar', 'echo'); - - it('should not try to extract the command if there is a reference to a file inside an .asar', function (done) { - try { - const stdout = ChildProcess.execSync('echo ' + echo + ' foo bar'); - expect(stdout.toString().replace(/\r/g, '')).to.equal(echo + ' foo bar\n'); - done(); - } catch (e) { - done(e); - } - }); - }); - - ifdescribe(process.platform === 'darwin' && process.arch !== 'arm64')('child_process.execFile', function () { - const execFile = ChildProcess.execFile; - const execFileSync = ChildProcess.execFileSync; - const echo = path.join(asarDir, 'echo.asar', 'echo'); - - it('executes binaries', function (done) { - execFile(echo, ['test'], function (error, stdout) { - try { - expect(error).to.be.null(); - expect(stdout).to.equal('test\n'); - done(); - } catch (e) { - done(e); - } - }); - }); - - it('executes binaries without callback', function (done) { - const process = execFile(echo, ['test']); - process.on('close', function (code) { - try { - expect(code).to.equal(0); - done(); - } catch (e) { - done(e); - } - }); - process.on('error', function () { - done('error'); - }); - }); - - it('execFileSync executes binaries', function () { - const output = execFileSync(echo, ['test']); - expect(String(output)).to.equal('test\n'); - }); - - it('can be promisified', () => { - return util.promisify(ChildProcess.execFile)(echo, ['test']).then(({ stdout }) => { - expect(stdout).to.equal('test\n'); - }); - }); - }); - - describe('internalModuleReadJSON', function () { - const { internalModuleReadJSON } = process.binding('fs'); - - it('reads a normal file', function () { - const file1 = path.join(asarDir, 'a.asar', 'file1'); - const [s1, c1] = internalModuleReadJSON(file1); - expect([s1.toString().trim(), c1]).to.eql(['file1', true]); - - const file2 = path.join(asarDir, 'a.asar', 'file2'); - const [s2, c2] = internalModuleReadJSON(file2); - expect([s2.toString().trim(), c2]).to.eql(['file2', true]); - - const file3 = path.join(asarDir, 'a.asar', 'file3'); - const [s3, c3] = internalModuleReadJSON(file3); - expect([s3.toString().trim(), c3]).to.eql(['file3', true]); - }); - - it('reads a normal file with unpacked files', function () { - const p = path.join(asarDir, 'unpack.asar', 'a.txt'); - const [s, c] = internalModuleReadJSON(p); - expect([s.toString().trim(), c]).to.eql(['a', true]); - }); - }); - - describe('util.promisify', function () { - it('can promisify all fs functions', function () { - const originalFs = require('original-fs'); - const { hasOwnProperty } = Object.prototype; - - for (const [propertyName, originalValue] of Object.entries(originalFs)) { - // Some properties exist but have a value of `undefined` on some platforms. - // E.g. `fs.lchmod`, which in only available on MacOS, see - // https://nodejs.org/docs/latest-v10.x/api/fs.html#fs_fs_lchmod_path_mode_callback - // Also check for `null`s, `hasOwnProperty()` can't handle them. - if (typeof originalValue === 'undefined' || originalValue === null) continue; - - if (hasOwnProperty.call(originalValue, util.promisify.custom)) { - expect(fs).to.have.own.property(propertyName) - .that.has.own.property(util.promisify.custom); - } - } - }); - }); - - describe('process.noAsar', function () { - const errorName = process.platform === 'win32' ? 'ENOENT' : 'ENOTDIR'; - - beforeEach(function () { - process.noAsar = true; - }); - - afterEach(function () { - process.noAsar = false; - }); - - it('disables asar support in sync API', function () { - const file = path.join(asarDir, 'a.asar', 'file1'); - const dir = path.join(asarDir, 'a.asar', 'dir1'); - expect(() => { - fs.readFileSync(file); - }).to.throw(new RegExp(errorName)); - expect(() => { - fs.lstatSync(file); - }).to.throw(new RegExp(errorName)); - expect(() => { - fs.realpathSync(file); - }).to.throw(new RegExp(errorName)); - expect(() => { - fs.readdirSync(dir); - }).to.throw(new RegExp(errorName)); - }); - - it('disables asar support in async API', function (done) { - const file = path.join(asarDir, 'a.asar', 'file1'); - const dir = path.join(asarDir, 'a.asar', 'dir1'); - fs.readFile(file, function (error) { - expect(error.code).to.equal(errorName); - fs.lstat(file, function (error) { - expect(error.code).to.equal(errorName); - fs.realpath(file, function (error) { - expect(error.code).to.equal(errorName); - fs.readdir(dir, function (error) { - expect(error.code).to.equal(errorName); - done(); - }); - }); - }); - }); - }); - - it('disables asar support in promises API', async function () { - const file = path.join(asarDir, 'a.asar', 'file1'); - const dir = path.join(asarDir, 'a.asar', 'dir1'); - await expect(fs.promises.readFile(file)).to.be.eventually.rejectedWith(Error, new RegExp(errorName)); - await expect(fs.promises.lstat(file)).to.be.eventually.rejectedWith(Error, new RegExp(errorName)); - await expect(fs.promises.realpath(file)).to.be.eventually.rejectedWith(Error, new RegExp(errorName)); - await expect(fs.promises.readdir(dir)).to.be.eventually.rejectedWith(Error, new RegExp(errorName)); - }); - - it('treats *.asar as normal file', function () { - const originalFs = require('original-fs'); - const asar = path.join(asarDir, 'a.asar'); - const content1 = fs.readFileSync(asar); - const content2 = originalFs.readFileSync(asar); - expect(content1.compare(content2)).to.equal(0); - expect(() => { - fs.readdirSync(asar); - }).to.throw(/ENOTDIR/); - }); - - it('is reset to its original value when execSync throws an error', function () { - process.noAsar = false; - expect(() => { - ChildProcess.execSync(path.join(__dirname, 'does-not-exist.txt')); - }).to.throw(); - expect(process.noAsar).to.be.false(); - }); - }); - - describe('process.env.ELECTRON_NO_ASAR', function () { - before(function () { - if (!features.isRunAsNodeEnabled()) { - this.skip(); - } - }); - - it('disables asar support in forked processes', function (done) { - const forked = ChildProcess.fork(path.join(__dirname, 'fixtures', 'module', 'no-asar.js'), [], { - env: { - ELECTRON_NO_ASAR: true - } - }); - forked.on('message', function (stats) { - try { - expect(stats.isFile).to.be.true(); - expect(stats.size).to.equal(3458); - done(); - } catch (e) { - done(e); - } - }); - }); - - it('disables asar support in spawned processes', function (done) { - const spawned = ChildProcess.spawn(process.execPath, [path.join(__dirname, 'fixtures', 'module', 'no-asar.js')], { - env: { - ELECTRON_NO_ASAR: true, - ELECTRON_RUN_AS_NODE: true - } - }); - - let output = ''; - spawned.stdout.on('data', function (data) { - output += data; - }); - spawned.stdout.on('close', function () { - try { - const stats = JSON.parse(output); - expect(stats.isFile).to.be.true(); - expect(stats.size).to.equal(3458); - done(); - } catch (e) { - done(e); - } - }); - }); - }); - }); - - describe('asar protocol', function () { - it('can request a file in package', async function () { - const p = path.resolve(asarDir, 'a.asar', 'file1'); - const response = await fetch('file://' + p); - const data = await response.text(); - expect(data.trim()).to.equal('file1'); - }); - - it('can request a file in package with unpacked files', async function () { - const p = path.resolve(asarDir, 'unpack.asar', 'a.txt'); - const response = await fetch('file://' + p); - const data = await response.text(); - expect(data.trim()).to.equal('a'); - }); - - it('can request a linked file in package', async function () { - const p = path.resolve(asarDir, 'a.asar', 'link2', 'link1'); - const response = await fetch('file://' + p); - const data = await response.text(); - expect(data.trim()).to.equal('file1'); - }); - - it('can request a file in filesystem', async function () { - const p = path.resolve(asarDir, 'file'); - const response = await fetch('file://' + p); - const data = await response.text(); - expect(data.trim()).to.equal('file'); - }); - - it('gets error when file is not found', async function () { - const p = path.resolve(asarDir, 'a.asar', 'no-exist'); - try { - const response = await fetch('file://' + p); - expect(response.status).to.equal(404); - } catch (error) { - expect(error.message).to.equal('Failed to fetch'); - } - }); - }); - - describe('original-fs module', function () { - const originalFs = require('original-fs'); - - it('treats .asar as file', function () { - const file = path.join(asarDir, 'a.asar'); - const stats = originalFs.statSync(file); - expect(stats.isFile()).to.be.true(); - }); - - ifit(features.isRunAsNodeEnabled())('is available in forked scripts', async function () { - const child = ChildProcess.fork(path.join(fixtures, 'module', 'original-fs.js')); - const message = emittedOnce(child, 'message'); - child.send('message'); - const [msg] = await message; - expect(msg).to.equal('object'); - }); - - it('can be used with streams', () => { - originalFs.createReadStream(path.join(asarDir, 'a.asar')); - }); - - it('can recursively delete a directory with an asar file in it', () => { - const deleteDir = path.join(asarDir, 'deleteme'); - fs.mkdirSync(deleteDir); - - originalFs.rmdirSync(deleteDir, { recursive: true }); - - expect(fs.existsSync(deleteDir)).to.be.false(); - }); - - it('has the same APIs as fs', function () { - expect(Object.keys(require('fs'))).to.deep.equal(Object.keys(require('original-fs'))); - expect(Object.keys(require('fs').promises)).to.deep.equal(Object.keys(require('original-fs').promises)); - }); - }); - - describe('graceful-fs module', function () { - const gfs = require('graceful-fs'); - - it('recognize asar archives', function () { - const p = path.join(asarDir, 'a.asar', 'link1'); - expect(gfs.readFileSync(p).toString().trim()).to.equal('file1'); - }); - it('does not touch global fs object', function () { - expect(fs.readdir).to.not.equal(gfs.readdir); - }); - }); - - describe('mkdirp module', function () { - const mkdirp = require('mkdirp'); - - it('throws error when calling inside asar archive', function () { - const p = path.join(asarDir, 'a.asar', 'not-exist'); - expect(() => { - mkdirp.sync(p); - }).to.throw(/ENOTDIR/); - }); - }); - - describe('native-image', function () { - it('reads image from asar archive', function () { - const p = path.join(asarDir, 'logo.asar', 'logo.png'); - const logo = nativeImage.createFromPath(p); - expect(logo.getSize()).to.deep.equal({ - width: 55, - height: 55 - }); - }); - - it('reads image from asar archive with unpacked files', function () { - const p = path.join(asarDir, 'unpack.asar', 'atom.png'); - const logo = nativeImage.createFromPath(p); - expect(logo.getSize()).to.deep.equal({ - width: 1024, - height: 1024 - }); - }); - }); -});