306 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			JavaScript
		
	
	
			
		
		
	
	
			306 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			JavaScript
		
	
	
"use strict";
 | 
						|
// parse a 512-byte header block to a data object, or vice-versa
 | 
						|
// encode returns `true` if a pax extended header is needed, because
 | 
						|
// the data could not be faithfully encoded in a simple header.
 | 
						|
// (Also, check header.needPax to see if it needs a pax header.)
 | 
						|
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
 | 
						|
    if (k2 === undefined) k2 = k;
 | 
						|
    var desc = Object.getOwnPropertyDescriptor(m, k);
 | 
						|
    if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
 | 
						|
      desc = { enumerable: true, get: function() { return m[k]; } };
 | 
						|
    }
 | 
						|
    Object.defineProperty(o, k2, desc);
 | 
						|
}) : (function(o, m, k, k2) {
 | 
						|
    if (k2 === undefined) k2 = k;
 | 
						|
    o[k2] = m[k];
 | 
						|
}));
 | 
						|
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
 | 
						|
    Object.defineProperty(o, "default", { enumerable: true, value: v });
 | 
						|
}) : function(o, v) {
 | 
						|
    o["default"] = v;
 | 
						|
});
 | 
						|
var __importStar = (this && this.__importStar) || function (mod) {
 | 
						|
    if (mod && mod.__esModule) return mod;
 | 
						|
    var result = {};
 | 
						|
    if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
 | 
						|
    __setModuleDefault(result, mod);
 | 
						|
    return result;
 | 
						|
};
 | 
						|
Object.defineProperty(exports, "__esModule", { value: true });
 | 
						|
exports.Header = void 0;
 | 
						|
const node_path_1 = require("node:path");
 | 
						|
const large = __importStar(require("./large-numbers.js"));
 | 
						|
const types = __importStar(require("./types.js"));
 | 
						|
class Header {
 | 
						|
    cksumValid = false;
 | 
						|
    needPax = false;
 | 
						|
    nullBlock = false;
 | 
						|
    block;
 | 
						|
    path;
 | 
						|
    mode;
 | 
						|
    uid;
 | 
						|
    gid;
 | 
						|
    size;
 | 
						|
    cksum;
 | 
						|
    #type = 'Unsupported';
 | 
						|
    linkpath;
 | 
						|
    uname;
 | 
						|
    gname;
 | 
						|
    devmaj = 0;
 | 
						|
    devmin = 0;
 | 
						|
    atime;
 | 
						|
    ctime;
 | 
						|
    mtime;
 | 
						|
    charset;
 | 
						|
    comment;
 | 
						|
    constructor(data, off = 0, ex, gex) {
 | 
						|
        if (Buffer.isBuffer(data)) {
 | 
						|
            this.decode(data, off || 0, ex, gex);
 | 
						|
        }
 | 
						|
        else if (data) {
 | 
						|
            this.#slurp(data);
 | 
						|
        }
 | 
						|
    }
 | 
						|
    decode(buf, off, ex, gex) {
 | 
						|
        if (!off) {
 | 
						|
            off = 0;
 | 
						|
        }
 | 
						|
        if (!buf || !(buf.length >= off + 512)) {
 | 
						|
            throw new Error('need 512 bytes for header');
 | 
						|
        }
 | 
						|
        this.path = decString(buf, off, 100);
 | 
						|
        this.mode = decNumber(buf, off + 100, 8);
 | 
						|
        this.uid = decNumber(buf, off + 108, 8);
 | 
						|
        this.gid = decNumber(buf, off + 116, 8);
 | 
						|
        this.size = decNumber(buf, off + 124, 12);
 | 
						|
        this.mtime = decDate(buf, off + 136, 12);
 | 
						|
        this.cksum = decNumber(buf, off + 148, 12);
 | 
						|
        // if we have extended or global extended headers, apply them now
 | 
						|
        // See https://github.com/npm/node-tar/pull/187
 | 
						|
        // Apply global before local, so it overrides
 | 
						|
        if (gex)
 | 
						|
            this.#slurp(gex, true);
 | 
						|
        if (ex)
 | 
						|
            this.#slurp(ex);
 | 
						|
        // old tar versions marked dirs as a file with a trailing /
 | 
						|
        const t = decString(buf, off + 156, 1);
 | 
						|
        if (types.isCode(t)) {
 | 
						|
            this.#type = t || '0';
 | 
						|
        }
 | 
						|
        if (this.#type === '0' && this.path.slice(-1) === '/') {
 | 
						|
            this.#type = '5';
 | 
						|
        }
 | 
						|
        // tar implementations sometimes incorrectly put the stat(dir).size
 | 
						|
        // as the size in the tarball, even though Directory entries are
 | 
						|
        // not able to have any body at all.  In the very rare chance that
 | 
						|
        // it actually DOES have a body, we weren't going to do anything with
 | 
						|
        // it anyway, and it'll just be a warning about an invalid header.
 | 
						|
        if (this.#type === '5') {
 | 
						|
            this.size = 0;
 | 
						|
        }
 | 
						|
        this.linkpath = decString(buf, off + 157, 100);
 | 
						|
        if (buf.subarray(off + 257, off + 265).toString() ===
 | 
						|
            'ustar\u000000') {
 | 
						|
            this.uname = decString(buf, off + 265, 32);
 | 
						|
            this.gname = decString(buf, off + 297, 32);
 | 
						|
            /* c8 ignore start */
 | 
						|
            this.devmaj = decNumber(buf, off + 329, 8) ?? 0;
 | 
						|
            this.devmin = decNumber(buf, off + 337, 8) ?? 0;
 | 
						|
            /* c8 ignore stop */
 | 
						|
            if (buf[off + 475] !== 0) {
 | 
						|
                // definitely a prefix, definitely >130 chars.
 | 
						|
                const prefix = decString(buf, off + 345, 155);
 | 
						|
                this.path = prefix + '/' + this.path;
 | 
						|
            }
 | 
						|
            else {
 | 
						|
                const prefix = decString(buf, off + 345, 130);
 | 
						|
                if (prefix) {
 | 
						|
                    this.path = prefix + '/' + this.path;
 | 
						|
                }
 | 
						|
                this.atime = decDate(buf, off + 476, 12);
 | 
						|
                this.ctime = decDate(buf, off + 488, 12);
 | 
						|
            }
 | 
						|
        }
 | 
						|
        let sum = 8 * 0x20;
 | 
						|
        for (let i = off; i < off + 148; i++) {
 | 
						|
            sum += buf[i];
 | 
						|
        }
 | 
						|
        for (let i = off + 156; i < off + 512; i++) {
 | 
						|
            sum += buf[i];
 | 
						|
        }
 | 
						|
        this.cksumValid = sum === this.cksum;
 | 
						|
        if (this.cksum === undefined && sum === 8 * 0x20) {
 | 
						|
            this.nullBlock = true;
 | 
						|
        }
 | 
						|
    }
 | 
						|
    #slurp(ex, gex = false) {
 | 
						|
        Object.assign(this, Object.fromEntries(Object.entries(ex).filter(([k, v]) => {
 | 
						|
            // we slurp in everything except for the path attribute in
 | 
						|
            // a global extended header, because that's weird. Also, any
 | 
						|
            // null/undefined values are ignored.
 | 
						|
            return !(v === null ||
 | 
						|
                v === undefined ||
 | 
						|
                (k === 'path' && gex) ||
 | 
						|
                (k === 'linkpath' && gex) ||
 | 
						|
                k === 'global');
 | 
						|
        })));
 | 
						|
    }
 | 
						|
    encode(buf, off = 0) {
 | 
						|
        if (!buf) {
 | 
						|
            buf = this.block = Buffer.alloc(512);
 | 
						|
        }
 | 
						|
        if (this.#type === 'Unsupported') {
 | 
						|
            this.#type = '0';
 | 
						|
        }
 | 
						|
        if (!(buf.length >= off + 512)) {
 | 
						|
            throw new Error('need 512 bytes for header');
 | 
						|
        }
 | 
						|
        const prefixSize = this.ctime || this.atime ? 130 : 155;
 | 
						|
        const split = splitPrefix(this.path || '', prefixSize);
 | 
						|
        const path = split[0];
 | 
						|
        const prefix = split[1];
 | 
						|
        this.needPax = !!split[2];
 | 
						|
        this.needPax = encString(buf, off, 100, path) || this.needPax;
 | 
						|
        this.needPax =
 | 
						|
            encNumber(buf, off + 100, 8, this.mode) || this.needPax;
 | 
						|
        this.needPax =
 | 
						|
            encNumber(buf, off + 108, 8, this.uid) || this.needPax;
 | 
						|
        this.needPax =
 | 
						|
            encNumber(buf, off + 116, 8, this.gid) || this.needPax;
 | 
						|
        this.needPax =
 | 
						|
            encNumber(buf, off + 124, 12, this.size) || this.needPax;
 | 
						|
        this.needPax =
 | 
						|
            encDate(buf, off + 136, 12, this.mtime) || this.needPax;
 | 
						|
        buf[off + 156] = this.#type.charCodeAt(0);
 | 
						|
        this.needPax =
 | 
						|
            encString(buf, off + 157, 100, this.linkpath) || this.needPax;
 | 
						|
        buf.write('ustar\u000000', off + 257, 8);
 | 
						|
        this.needPax =
 | 
						|
            encString(buf, off + 265, 32, this.uname) || this.needPax;
 | 
						|
        this.needPax =
 | 
						|
            encString(buf, off + 297, 32, this.gname) || this.needPax;
 | 
						|
        this.needPax =
 | 
						|
            encNumber(buf, off + 329, 8, this.devmaj) || this.needPax;
 | 
						|
        this.needPax =
 | 
						|
            encNumber(buf, off + 337, 8, this.devmin) || this.needPax;
 | 
						|
        this.needPax =
 | 
						|
            encString(buf, off + 345, prefixSize, prefix) || this.needPax;
 | 
						|
        if (buf[off + 475] !== 0) {
 | 
						|
            this.needPax =
 | 
						|
                encString(buf, off + 345, 155, prefix) || this.needPax;
 | 
						|
        }
 | 
						|
        else {
 | 
						|
            this.needPax =
 | 
						|
                encString(buf, off + 345, 130, prefix) || this.needPax;
 | 
						|
            this.needPax =
 | 
						|
                encDate(buf, off + 476, 12, this.atime) || this.needPax;
 | 
						|
            this.needPax =
 | 
						|
                encDate(buf, off + 488, 12, this.ctime) || this.needPax;
 | 
						|
        }
 | 
						|
        let sum = 8 * 0x20;
 | 
						|
        for (let i = off; i < off + 148; i++) {
 | 
						|
            sum += buf[i];
 | 
						|
        }
 | 
						|
        for (let i = off + 156; i < off + 512; i++) {
 | 
						|
            sum += buf[i];
 | 
						|
        }
 | 
						|
        this.cksum = sum;
 | 
						|
        encNumber(buf, off + 148, 8, this.cksum);
 | 
						|
        this.cksumValid = true;
 | 
						|
        return this.needPax;
 | 
						|
    }
 | 
						|
    get type() {
 | 
						|
        return (this.#type === 'Unsupported' ?
 | 
						|
            this.#type
 | 
						|
            : types.name.get(this.#type));
 | 
						|
    }
 | 
						|
    get typeKey() {
 | 
						|
        return this.#type;
 | 
						|
    }
 | 
						|
    set type(type) {
 | 
						|
        const c = String(types.code.get(type));
 | 
						|
        if (types.isCode(c) || c === 'Unsupported') {
 | 
						|
            this.#type = c;
 | 
						|
        }
 | 
						|
        else if (types.isCode(type)) {
 | 
						|
            this.#type = type;
 | 
						|
        }
 | 
						|
        else {
 | 
						|
            throw new TypeError('invalid entry type: ' + type);
 | 
						|
        }
 | 
						|
    }
 | 
						|
}
 | 
						|
exports.Header = Header;
 | 
						|
const splitPrefix = (p, prefixSize) => {
 | 
						|
    const pathSize = 100;
 | 
						|
    let pp = p;
 | 
						|
    let prefix = '';
 | 
						|
    let ret = undefined;
 | 
						|
    const root = node_path_1.posix.parse(p).root || '.';
 | 
						|
    if (Buffer.byteLength(pp) < pathSize) {
 | 
						|
        ret = [pp, prefix, false];
 | 
						|
    }
 | 
						|
    else {
 | 
						|
        // first set prefix to the dir, and path to the base
 | 
						|
        prefix = node_path_1.posix.dirname(pp);
 | 
						|
        pp = node_path_1.posix.basename(pp);
 | 
						|
        do {
 | 
						|
            if (Buffer.byteLength(pp) <= pathSize &&
 | 
						|
                Buffer.byteLength(prefix) <= prefixSize) {
 | 
						|
                // both fit!
 | 
						|
                ret = [pp, prefix, false];
 | 
						|
            }
 | 
						|
            else if (Buffer.byteLength(pp) > pathSize &&
 | 
						|
                Buffer.byteLength(prefix) <= prefixSize) {
 | 
						|
                // prefix fits in prefix, but path doesn't fit in path
 | 
						|
                ret = [pp.slice(0, pathSize - 1), prefix, true];
 | 
						|
            }
 | 
						|
            else {
 | 
						|
                // make path take a bit from prefix
 | 
						|
                pp = node_path_1.posix.join(node_path_1.posix.basename(prefix), pp);
 | 
						|
                prefix = node_path_1.posix.dirname(prefix);
 | 
						|
            }
 | 
						|
        } while (prefix !== root && ret === undefined);
 | 
						|
        // at this point, found no resolution, just truncate
 | 
						|
        if (!ret) {
 | 
						|
            ret = [p.slice(0, pathSize - 1), '', true];
 | 
						|
        }
 | 
						|
    }
 | 
						|
    return ret;
 | 
						|
};
 | 
						|
const decString = (buf, off, size) => buf
 | 
						|
    .subarray(off, off + size)
 | 
						|
    .toString('utf8')
 | 
						|
    .replace(/\0.*/, '');
 | 
						|
const decDate = (buf, off, size) => numToDate(decNumber(buf, off, size));
 | 
						|
const numToDate = (num) => num === undefined ? undefined : new Date(num * 1000);
 | 
						|
const decNumber = (buf, off, size) => Number(buf[off]) & 0x80 ?
 | 
						|
    large.parse(buf.subarray(off, off + size))
 | 
						|
    : decSmallNumber(buf, off, size);
 | 
						|
const nanUndef = (value) => (isNaN(value) ? undefined : value);
 | 
						|
const decSmallNumber = (buf, off, size) => nanUndef(parseInt(buf
 | 
						|
    .subarray(off, off + size)
 | 
						|
    .toString('utf8')
 | 
						|
    .replace(/\0.*$/, '')
 | 
						|
    .trim(), 8));
 | 
						|
// the maximum encodable as a null-terminated octal, by field size
 | 
						|
const MAXNUM = {
 | 
						|
    12: 0o77777777777,
 | 
						|
    8: 0o7777777,
 | 
						|
};
 | 
						|
const encNumber = (buf, off, size, num) => num === undefined ? false
 | 
						|
    : num > MAXNUM[size] || num < 0 ?
 | 
						|
        (large.encode(num, buf.subarray(off, off + size)), true)
 | 
						|
        : (encSmallNumber(buf, off, size, num), false);
 | 
						|
const encSmallNumber = (buf, off, size, num) => buf.write(octalString(num, size), off, size, 'ascii');
 | 
						|
const octalString = (num, size) => padOctal(Math.floor(num).toString(8), size);
 | 
						|
const padOctal = (str, size) => (str.length === size - 1 ?
 | 
						|
    str
 | 
						|
    : new Array(size - str.length - 1).join('0') + str + ' ') + '\0';
 | 
						|
const encDate = (buf, off, size, date) => date === undefined ? false : (encNumber(buf, off, size, date.getTime() / 1000));
 | 
						|
// enough to fill the longest string we've got
 | 
						|
const NULLS = new Array(156).join('\0');
 | 
						|
// pad with nulls, return true if it's longer or non-ascii
 | 
						|
const encString = (buf, off, size, str) => str === undefined ? false : ((buf.write(str + NULLS, off, size, 'utf8'),
 | 
						|
    str.length !== Buffer.byteLength(str) || str.length > size));
 | 
						|
//# sourceMappingURL=header.js.map
 |