@fuman/plist 0.1.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +8 -0
- package/README.md +41 -0
- package/_constants.cjs +11 -0
- package/_constants.d.cts +4 -0
- package/_constants.d.ts +4 -0
- package/_constants.js +11 -0
- package/_utils.cjs +12 -0
- package/_utils.d.cts +1 -0
- package/_utils.d.ts +1 -0
- package/_utils.js +12 -0
- package/bplist-reader.cjs +210 -0
- package/bplist-reader.d.cts +14 -0
- package/bplist-reader.d.ts +14 -0
- package/bplist-reader.js +210 -0
- package/bplist-writer.cjs +344 -0
- package/bplist-writer.d.cts +1 -0
- package/bplist-writer.d.ts +1 -0
- package/bplist-writer.js +344 -0
- package/index.cjs +17 -0
- package/index.d.cts +7 -0
- package/index.d.ts +7 -0
- package/index.js +17 -0
- package/ns-keyed-archiver.cjs +129 -0
- package/ns-keyed-archiver.d.cts +1 -0
- package/ns-keyed-archiver.d.ts +1 -0
- package/ns-keyed-archiver.js +129 -0
- package/ns-keyed-unarchiver.cjs +155 -0
- package/ns-keyed-unarchiver.d.cts +9 -0
- package/ns-keyed-unarchiver.d.ts +9 -0
- package/ns-keyed-unarchiver.js +155 -0
- package/package.json +30 -0
- package/plist-reader.cjs +79 -0
- package/plist-reader.d.cts +15 -0
- package/plist-reader.d.ts +15 -0
- package/plist-reader.js +79 -0
- package/plist-writer.cjs +120 -0
- package/plist-writer.d.cts +27 -0
- package/plist-writer.d.ts +27 -0
- package/plist-writer.js +120 -0
- package/types.cjs +25 -0
- package/types.d.cts +27 -0
- package/types.d.ts +27 -0
- package/types.js +25 -0
package/LICENSE
ADDED
|
@@ -0,0 +1,8 @@
|
|
|
1
|
+
Copyright 2024 alina sireneva
|
|
2
|
+
|
|
3
|
+
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the “Software”), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
|
|
4
|
+
|
|
5
|
+
The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
|
|
6
|
+
|
|
7
|
+
THE SOFTWARE IS PROVIDED “AS IS”, WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
8
|
+
|
package/README.md
ADDED
|
@@ -0,0 +1,41 @@
|
|
|
1
|
+
## @fuman/plist
|
|
2
|
+
|
|
3
|
+
modern typescript utilities for working with property lists (the `.plist` files used in apple operating systems),
|
|
4
|
+
both reading and writing, both binary and xml-based. also supports NSKeyedArchiver/NSKeyedUnarchiver.
|
|
5
|
+
|
|
6
|
+
### binary plist
|
|
7
|
+
|
|
8
|
+
```ts
|
|
9
|
+
import { readBinaryPlist, writeBinaryPlist } from '@fuman/plist'
|
|
10
|
+
|
|
11
|
+
const data = writeBinaryPlist({ meow: 'purr' })
|
|
12
|
+
console.log(readBinaryPlist(data))
|
|
13
|
+
```
|
|
14
|
+
|
|
15
|
+
### xml plist
|
|
16
|
+
|
|
17
|
+
```ts
|
|
18
|
+
import { readXmlPlist, writeXmlPlist } from '@fuman/plist'
|
|
19
|
+
|
|
20
|
+
const data = writeXmlPlist({ meow: 'purr' })
|
|
21
|
+
console.log(readXmlPlist(data))
|
|
22
|
+
```
|
|
23
|
+
|
|
24
|
+
> note: `readXmlPlist` uses [`DOMParser`](https://developer.mozilla.org/en-US/docs/Web/API/DOMParser), which
|
|
25
|
+
> might not be available in your environment (notably, under node.js as of writing). consider using `@xmldom/xmldom`:
|
|
26
|
+
>
|
|
27
|
+
> ```ts
|
|
28
|
+
> import { DOMParser } from '@xmldom/xmldom'
|
|
29
|
+
> console.log(readXmlPlist(data, { DOMParser }))
|
|
30
|
+
> ```
|
|
31
|
+
|
|
32
|
+
### NSKeyedArchiver
|
|
33
|
+
|
|
34
|
+
```ts
|
|
35
|
+
import { nsKeyedArchive, nsKeyedUnarchive, readBinaryPlist, writeBinaryPlist } from '@fuman/plist'
|
|
36
|
+
|
|
37
|
+
const file = nsKeyedUnarchive(readBinaryPlist(await readFile('meow.plist')))
|
|
38
|
+
console.log(file)
|
|
39
|
+
file.foo = 123
|
|
40
|
+
await writeFile('meow2.plist', writeBinaryPlist(nsKeyedArchive(file)))
|
|
41
|
+
```
|
package/_constants.cjs
ADDED
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" });
|
|
3
|
+
const utils = require("@fuman/utils");
|
|
4
|
+
const BPLIST_MAGIC = /* @__PURE__ */ utils.utf8.encoder.encode("bplist");
|
|
5
|
+
const CORE_DATA_EPOCH = 9783072e5;
|
|
6
|
+
const MAX_OBJECT_COUNT = 1024 * 1024;
|
|
7
|
+
const NS_KEYED_ARCHIVER_VERSION = 1e5;
|
|
8
|
+
exports.BPLIST_MAGIC = BPLIST_MAGIC;
|
|
9
|
+
exports.CORE_DATA_EPOCH = CORE_DATA_EPOCH;
|
|
10
|
+
exports.MAX_OBJECT_COUNT = MAX_OBJECT_COUNT;
|
|
11
|
+
exports.NS_KEYED_ARCHIVER_VERSION = NS_KEYED_ARCHIVER_VERSION;
|
package/_constants.d.cts
ADDED
package/_constants.d.ts
ADDED
package/_constants.js
ADDED
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
import { utf8 } from "@fuman/utils";
|
|
2
|
+
const BPLIST_MAGIC = /* @__PURE__ */ utf8.encoder.encode("bplist");
|
|
3
|
+
const CORE_DATA_EPOCH = 9783072e5;
|
|
4
|
+
const MAX_OBJECT_COUNT = 1024 * 1024;
|
|
5
|
+
const NS_KEYED_ARCHIVER_VERSION = 1e5;
|
|
6
|
+
export {
|
|
7
|
+
BPLIST_MAGIC,
|
|
8
|
+
CORE_DATA_EPOCH,
|
|
9
|
+
MAX_OBJECT_COUNT,
|
|
10
|
+
NS_KEYED_ARCHIVER_VERSION
|
|
11
|
+
};
|
package/_utils.cjs
ADDED
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" });
|
|
3
|
+
function safeToNumber(value) {
|
|
4
|
+
if (typeof value === "bigint") {
|
|
5
|
+
if (value > Number.MAX_SAFE_INTEGER) {
|
|
6
|
+
throw new Error(`value is too large: ${value}`);
|
|
7
|
+
}
|
|
8
|
+
return Number(value);
|
|
9
|
+
}
|
|
10
|
+
return value;
|
|
11
|
+
}
|
|
12
|
+
exports.safeToNumber = safeToNumber;
|
package/_utils.d.cts
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export declare function safeToNumber(value: number | bigint): number;
|
package/_utils.d.ts
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export declare function safeToNumber(value: number | bigint): number;
|
package/_utils.js
ADDED
|
@@ -0,0 +1,210 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" });
|
|
3
|
+
const io = require("@fuman/io");
|
|
4
|
+
const utils = require("@fuman/utils");
|
|
5
|
+
const _constants = require("./_constants.cjs");
|
|
6
|
+
const _utils = require("./_utils.cjs");
|
|
7
|
+
const types = require("./types.cjs");
|
|
8
|
+
const UINT_SIZES = new Uint8Array([1, 2, 4, 8]);
|
|
9
|
+
function readBinaryPlist(data, options) {
|
|
10
|
+
const {
|
|
11
|
+
maxObjectCount = _constants.MAX_OBJECT_COUNT,
|
|
12
|
+
skipMagicCheck = false,
|
|
13
|
+
preserveType = false
|
|
14
|
+
} = options ?? {};
|
|
15
|
+
if (data.length < 38) {
|
|
16
|
+
throw new Error("bplist is too small");
|
|
17
|
+
}
|
|
18
|
+
const magic = data.subarray(0, 6);
|
|
19
|
+
if (!skipMagicCheck && !utils.typed.equal(magic, _constants.BPLIST_MAGIC)) {
|
|
20
|
+
throw new Error(`bplist magic is invalid: ${utils.utf8.decoder.decode(magic)}`);
|
|
21
|
+
}
|
|
22
|
+
const trailer = data.subarray(-32);
|
|
23
|
+
const offsetSize = trailer[6];
|
|
24
|
+
const objectRefSize = trailer[7];
|
|
25
|
+
const numObjects = _utils.safeToNumber(io.read.uint64be(trailer.subarray(8, 16)));
|
|
26
|
+
const topObject = _utils.safeToNumber(io.read.uint64be(trailer.subarray(16, 24)));
|
|
27
|
+
const offsetTableOffset = _utils.safeToNumber(io.read.uint64be(trailer.subarray(24, 32)));
|
|
28
|
+
if (numObjects > maxObjectCount) {
|
|
29
|
+
throw new Error("MAX_OBJECT_COUNT exceeded");
|
|
30
|
+
}
|
|
31
|
+
function checkBounds(offset) {
|
|
32
|
+
if (offset >= data.length) {
|
|
33
|
+
throw new Error(`out of bounds: offset ${offset} >= ${data.length}`);
|
|
34
|
+
}
|
|
35
|
+
}
|
|
36
|
+
function readUint(offset, size) {
|
|
37
|
+
switch (size) {
|
|
38
|
+
case 1: {
|
|
39
|
+
checkBounds(offset);
|
|
40
|
+
return data[offset];
|
|
41
|
+
}
|
|
42
|
+
case 2:
|
|
43
|
+
return io.read.uint16be(data.subarray(offset, offset + 2));
|
|
44
|
+
case 4:
|
|
45
|
+
return io.read.uint32be(data.subarray(offset, offset + 4));
|
|
46
|
+
case 8:
|
|
47
|
+
return io.read.uint64be(data.subarray(offset, offset + 8));
|
|
48
|
+
default:
|
|
49
|
+
throw new Error(`invalid int size: ${size}`);
|
|
50
|
+
}
|
|
51
|
+
}
|
|
52
|
+
const readNumber = (offset, size) => _utils.safeToNumber(readUint(offset, size));
|
|
53
|
+
const offsetTable = [];
|
|
54
|
+
for (let i = 0, offset = offsetTableOffset; i < numObjects; i++, offset += offsetSize) {
|
|
55
|
+
offsetTable[i] = readNumber(offset, offsetSize);
|
|
56
|
+
}
|
|
57
|
+
function readBytes(offset, size, isUtf16 = false) {
|
|
58
|
+
if (size === 15) {
|
|
59
|
+
const intSize = UINT_SIZES[data[offset + 1] & 15];
|
|
60
|
+
size = readNumber(offset + 2, intSize);
|
|
61
|
+
offset += 1 + intSize;
|
|
62
|
+
}
|
|
63
|
+
if (isUtf16) size *= 2;
|
|
64
|
+
checkBounds(offset + 1 + size);
|
|
65
|
+
return data.subarray(offset + 1, offset + 1 + size);
|
|
66
|
+
}
|
|
67
|
+
function readObject(ref) {
|
|
68
|
+
const offset = offsetTable[ref];
|
|
69
|
+
if (!offset) throw new Error(`object ${ref} not found`);
|
|
70
|
+
checkBounds(offset);
|
|
71
|
+
const type = data[offset];
|
|
72
|
+
const low = type & 15;
|
|
73
|
+
const high = type >> 4 & 15;
|
|
74
|
+
switch (high) {
|
|
75
|
+
case 0:
|
|
76
|
+
switch (low) {
|
|
77
|
+
case 0:
|
|
78
|
+
return null;
|
|
79
|
+
case 8:
|
|
80
|
+
return false;
|
|
81
|
+
case 9:
|
|
82
|
+
return true;
|
|
83
|
+
case 15:
|
|
84
|
+
return void 0;
|
|
85
|
+
// padding
|
|
86
|
+
default:
|
|
87
|
+
throw new Error(`invalid type 0b${type.toString(2)}`);
|
|
88
|
+
}
|
|
89
|
+
case 1: {
|
|
90
|
+
let int;
|
|
91
|
+
switch (low) {
|
|
92
|
+
case 0:
|
|
93
|
+
int = data[offset + 1];
|
|
94
|
+
break;
|
|
95
|
+
case 1:
|
|
96
|
+
int = io.read.uint16be(data.subarray(offset + 1, offset + 3));
|
|
97
|
+
break;
|
|
98
|
+
case 2:
|
|
99
|
+
int = io.read.uint32be(data.subarray(offset + 1, offset + 7));
|
|
100
|
+
break;
|
|
101
|
+
case 3:
|
|
102
|
+
int = io.read.int64be(data.subarray(offset + 1, offset + 9));
|
|
103
|
+
break;
|
|
104
|
+
case 4:
|
|
105
|
+
int = io.read.intbe(data.subarray(offset + 1, offset + 17), 16);
|
|
106
|
+
break;
|
|
107
|
+
default:
|
|
108
|
+
throw new Error(`invalid type 0b${type.toString(2)}`);
|
|
109
|
+
}
|
|
110
|
+
return preserveType ? new types.PlistValue("int", int) : int;
|
|
111
|
+
}
|
|
112
|
+
case 2: {
|
|
113
|
+
let float;
|
|
114
|
+
switch (low) {
|
|
115
|
+
case 2:
|
|
116
|
+
float = io.read.float32be(data.subarray(offset + 1));
|
|
117
|
+
break;
|
|
118
|
+
case 3:
|
|
119
|
+
float = io.read.float64be(data.subarray(offset + 1));
|
|
120
|
+
break;
|
|
121
|
+
default:
|
|
122
|
+
throw new Error(`invalid type 0b${type.toString(2)}`);
|
|
123
|
+
}
|
|
124
|
+
return preserveType ? new types.PlistValue(low === 2 ? "float32" : "float64", float) : float;
|
|
125
|
+
}
|
|
126
|
+
case 3: {
|
|
127
|
+
if (low !== 3) throw new Error(`invalid type 0b${type.toString(2)}`);
|
|
128
|
+
const seconds = io.read.float64be(data.subarray(offset + 1));
|
|
129
|
+
return new Date(_constants.CORE_DATA_EPOCH + seconds * 1e3);
|
|
130
|
+
}
|
|
131
|
+
case 4:
|
|
132
|
+
return readBytes(offset, low);
|
|
133
|
+
case 5: {
|
|
134
|
+
const bytes = readBytes(offset, low);
|
|
135
|
+
let str = "";
|
|
136
|
+
for (let i = 0; i < bytes.length; i++) {
|
|
137
|
+
const char = bytes[i];
|
|
138
|
+
if (char === 0) break;
|
|
139
|
+
str += String.fromCharCode(char);
|
|
140
|
+
}
|
|
141
|
+
return preserveType ? new types.PlistValue("ascii", str) : str;
|
|
142
|
+
}
|
|
143
|
+
case 6: {
|
|
144
|
+
const bytes = readBytes(offset, low, true);
|
|
145
|
+
const str = io.read.utf16beString(bytes, bytes.length);
|
|
146
|
+
return preserveType ? new types.PlistValue("utf16", str) : str;
|
|
147
|
+
}
|
|
148
|
+
case 7: {
|
|
149
|
+
const str = utils.utf8.decoder.decode(readBytes(offset, low));
|
|
150
|
+
return preserveType ? new types.PlistValue("utf8", str) : str;
|
|
151
|
+
}
|
|
152
|
+
case 8: {
|
|
153
|
+
const int = readUint(offset + 1, UINT_SIZES[low]);
|
|
154
|
+
return new types.PlistValue("uid", int);
|
|
155
|
+
}
|
|
156
|
+
case 10:
|
|
157
|
+
// array-like
|
|
158
|
+
case 11: {
|
|
159
|
+
const array = [];
|
|
160
|
+
let arraySize = low;
|
|
161
|
+
let arrayOffset = offset + 1;
|
|
162
|
+
if (arraySize === 15) {
|
|
163
|
+
const intSize = UINT_SIZES[data[offset + 1] & 15];
|
|
164
|
+
arraySize = readNumber(offset + 2, intSize);
|
|
165
|
+
arrayOffset += 1 + intSize;
|
|
166
|
+
}
|
|
167
|
+
checkBounds(arrayOffset + arraySize * objectRefSize);
|
|
168
|
+
for (let i = 0, offset2 = arrayOffset; i < arraySize; i++, offset2 += objectRefSize) {
|
|
169
|
+
array[i] = readObject(readNumber(offset2, objectRefSize));
|
|
170
|
+
}
|
|
171
|
+
switch (high) {
|
|
172
|
+
case 10:
|
|
173
|
+
return array;
|
|
174
|
+
case 11:
|
|
175
|
+
return new Set(array);
|
|
176
|
+
default:
|
|
177
|
+
return utils.unreachable();
|
|
178
|
+
}
|
|
179
|
+
}
|
|
180
|
+
case 13: {
|
|
181
|
+
const dict = {};
|
|
182
|
+
let dictSize = low;
|
|
183
|
+
let keysOffset = offset + 1;
|
|
184
|
+
if (dictSize === 15) {
|
|
185
|
+
const intSize = UINT_SIZES[data[offset + 1] & 15];
|
|
186
|
+
dictSize = readNumber(offset + 2, intSize);
|
|
187
|
+
keysOffset += 1 + intSize;
|
|
188
|
+
}
|
|
189
|
+
const valuesOffset = keysOffset + objectRefSize * dictSize;
|
|
190
|
+
checkBounds(valuesOffset + dictSize * objectRefSize);
|
|
191
|
+
for (let i = 0, keyOffset = keysOffset, valueOffset = valuesOffset; i < dictSize; i++, keyOffset += objectRefSize, valueOffset += objectRefSize) {
|
|
192
|
+
let key = readObject(readNumber(keyOffset, objectRefSize));
|
|
193
|
+
if (preserveType && key instanceof types.PlistValue) {
|
|
194
|
+
key = key.valueOf();
|
|
195
|
+
}
|
|
196
|
+
if (typeof key !== "string" && typeof key !== "number") {
|
|
197
|
+
throw new TypeError(`invalid key: ${String(key)}`);
|
|
198
|
+
}
|
|
199
|
+
const value = readObject(readNumber(valueOffset, objectRefSize));
|
|
200
|
+
dict[key] = value;
|
|
201
|
+
}
|
|
202
|
+
return dict;
|
|
203
|
+
}
|
|
204
|
+
default:
|
|
205
|
+
throw new Error(`invalid type 0b${type.toString(2)}`);
|
|
206
|
+
}
|
|
207
|
+
}
|
|
208
|
+
return readObject(topObject);
|
|
209
|
+
}
|
|
210
|
+
exports.readBinaryPlist = readBinaryPlist;
|
|
@@ -0,0 +1,14 @@
|
|
|
1
|
+
export declare function readBinaryPlist(data: Uint8Array, options?: {
|
|
2
|
+
/**
|
|
3
|
+
* maximum number of objects to read
|
|
4
|
+
* @default 1024 * 1024
|
|
5
|
+
*/
|
|
6
|
+
maxObjectCount?: number;
|
|
7
|
+
/** when true, the magic check will be skipped */
|
|
8
|
+
skipMagicCheck?: boolean;
|
|
9
|
+
/**
|
|
10
|
+
* when true, some primitive values will be wrapped into {@link PlistValueType} to allow for lossless operation
|
|
11
|
+
* otherwise, only `uid*` values will be wrapped
|
|
12
|
+
*/
|
|
13
|
+
preserveType?: boolean;
|
|
14
|
+
}): unknown;
|
|
@@ -0,0 +1,14 @@
|
|
|
1
|
+
export declare function readBinaryPlist(data: Uint8Array, options?: {
|
|
2
|
+
/**
|
|
3
|
+
* maximum number of objects to read
|
|
4
|
+
* @default 1024 * 1024
|
|
5
|
+
*/
|
|
6
|
+
maxObjectCount?: number;
|
|
7
|
+
/** when true, the magic check will be skipped */
|
|
8
|
+
skipMagicCheck?: boolean;
|
|
9
|
+
/**
|
|
10
|
+
* when true, some primitive values will be wrapped into {@link PlistValueType} to allow for lossless operation
|
|
11
|
+
* otherwise, only `uid*` values will be wrapped
|
|
12
|
+
*/
|
|
13
|
+
preserveType?: boolean;
|
|
14
|
+
}): unknown;
|
package/bplist-reader.js
ADDED
|
@@ -0,0 +1,210 @@
|
|
|
1
|
+
import { read } from "@fuman/io";
|
|
2
|
+
import { typed, utf8, unreachable } from "@fuman/utils";
|
|
3
|
+
import { BPLIST_MAGIC, MAX_OBJECT_COUNT, CORE_DATA_EPOCH } from "./_constants.js";
|
|
4
|
+
import { safeToNumber } from "./_utils.js";
|
|
5
|
+
import { PlistValue } from "./types.js";
|
|
6
|
+
const UINT_SIZES = new Uint8Array([1, 2, 4, 8]);
|
|
7
|
+
function readBinaryPlist(data, options) {
|
|
8
|
+
const {
|
|
9
|
+
maxObjectCount = MAX_OBJECT_COUNT,
|
|
10
|
+
skipMagicCheck = false,
|
|
11
|
+
preserveType = false
|
|
12
|
+
} = options ?? {};
|
|
13
|
+
if (data.length < 38) {
|
|
14
|
+
throw new Error("bplist is too small");
|
|
15
|
+
}
|
|
16
|
+
const magic = data.subarray(0, 6);
|
|
17
|
+
if (!skipMagicCheck && !typed.equal(magic, BPLIST_MAGIC)) {
|
|
18
|
+
throw new Error(`bplist magic is invalid: ${utf8.decoder.decode(magic)}`);
|
|
19
|
+
}
|
|
20
|
+
const trailer = data.subarray(-32);
|
|
21
|
+
const offsetSize = trailer[6];
|
|
22
|
+
const objectRefSize = trailer[7];
|
|
23
|
+
const numObjects = safeToNumber(read.uint64be(trailer.subarray(8, 16)));
|
|
24
|
+
const topObject = safeToNumber(read.uint64be(trailer.subarray(16, 24)));
|
|
25
|
+
const offsetTableOffset = safeToNumber(read.uint64be(trailer.subarray(24, 32)));
|
|
26
|
+
if (numObjects > maxObjectCount) {
|
|
27
|
+
throw new Error("MAX_OBJECT_COUNT exceeded");
|
|
28
|
+
}
|
|
29
|
+
function checkBounds(offset) {
|
|
30
|
+
if (offset >= data.length) {
|
|
31
|
+
throw new Error(`out of bounds: offset ${offset} >= ${data.length}`);
|
|
32
|
+
}
|
|
33
|
+
}
|
|
34
|
+
function readUint(offset, size) {
|
|
35
|
+
switch (size) {
|
|
36
|
+
case 1: {
|
|
37
|
+
checkBounds(offset);
|
|
38
|
+
return data[offset];
|
|
39
|
+
}
|
|
40
|
+
case 2:
|
|
41
|
+
return read.uint16be(data.subarray(offset, offset + 2));
|
|
42
|
+
case 4:
|
|
43
|
+
return read.uint32be(data.subarray(offset, offset + 4));
|
|
44
|
+
case 8:
|
|
45
|
+
return read.uint64be(data.subarray(offset, offset + 8));
|
|
46
|
+
default:
|
|
47
|
+
throw new Error(`invalid int size: ${size}`);
|
|
48
|
+
}
|
|
49
|
+
}
|
|
50
|
+
const readNumber = (offset, size) => safeToNumber(readUint(offset, size));
|
|
51
|
+
const offsetTable = [];
|
|
52
|
+
for (let i = 0, offset = offsetTableOffset; i < numObjects; i++, offset += offsetSize) {
|
|
53
|
+
offsetTable[i] = readNumber(offset, offsetSize);
|
|
54
|
+
}
|
|
55
|
+
function readBytes(offset, size, isUtf16 = false) {
|
|
56
|
+
if (size === 15) {
|
|
57
|
+
const intSize = UINT_SIZES[data[offset + 1] & 15];
|
|
58
|
+
size = readNumber(offset + 2, intSize);
|
|
59
|
+
offset += 1 + intSize;
|
|
60
|
+
}
|
|
61
|
+
if (isUtf16) size *= 2;
|
|
62
|
+
checkBounds(offset + 1 + size);
|
|
63
|
+
return data.subarray(offset + 1, offset + 1 + size);
|
|
64
|
+
}
|
|
65
|
+
function readObject(ref) {
|
|
66
|
+
const offset = offsetTable[ref];
|
|
67
|
+
if (!offset) throw new Error(`object ${ref} not found`);
|
|
68
|
+
checkBounds(offset);
|
|
69
|
+
const type = data[offset];
|
|
70
|
+
const low = type & 15;
|
|
71
|
+
const high = type >> 4 & 15;
|
|
72
|
+
switch (high) {
|
|
73
|
+
case 0:
|
|
74
|
+
switch (low) {
|
|
75
|
+
case 0:
|
|
76
|
+
return null;
|
|
77
|
+
case 8:
|
|
78
|
+
return false;
|
|
79
|
+
case 9:
|
|
80
|
+
return true;
|
|
81
|
+
case 15:
|
|
82
|
+
return void 0;
|
|
83
|
+
// padding
|
|
84
|
+
default:
|
|
85
|
+
throw new Error(`invalid type 0b${type.toString(2)}`);
|
|
86
|
+
}
|
|
87
|
+
case 1: {
|
|
88
|
+
let int;
|
|
89
|
+
switch (low) {
|
|
90
|
+
case 0:
|
|
91
|
+
int = data[offset + 1];
|
|
92
|
+
break;
|
|
93
|
+
case 1:
|
|
94
|
+
int = read.uint16be(data.subarray(offset + 1, offset + 3));
|
|
95
|
+
break;
|
|
96
|
+
case 2:
|
|
97
|
+
int = read.uint32be(data.subarray(offset + 1, offset + 7));
|
|
98
|
+
break;
|
|
99
|
+
case 3:
|
|
100
|
+
int = read.int64be(data.subarray(offset + 1, offset + 9));
|
|
101
|
+
break;
|
|
102
|
+
case 4:
|
|
103
|
+
int = read.intbe(data.subarray(offset + 1, offset + 17), 16);
|
|
104
|
+
break;
|
|
105
|
+
default:
|
|
106
|
+
throw new Error(`invalid type 0b${type.toString(2)}`);
|
|
107
|
+
}
|
|
108
|
+
return preserveType ? new PlistValue("int", int) : int;
|
|
109
|
+
}
|
|
110
|
+
case 2: {
|
|
111
|
+
let float;
|
|
112
|
+
switch (low) {
|
|
113
|
+
case 2:
|
|
114
|
+
float = read.float32be(data.subarray(offset + 1));
|
|
115
|
+
break;
|
|
116
|
+
case 3:
|
|
117
|
+
float = read.float64be(data.subarray(offset + 1));
|
|
118
|
+
break;
|
|
119
|
+
default:
|
|
120
|
+
throw new Error(`invalid type 0b${type.toString(2)}`);
|
|
121
|
+
}
|
|
122
|
+
return preserveType ? new PlistValue(low === 2 ? "float32" : "float64", float) : float;
|
|
123
|
+
}
|
|
124
|
+
case 3: {
|
|
125
|
+
if (low !== 3) throw new Error(`invalid type 0b${type.toString(2)}`);
|
|
126
|
+
const seconds = read.float64be(data.subarray(offset + 1));
|
|
127
|
+
return new Date(CORE_DATA_EPOCH + seconds * 1e3);
|
|
128
|
+
}
|
|
129
|
+
case 4:
|
|
130
|
+
return readBytes(offset, low);
|
|
131
|
+
case 5: {
|
|
132
|
+
const bytes = readBytes(offset, low);
|
|
133
|
+
let str = "";
|
|
134
|
+
for (let i = 0; i < bytes.length; i++) {
|
|
135
|
+
const char = bytes[i];
|
|
136
|
+
if (char === 0) break;
|
|
137
|
+
str += String.fromCharCode(char);
|
|
138
|
+
}
|
|
139
|
+
return preserveType ? new PlistValue("ascii", str) : str;
|
|
140
|
+
}
|
|
141
|
+
case 6: {
|
|
142
|
+
const bytes = readBytes(offset, low, true);
|
|
143
|
+
const str = read.utf16beString(bytes, bytes.length);
|
|
144
|
+
return preserveType ? new PlistValue("utf16", str) : str;
|
|
145
|
+
}
|
|
146
|
+
case 7: {
|
|
147
|
+
const str = utf8.decoder.decode(readBytes(offset, low));
|
|
148
|
+
return preserveType ? new PlistValue("utf8", str) : str;
|
|
149
|
+
}
|
|
150
|
+
case 8: {
|
|
151
|
+
const int = readUint(offset + 1, UINT_SIZES[low]);
|
|
152
|
+
return new PlistValue("uid", int);
|
|
153
|
+
}
|
|
154
|
+
case 10:
|
|
155
|
+
// array-like
|
|
156
|
+
case 11: {
|
|
157
|
+
const array = [];
|
|
158
|
+
let arraySize = low;
|
|
159
|
+
let arrayOffset = offset + 1;
|
|
160
|
+
if (arraySize === 15) {
|
|
161
|
+
const intSize = UINT_SIZES[data[offset + 1] & 15];
|
|
162
|
+
arraySize = readNumber(offset + 2, intSize);
|
|
163
|
+
arrayOffset += 1 + intSize;
|
|
164
|
+
}
|
|
165
|
+
checkBounds(arrayOffset + arraySize * objectRefSize);
|
|
166
|
+
for (let i = 0, offset2 = arrayOffset; i < arraySize; i++, offset2 += objectRefSize) {
|
|
167
|
+
array[i] = readObject(readNumber(offset2, objectRefSize));
|
|
168
|
+
}
|
|
169
|
+
switch (high) {
|
|
170
|
+
case 10:
|
|
171
|
+
return array;
|
|
172
|
+
case 11:
|
|
173
|
+
return new Set(array);
|
|
174
|
+
default:
|
|
175
|
+
return unreachable();
|
|
176
|
+
}
|
|
177
|
+
}
|
|
178
|
+
case 13: {
|
|
179
|
+
const dict = {};
|
|
180
|
+
let dictSize = low;
|
|
181
|
+
let keysOffset = offset + 1;
|
|
182
|
+
if (dictSize === 15) {
|
|
183
|
+
const intSize = UINT_SIZES[data[offset + 1] & 15];
|
|
184
|
+
dictSize = readNumber(offset + 2, intSize);
|
|
185
|
+
keysOffset += 1 + intSize;
|
|
186
|
+
}
|
|
187
|
+
const valuesOffset = keysOffset + objectRefSize * dictSize;
|
|
188
|
+
checkBounds(valuesOffset + dictSize * objectRefSize);
|
|
189
|
+
for (let i = 0, keyOffset = keysOffset, valueOffset = valuesOffset; i < dictSize; i++, keyOffset += objectRefSize, valueOffset += objectRefSize) {
|
|
190
|
+
let key = readObject(readNumber(keyOffset, objectRefSize));
|
|
191
|
+
if (preserveType && key instanceof PlistValue) {
|
|
192
|
+
key = key.valueOf();
|
|
193
|
+
}
|
|
194
|
+
if (typeof key !== "string" && typeof key !== "number") {
|
|
195
|
+
throw new TypeError(`invalid key: ${String(key)}`);
|
|
196
|
+
}
|
|
197
|
+
const value = readObject(readNumber(valueOffset, objectRefSize));
|
|
198
|
+
dict[key] = value;
|
|
199
|
+
}
|
|
200
|
+
return dict;
|
|
201
|
+
}
|
|
202
|
+
default:
|
|
203
|
+
throw new Error(`invalid type 0b${type.toString(2)}`);
|
|
204
|
+
}
|
|
205
|
+
}
|
|
206
|
+
return readObject(topObject);
|
|
207
|
+
}
|
|
208
|
+
export {
|
|
209
|
+
readBinaryPlist
|
|
210
|
+
};
|