2023-10-03 11:14:36 +08:00
|
|
|
'use strict'
|
|
|
|
const Header = require('./header.js')
|
|
|
|
const path = require('path')
|
|
|
|
|
|
|
|
class Pax {
|
|
|
|
constructor (obj, global) {
|
|
|
|
this.atime = obj.atime || null
|
|
|
|
this.charset = obj.charset || null
|
|
|
|
this.comment = obj.comment || null
|
|
|
|
this.ctime = obj.ctime || null
|
|
|
|
this.gid = obj.gid || null
|
|
|
|
this.gname = obj.gname || null
|
|
|
|
this.linkpath = obj.linkpath || null
|
|
|
|
this.mtime = obj.mtime || null
|
|
|
|
this.path = obj.path || null
|
|
|
|
this.size = obj.size || null
|
|
|
|
this.uid = obj.uid || null
|
|
|
|
this.uname = obj.uname || null
|
|
|
|
this.dev = obj.dev || null
|
|
|
|
this.ino = obj.ino || null
|
|
|
|
this.nlink = obj.nlink || null
|
|
|
|
this.global = global || false
|
|
|
|
}
|
|
|
|
|
|
|
|
encode () {
|
|
|
|
const body = this.encodeBody()
|
|
|
|
if (body === '') {
|
|
|
|
return null
|
|
|
|
}
|
|
|
|
|
|
|
|
const bodyLen = Buffer.byteLength(body)
|
|
|
|
// round up to 512 bytes
|
|
|
|
// add 512 for header
|
|
|
|
const bufLen = 512 * Math.ceil(1 + bodyLen / 512)
|
|
|
|
const buf = Buffer.allocUnsafe(bufLen)
|
|
|
|
|
|
|
|
// 0-fill the header section, it might not hit every field
|
|
|
|
for (let i = 0; i < 512; i++) {
|
|
|
|
buf[i] = 0
|
|
|
|
}
|
|
|
|
|
|
|
|
new Header({
|
|
|
|
// XXX split the path
|
|
|
|
// then the path should be PaxHeader + basename, but less than 99,
|
|
|
|
// prepend with the dirname
|
|
|
|
path: ('PaxHeader/' + path.basename(this.path)).slice(0, 99),
|
|
|
|
mode: this.mode || 0o644,
|
|
|
|
uid: this.uid || null,
|
|
|
|
gid: this.gid || null,
|
|
|
|
size: bodyLen,
|
|
|
|
mtime: this.mtime || null,
|
|
|
|
type: this.global ? 'GlobalExtendedHeader' : 'ExtendedHeader',
|
|
|
|
linkpath: '',
|
|
|
|
uname: this.uname || '',
|
|
|
|
gname: this.gname || '',
|
|
|
|
devmaj: 0,
|
|
|
|
devmin: 0,
|
|
|
|
atime: this.atime || null,
|
|
|
|
ctime: this.ctime || null,
|
|
|
|
}).encode(buf)
|
|
|
|
|
|
|
|
buf.write(body, 512, bodyLen, 'utf8')
|
|
|
|
|
|
|
|
// null pad after the body
|
|
|
|
for (let i = bodyLen + 512; i < buf.length; i++) {
|
|
|
|
buf[i] = 0
|
|
|
|
}
|
|
|
|
|
|
|
|
return buf
|
|
|
|
}
|
|
|
|
|
|
|
|
encodeBody () {
|
|
|
|
return (
|
|
|
|
this.encodeField('path') +
|
|
|
|
this.encodeField('ctime') +
|
|
|
|
this.encodeField('atime') +
|
|
|
|
this.encodeField('dev') +
|
|
|
|
this.encodeField('ino') +
|
|
|
|
this.encodeField('nlink') +
|
|
|
|
this.encodeField('charset') +
|
|
|
|
this.encodeField('comment') +
|
|
|
|
this.encodeField('gid') +
|
|
|
|
this.encodeField('gname') +
|
|
|
|
this.encodeField('linkpath') +
|
|
|
|
this.encodeField('mtime') +
|
|
|
|
this.encodeField('size') +
|
|
|
|
this.encodeField('uid') +
|
|
|
|
this.encodeField('uname')
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
encodeField (field) {
|
|
|
|
if (this[field] === null || this[field] === undefined) {
|
|
|
|
return ''
|
|
|
|
}
|
|
|
|
const v = this[field] instanceof Date ? this[field].getTime() / 1000
|
|
|
|
: this[field]
|
|
|
|
const s = ' ' +
|
|
|
|
(field === 'dev' || field === 'ino' || field === 'nlink'
|
|
|
|
? 'SCHILY.' : '') +
|
|
|
|
field + '=' + v + '\n'
|
|
|
|
const byteLen = Buffer.byteLength(s)
|
|
|
|
// the digits includes the length of the digits in ascii base-10
|
|
|
|
// so if it's 9 characters, then adding 1 for the 9 makes it 10
|
|
|
|
// which makes it 11 chars.
|
|
|
|
let digits = Math.floor(Math.log(byteLen) / Math.log(10)) + 1
|
|
|
|
if (byteLen + digits >= Math.pow(10, digits)) {
|
|
|
|
digits += 1
|
|
|
|
}
|
|
|
|
const len = digits + byteLen
|
|
|
|
return len + s
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Pax.parse = (string, ex, g) => new Pax(merge(parseKV(string), ex), g)
|
|
|
|
|
|
|
|
const merge = (a, b) =>
|
|
|
|
b ? Object.keys(a).reduce((s, k) => (s[k] = a[k], s), b) : a
|
|
|
|
|
|
|
|
const parseKV = string =>
|
|
|
|
string
|
|
|
|
.replace(/\n$/, '')
|
|
|
|
.split('\n')
|
|
|
|
.reduce(parseKVLine, Object.create(null))
|
|
|
|
|
|
|
|
const parseKVLine = (set, line) => {
|
|
|
|
const n = parseInt(line, 10)
|
|
|
|
|
|
|
|
// XXX Values with \n in them will fail this.
|
|
|
|
// Refactor to not be a naive line-by-line parse.
|
|
|
|
if (n !== Buffer.byteLength(line) + 1) {
|
|
|
|
return set
|
|
|
|
}
|
|
|
|
|
|
|
|
line = line.slice((n + ' ').length)
|
|
|
|
const kv = line.split('=')
|
|
|
|
const k = kv.shift().replace(/^SCHILY\.(dev|ino|nlink)/, '$1')
|
|
|
|
if (!k) {
|
|
|
|
return set
|
|
|
|
}
|
|
|
|
|
|
|
|
const v = kv.join('=')
|
|
|
|
set[k] = /^([A-Z]+\.)?([mac]|birth|creation)time$/.test(k)
|
|
|
|
? new Date(v * 1000)
|
|
|
|
: /^[0-9]+$/.test(v) ? +v
|
|
|
|
: v
|
|
|
|
return set
|
|
|
|
}
|
|
|
|
|
|
|
|
module.exports = Pax
|