electron/lib/common/asar.js

599 lines
17 KiB
JavaScript
Raw Normal View History

(function () {
2016-01-19 18:25:03 +00:00
const asar = process.binding('atom_common_asar');
const child_process = require('child_process');
const path = require('path');
const util = require('util');
2016-01-12 02:40:23 +00:00
2016-01-19 18:25:03 +00:00
var hasProp = {}.hasOwnProperty;
2016-01-12 02:40:23 +00:00
2016-01-19 18:25:03 +00:00
// Cache asar archive objects.
var cachedArchives = {};
2016-01-12 02:40:23 +00:00
2016-01-19 18:25:03 +00:00
var getOrCreateArchive = function(p) {
var archive;
2016-01-12 02:40:23 +00:00
archive = cachedArchives[p];
2016-01-19 18:25:03 +00:00
if (archive != null) {
return archive;
}
archive = asar.createArchive(p);
if (!archive) {
2016-01-12 02:40:23 +00:00
return false;
}
2016-01-19 18:25:03 +00:00
return cachedArchives[p] = archive;
2016-01-12 02:40:23 +00:00
};
2016-01-19 18:25:03 +00:00
// Clean cache on quit.
process.on('exit', function() {
var archive, p;
2016-01-19 18:25:03 +00:00
for (p in cachedArchives) {
if (!hasProp.call(cachedArchives, p)) continue;
archive = cachedArchives[p];
archive.destroy();
2016-01-19 18:25:03 +00:00
}
2016-01-12 02:40:23 +00:00
});
2016-01-19 18:25:03 +00:00
// Separate asar package's path from full path.
var splitPath = function(p) {
var index;
2016-01-12 02:40:23 +00:00
2016-01-19 18:25:03 +00:00
// shortcut to disable asar.
if (process.noAsar) {
return [false];
2016-01-12 02:40:23 +00:00
}
2016-01-19 18:25:03 +00:00
if (typeof p !== 'string') {
return [false];
2016-01-12 02:40:23 +00:00
}
2016-01-19 18:25:03 +00:00
if (p.substr(-5) === '.asar') {
return [true, p, ''];
2016-01-12 02:40:23 +00:00
}
2016-01-19 18:25:03 +00:00
p = path.normalize(p);
index = p.lastIndexOf(".asar" + path.sep);
if (index === -1) {
return [false];
2016-01-12 02:40:23 +00:00
}
2016-01-19 18:25:03 +00:00
return [true, p.substr(0, index + 5), p.substr(index + 6)];
2016-01-12 02:40:23 +00:00
};
2016-01-19 18:25:03 +00:00
// Convert asar archive's Stats object to fs's Stats object.
var nextInode = 0;
var uid = process.getuid != null ? process.getuid() : 0;
var gid = process.getgid != null ? process.getgid() : 0;
var fakeTime = new Date();
var asarStatsToFsStats = function(stats) {
return {
dev: 1,
ino: ++nextInode,
mode: 33188,
nlink: 1,
uid: uid,
gid: gid,
rdev: 0,
atime: stats.atime || fakeTime,
birthtime: stats.birthtime || fakeTime,
mtime: stats.mtime || fakeTime,
ctime: stats.ctime || fakeTime,
size: stats.size,
isFile: function() {
return stats.isFile;
},
isDirectory: function() {
return stats.isDirectory;
},
isSymbolicLink: function() {
return stats.isLink;
},
isBlockDevice: function() {
return false;
},
isCharacterDevice: function() {
return false;
},
isFIFO: function() {
return false;
},
isSocket: function() {
return false;
}
};
2016-01-12 02:40:23 +00:00
};
2016-01-19 18:25:03 +00:00
// Create a ENOENT error.
var notFoundError = function(asarPath, filePath, callback) {
var error;
error = new Error("ENOENT, " + filePath + " not found in " + asarPath);
error.code = "ENOENT";
error.errno = -2;
if (typeof callback !== 'function') {
throw error;
2016-01-12 02:40:23 +00:00
}
return process.nextTick(function() {
2016-01-19 18:25:03 +00:00
return callback(error);
2016-01-12 02:40:23 +00:00
});
};
2016-01-19 18:25:03 +00:00
// Create a ENOTDIR error.
var notDirError = function(callback) {
var error;
error = new Error('ENOTDIR, not a directory');
error.code = 'ENOTDIR';
error.errno = -20;
if (typeof callback !== 'function') {
throw error;
2016-01-12 02:40:23 +00:00
}
return process.nextTick(function() {
2016-01-19 18:25:03 +00:00
return callback(error);
2016-01-12 02:40:23 +00:00
});
};
2016-01-19 18:25:03 +00:00
// Create invalid archive error.
var invalidArchiveError = function(asarPath, callback) {
var error;
error = new Error("Invalid package " + asarPath);
if (typeof callback !== 'function') {
throw error;
2016-01-12 02:40:23 +00:00
}
return process.nextTick(function() {
2016-01-19 18:25:03 +00:00
return callback(error);
2016-01-12 02:40:23 +00:00
});
};
2016-01-19 18:25:03 +00:00
// Override APIs that rely on passing file path instead of content to C++.
var overrideAPISync = function(module, name, arg) {
var old;
if (arg == null) {
arg = 0;
}
old = module[name];
return module[name] = function() {
2016-03-21 16:39:07 +00:00
var archive, newPath, p;
2016-01-19 18:25:03 +00:00
p = arguments[arg];
2016-03-21 16:39:07 +00:00
const [isAsar, asarPath, filePath] = splitPath(p);
2016-01-19 18:25:03 +00:00
if (!isAsar) {
return old.apply(this, arguments);
2016-01-12 02:40:23 +00:00
}
2016-01-19 18:25:03 +00:00
archive = getOrCreateArchive(asarPath);
if (!archive) {
invalidArchiveError(asarPath);
}
newPath = archive.copyFileOut(filePath);
if (!newPath) {
notFoundError(asarPath, filePath);
}
arguments[arg] = newPath;
return old.apply(this, arguments);
};
2016-01-12 02:40:23 +00:00
};
2016-01-19 18:25:03 +00:00
var overrideAPI = function(module, name, arg) {
var old;
if (arg == null) {
arg = 0;
}
old = module[name];
return module[name] = function() {
2016-03-21 16:39:07 +00:00
var archive, callback, newPath, p;
2016-01-19 18:25:03 +00:00
p = arguments[arg];
2016-03-21 16:39:07 +00:00
const [isAsar, asarPath, filePath] = splitPath(p);
2016-01-19 18:25:03 +00:00
if (!isAsar) {
return old.apply(this, arguments);
}
callback = arguments[arguments.length - 1];
if (typeof callback !== 'function') {
return overrideAPISync(module, name, arg);
}
archive = getOrCreateArchive(asarPath);
if (!archive) {
return invalidArchiveError(asarPath, callback);
}
newPath = archive.copyFileOut(filePath);
if (!newPath) {
return notFoundError(asarPath, filePath, callback);
}
arguments[arg] = newPath;
return old.apply(this, arguments);
};
2016-01-12 02:40:23 +00:00
};
2016-01-19 18:25:03 +00:00
// Override fs APIs.
exports.wrapFsWithAsar = function(fs) {
var exists, existsSync, internalModuleReadFile, internalModuleStat, lstat, lstatSync, mkdir, mkdirSync, readFile, readFileSync, readdir, readdirSync, realpath, realpathSync, stat, statSync, statSyncNoException, logFDs, logASARAccess;
logFDs = {};
logASARAccess = function(asarPath, filePath, offset) {
if (!process.env.ELECTRON_LOG_ASAR_READS) {
return;
}
if (!logFDs[asarPath]) {
var logFilename, logPath;
const path = require('path');
logFilename = path.basename(asarPath, '.asar') + '-access-log.txt';
logPath = path.join(require('os').tmpdir(), logFilename);
logFDs[asarPath] = fs.openSync(logPath, 'a');
console.log('Logging ' + asarPath + ' access to ' + logPath);
}
fs.writeSync(logFDs[asarPath], offset + ': ' + filePath + '\n');
};
2016-01-19 18:25:03 +00:00
lstatSync = fs.lstatSync;
fs.lstatSync = function(p) {
2016-03-21 16:39:07 +00:00
var archive, stats;
const [isAsar, asarPath, filePath] = splitPath(p);
2016-01-19 18:25:03 +00:00
if (!isAsar) {
return lstatSync(p);
}
archive = getOrCreateArchive(asarPath);
if (!archive) {
invalidArchiveError(asarPath);
}
stats = archive.stat(filePath);
if (!stats) {
notFoundError(asarPath, filePath);
}
return asarStatsToFsStats(stats);
};
lstat = fs.lstat;
fs.lstat = function(p, callback) {
2016-03-21 16:39:07 +00:00
var archive, stats;
const [isAsar, asarPath, filePath] = splitPath(p);
2016-01-19 18:25:03 +00:00
if (!isAsar) {
return lstat(p, callback);
}
archive = getOrCreateArchive(asarPath);
if (!archive) {
return invalidArchiveError(asarPath, callback);
}
stats = getOrCreateArchive(asarPath).stat(filePath);
if (!stats) {
return notFoundError(asarPath, filePath, callback);
}
return process.nextTick(function() {
return callback(null, asarStatsToFsStats(stats));
2016-01-12 02:40:23 +00:00
});
2016-01-19 18:25:03 +00:00
};
statSync = fs.statSync;
fs.statSync = function(p) {
2016-03-21 16:39:07 +00:00
const [isAsar] = splitPath(p);
2016-01-19 18:25:03 +00:00
if (!isAsar) {
return statSync(p);
}
2016-01-12 02:40:23 +00:00
2016-01-19 18:25:03 +00:00
// Do not distinguish links for now.
return fs.lstatSync(p);
};
stat = fs.stat;
fs.stat = function(p, callback) {
2016-03-21 16:39:07 +00:00
const [isAsar] = splitPath(p);
2016-01-19 18:25:03 +00:00
if (!isAsar) {
return stat(p, callback);
}
2016-01-12 02:40:23 +00:00
2016-01-19 18:25:03 +00:00
// Do not distinguish links for now.
return process.nextTick(function() {
return fs.lstat(p, callback);
});
};
statSyncNoException = fs.statSyncNoException;
fs.statSyncNoException = function(p) {
2016-03-21 16:39:07 +00:00
var archive, stats;
const [isAsar, asarPath, filePath] = splitPath(p);
2016-01-19 18:25:03 +00:00
if (!isAsar) {
return statSyncNoException(p);
}
archive = getOrCreateArchive(asarPath);
if (!archive) {
return false;
}
stats = archive.stat(filePath);
if (!stats) {
return false;
}
return asarStatsToFsStats(stats);
};
realpathSync = fs.realpathSync;
fs.realpathSync = function(p) {
2016-03-21 16:39:07 +00:00
var archive, real;
const [isAsar, asarPath, filePath] = splitPath(p);
2016-01-19 18:25:03 +00:00
if (!isAsar) {
return realpathSync.apply(this, arguments);
}
archive = getOrCreateArchive(asarPath);
if (!archive) {
invalidArchiveError(asarPath);
}
real = archive.realpath(filePath);
if (real === false) {
notFoundError(asarPath, filePath);
}
return path.join(realpathSync(asarPath), real);
};
realpath = fs.realpath;
fs.realpath = function(p, cache, callback) {
2016-03-21 16:39:07 +00:00
var archive, real;
const [isAsar, asarPath, filePath] = splitPath(p);
2016-01-19 18:25:03 +00:00
if (!isAsar) {
return realpath.apply(this, arguments);
}
if (typeof cache === 'function') {
callback = cache;
cache = void 0;
}
archive = getOrCreateArchive(asarPath);
if (!archive) {
return invalidArchiveError(asarPath, callback);
}
real = archive.realpath(filePath);
if (real === false) {
return notFoundError(asarPath, filePath, callback);
}
return realpath(asarPath, function(err, p) {
if (err) {
return callback(err);
}
return callback(null, path.join(p, real));
});
};
exists = fs.exists;
fs.exists = function(p, callback) {
2016-03-21 16:39:07 +00:00
var archive;
const [isAsar, asarPath, filePath] = splitPath(p);
2016-01-19 18:25:03 +00:00
if (!isAsar) {
return exists(p, callback);
}
archive = getOrCreateArchive(asarPath);
if (!archive) {
return invalidArchiveError(asarPath, callback);
}
return process.nextTick(function() {
return callback(archive.stat(filePath) !== false);
});
};
existsSync = fs.existsSync;
fs.existsSync = function(p) {
2016-03-21 16:39:07 +00:00
var archive;
const [isAsar, asarPath, filePath] = splitPath(p);
2016-01-19 18:25:03 +00:00
if (!isAsar) {
return existsSync(p);
}
archive = getOrCreateArchive(asarPath);
if (!archive) {
return false;
}
return archive.stat(filePath) !== false;
};
readFile = fs.readFile;
fs.readFile = function(p, options, callback) {
2016-03-21 16:39:07 +00:00
var archive, buffer, encoding, fd, info, realPath;
const [isAsar, asarPath, filePath] = splitPath(p);
2016-01-19 18:25:03 +00:00
if (!isAsar) {
return readFile.apply(this, arguments);
}
if (typeof options === 'function') {
callback = options;
options = void 0;
}
archive = getOrCreateArchive(asarPath);
if (!archive) {
return invalidArchiveError(asarPath, callback);
}
info = archive.getFileInfo(filePath);
if (!info) {
return notFoundError(asarPath, filePath, callback);
}
if (info.size === 0) {
return process.nextTick(function() {
return callback(null, new Buffer(0));
});
}
if (info.unpacked) {
realPath = archive.copyFileOut(filePath);
return fs.readFile(realPath, options, callback);
}
if (!options) {
options = {
encoding: null
};
} else if (util.isString(options)) {
options = {
encoding: options
};
} else if (!util.isObject(options)) {
throw new TypeError('Bad arguments');
}
encoding = options.encoding;
buffer = new Buffer(info.size);
fd = archive.getFd();
if (!(fd >= 0)) {
return notFoundError(asarPath, filePath, callback);
}
logASARAccess(asarPath, filePath, info.offset);
2016-01-19 18:25:03 +00:00
return fs.read(fd, buffer, 0, info.size, info.offset, function(error) {
return callback(error, encoding ? buffer.toString(encoding) : buffer);
});
};
readFileSync = fs.readFileSync;
fs.readFileSync = function(p, opts) {
// this allows v8 to optimize this function
2016-03-21 16:39:07 +00:00
var archive, buffer, encoding, fd, info, options, realPath;
2016-01-19 18:25:03 +00:00
options = opts;
2016-03-21 16:39:07 +00:00
const [isAsar, asarPath, filePath] = splitPath(p);
2016-01-19 18:25:03 +00:00
if (!isAsar) {
return readFileSync.apply(this, arguments);
}
archive = getOrCreateArchive(asarPath);
if (!archive) {
invalidArchiveError(asarPath);
}
info = archive.getFileInfo(filePath);
if (!info) {
notFoundError(asarPath, filePath);
}
if (info.size === 0) {
if (options) {
return '';
} else {
return new Buffer(0);
}
}
if (info.unpacked) {
realPath = archive.copyFileOut(filePath);
return fs.readFileSync(realPath, options);
}
if (!options) {
options = {
encoding: null
};
} else if (util.isString(options)) {
options = {
encoding: options
};
} else if (!util.isObject(options)) {
throw new TypeError('Bad arguments');
}
encoding = options.encoding;
buffer = new Buffer(info.size);
fd = archive.getFd();
if (!(fd >= 0)) {
notFoundError(asarPath, filePath);
}
logASARAccess(asarPath, filePath, info.offset);
2016-01-19 18:25:03 +00:00
fs.readSync(fd, buffer, 0, info.size, info.offset);
if (encoding) {
return buffer.toString(encoding);
} else {
return buffer;
}
};
readdir = fs.readdir;
fs.readdir = function(p, callback) {
2016-03-21 16:39:07 +00:00
var archive, files;
const [isAsar, asarPath, filePath] = splitPath(p);
2016-01-19 18:25:03 +00:00
if (!isAsar) {
return readdir.apply(this, arguments);
}
archive = getOrCreateArchive(asarPath);
if (!archive) {
return invalidArchiveError(asarPath, callback);
}
files = archive.readdir(filePath);
if (!files) {
return notFoundError(asarPath, filePath, callback);
2016-01-12 02:40:23 +00:00
}
2016-01-19 18:25:03 +00:00
return process.nextTick(function() {
return callback(null, files);
});
};
readdirSync = fs.readdirSync;
fs.readdirSync = function(p) {
2016-03-21 16:39:07 +00:00
var archive, files;
const [isAsar, asarPath, filePath] = splitPath(p);
2016-01-19 18:25:03 +00:00
if (!isAsar) {
return readdirSync.apply(this, arguments);
}
archive = getOrCreateArchive(asarPath);
if (!archive) {
invalidArchiveError(asarPath);
2016-01-12 02:40:23 +00:00
}
2016-01-19 18:25:03 +00:00
files = archive.readdir(filePath);
if (!files) {
notFoundError(asarPath, filePath);
}
return files;
2016-01-12 02:40:23 +00:00
};
2016-01-19 18:25:03 +00:00
internalModuleReadFile = process.binding('fs').internalModuleReadFile;
process.binding('fs').internalModuleReadFile = function(p) {
2016-03-21 16:39:07 +00:00
var archive, buffer, fd, info, realPath;
const [isAsar, asarPath, filePath] = splitPath(p);
2016-01-19 18:25:03 +00:00
if (!isAsar) {
return internalModuleReadFile(p);
}
archive = getOrCreateArchive(asarPath);
if (!archive) {
return void 0;
}
info = archive.getFileInfo(filePath);
if (!info) {
return void 0;
}
if (info.size === 0) {
return '';
}
if (info.unpacked) {
realPath = archive.copyFileOut(filePath);
return fs.readFileSync(realPath, {
encoding: 'utf8'
});
}
buffer = new Buffer(info.size);
fd = archive.getFd();
if (!(fd >= 0)) {
return void 0;
}
logASARAccess(asarPath, filePath, info.offset);
2016-01-19 18:25:03 +00:00
fs.readSync(fd, buffer, 0, info.size, info.offset);
return buffer.toString('utf8');
};
internalModuleStat = process.binding('fs').internalModuleStat;
process.binding('fs').internalModuleStat = function(p) {
2016-03-21 16:39:07 +00:00
var archive, stats;
const [isAsar, asarPath, filePath] = splitPath(p);
2016-01-19 18:25:03 +00:00
if (!isAsar) {
return internalModuleStat(p);
}
archive = getOrCreateArchive(asarPath);
// -ENOENT
if (!archive) {
return -34;
}
stats = archive.stat(filePath);
// -ENOENT
if (!stats) {
return -34;
}
if (stats.isDirectory) {
return 1;
} else {
return 0;
2016-01-12 02:40:23 +00:00
}
};
2016-01-19 18:25:03 +00:00
// Calling mkdir for directory inside asar archive should throw ENOTDIR
// error, but on Windows it throws ENOENT.
// This is to work around the recursive looping bug of mkdirp since it is
// widely used.
if (process.platform === 'win32') {
mkdir = fs.mkdir;
fs.mkdir = function(p, mode, callback) {
if (typeof mode === 'function') {
callback = mode;
}
2016-03-21 16:39:07 +00:00
const [isAsar, , filePath] = splitPath(p);
2016-01-19 18:25:03 +00:00
if (isAsar && filePath.length) {
return notDirError(callback);
}
return mkdir(p, mode, callback);
};
mkdirSync = fs.mkdirSync;
fs.mkdirSync = function(p, mode) {
2016-03-21 16:39:07 +00:00
const [isAsar, , filePath] = splitPath(p);
2016-01-19 18:25:03 +00:00
if (isAsar && filePath.length) {
notDirError();
}
return mkdirSync(p, mode);
};
}
overrideAPI(fs, 'open');
overrideAPI(child_process, 'execFile');
overrideAPISync(process, 'dlopen', 1);
overrideAPISync(require('module')._extensions, '.node', 1);
overrideAPISync(fs, 'openSync');
return overrideAPISync(child_process, 'execFileSync');
};
2016-01-19 22:53:59 +00:00
})();