124 lines
3.9 KiB
JavaScript
124 lines
3.9 KiB
JavaScript
|
'use strict'
|
||
|
|
||
|
const fs = require('./wrapped-fs')
|
||
|
const path = require('path')
|
||
|
const pickle = require('chromium-pickle-js')
|
||
|
|
||
|
const Filesystem = require('./filesystem')
|
||
|
let filesystemCache = {}
|
||
|
|
||
|
async function copyFile (dest, src, filename) {
|
||
|
const srcFile = path.join(src, filename)
|
||
|
const targetFile = path.join(dest, filename)
|
||
|
|
||
|
const [content, stats] = await Promise.all([fs.readFile(srcFile), fs.stat(srcFile), fs.mkdirp(path.dirname(targetFile))])
|
||
|
return fs.writeFile(targetFile, content, { mode: stats.mode })
|
||
|
}
|
||
|
|
||
|
async function streamTransformedFile (originalFilename, outStream, transformed) {
|
||
|
return new Promise((resolve, reject) => {
|
||
|
const stream = fs.createReadStream(transformed ? transformed.path : originalFilename)
|
||
|
stream.pipe(outStream, { end: false })
|
||
|
stream.on('error', reject)
|
||
|
stream.on('end', () => resolve())
|
||
|
})
|
||
|
}
|
||
|
|
||
|
const writeFileListToStream = async function (dest, filesystem, out, list, metadata) {
|
||
|
for (const file of list) {
|
||
|
if (file.unpack) { // the file should not be packed into archive
|
||
|
const filename = path.relative(filesystem.src, file.filename)
|
||
|
await copyFile(`${dest}.unpacked`, filesystem.src, filename)
|
||
|
} else {
|
||
|
await streamTransformedFile(file.filename, out, metadata[file.filename].transformed)
|
||
|
}
|
||
|
}
|
||
|
return out.end()
|
||
|
}
|
||
|
|
||
|
module.exports.writeFilesystem = async function (dest, filesystem, files, metadata) {
|
||
|
const headerPickle = pickle.createEmpty()
|
||
|
headerPickle.writeString(JSON.stringify(filesystem.header))
|
||
|
const headerBuf = headerPickle.toBuffer()
|
||
|
|
||
|
const sizePickle = pickle.createEmpty()
|
||
|
sizePickle.writeUInt32(headerBuf.length)
|
||
|
const sizeBuf = sizePickle.toBuffer()
|
||
|
|
||
|
const out = fs.createWriteStream(dest)
|
||
|
await new Promise((resolve, reject) => {
|
||
|
out.on('error', reject)
|
||
|
out.write(sizeBuf)
|
||
|
return out.write(headerBuf, () => resolve())
|
||
|
})
|
||
|
return writeFileListToStream(dest, filesystem, out, files, metadata)
|
||
|
}
|
||
|
|
||
|
module.exports.readArchiveHeaderSync = function (archive) {
|
||
|
const fd = fs.openSync(archive, 'r')
|
||
|
let size
|
||
|
let headerBuf
|
||
|
try {
|
||
|
const sizeBuf = Buffer.alloc(8)
|
||
|
if (fs.readSync(fd, sizeBuf, 0, 8, null) !== 8) {
|
||
|
throw new Error('Unable to read header size')
|
||
|
}
|
||
|
|
||
|
const sizePickle = pickle.createFromBuffer(sizeBuf)
|
||
|
size = sizePickle.createIterator().readUInt32()
|
||
|
headerBuf = Buffer.alloc(size)
|
||
|
if (fs.readSync(fd, headerBuf, 0, size, null) !== size) {
|
||
|
throw new Error('Unable to read header')
|
||
|
}
|
||
|
} finally {
|
||
|
fs.closeSync(fd)
|
||
|
}
|
||
|
|
||
|
const headerPickle = pickle.createFromBuffer(headerBuf)
|
||
|
const header = headerPickle.createIterator().readString()
|
||
|
return { header: JSON.parse(header), headerSize: size }
|
||
|
}
|
||
|
|
||
|
module.exports.readFilesystemSync = function (archive) {
|
||
|
if (!filesystemCache[archive]) {
|
||
|
const header = this.readArchiveHeaderSync(archive)
|
||
|
const filesystem = new Filesystem(archive)
|
||
|
filesystem.header = header.header
|
||
|
filesystem.headerSize = header.headerSize
|
||
|
filesystemCache[archive] = filesystem
|
||
|
}
|
||
|
return filesystemCache[archive]
|
||
|
}
|
||
|
|
||
|
module.exports.uncacheFilesystem = function (archive) {
|
||
|
if (filesystemCache[archive]) {
|
||
|
filesystemCache[archive] = undefined
|
||
|
return true
|
||
|
}
|
||
|
return false
|
||
|
}
|
||
|
|
||
|
module.exports.uncacheAll = function () {
|
||
|
filesystemCache = {}
|
||
|
}
|
||
|
|
||
|
module.exports.readFileSync = function (filesystem, filename, info) {
|
||
|
let buffer = Buffer.alloc(info.size)
|
||
|
if (info.size <= 0) { return buffer }
|
||
|
if (info.unpacked) {
|
||
|
// it's an unpacked file, copy it.
|
||
|
buffer = fs.readFileSync(path.join(`${filesystem.src}.unpacked`, filename))
|
||
|
} else {
|
||
|
// Node throws an exception when reading 0 bytes into a 0-size buffer,
|
||
|
// so we short-circuit the read in this case.
|
||
|
const fd = fs.openSync(filesystem.src, 'r')
|
||
|
try {
|
||
|
const offset = 8 + filesystem.headerSize + parseInt(info.offset)
|
||
|
fs.readSync(fd, buffer, 0, info.size, offset)
|
||
|
} finally {
|
||
|
fs.closeSync(fd)
|
||
|
}
|
||
|
}
|
||
|
return buffer
|
||
|
}
|