2023-06-15 18:30:33 +00:00
|
|
|
import { formatHex } from './formatHex';
|
|
|
|
|
2023-06-12 23:26:30 +00:00
|
|
|
const textDecoder = new TextDecoder('utf-8', { ignoreBOM: true });
|
|
|
|
|
|
|
|
export class MMKVParser {
|
2023-06-15 18:30:33 +00:00
|
|
|
private offset = 4;
|
2023-06-12 23:26:30 +00:00
|
|
|
private length: number;
|
|
|
|
|
|
|
|
constructor(private view: DataView) {
|
|
|
|
const payloadLength = view.getUint32(0, true);
|
|
|
|
this.length = 4 + payloadLength;
|
2023-06-15 18:30:33 +00:00
|
|
|
|
|
|
|
// skip unused str
|
|
|
|
this.readInt();
|
2023-06-12 23:26:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
toString() {
|
2023-06-15 18:30:33 +00:00
|
|
|
const offset = formatHex(this.offset, 8);
|
|
|
|
const length = formatHex(this.length, 8);
|
|
|
|
return `<MMKVParser offset=${offset} len=${length}>`;
|
2023-06-12 23:26:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
get eof() {
|
|
|
|
return this.offset >= this.length;
|
|
|
|
}
|
|
|
|
|
|
|
|
public readByte() {
|
|
|
|
return this.view.getUint8(this.offset++);
|
|
|
|
}
|
|
|
|
|
|
|
|
public readBigInt() {
|
|
|
|
let value = 0n;
|
|
|
|
let shift = 0n;
|
|
|
|
|
|
|
|
let b: number;
|
|
|
|
do {
|
|
|
|
b = this.readByte();
|
|
|
|
value |= BigInt(b & 0x7f) << shift;
|
|
|
|
shift += 7n;
|
|
|
|
} while ((b & 0x80) !== 0);
|
|
|
|
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
|
|
|
public readInt() {
|
|
|
|
const value = this.readBigInt();
|
|
|
|
|
|
|
|
if (value > Number.MAX_SAFE_INTEGER) {
|
|
|
|
throw new Error('Runtime Error: BigInt too large to cast as number');
|
|
|
|
}
|
|
|
|
|
|
|
|
return Number(value);
|
|
|
|
}
|
|
|
|
|
|
|
|
public readBytes(n: number) {
|
|
|
|
const offset = this.offset;
|
|
|
|
const end = offset + n;
|
|
|
|
this.offset = end;
|
|
|
|
return new Uint8Array(this.view.buffer.slice(offset, end));
|
|
|
|
}
|
|
|
|
|
|
|
|
public readString() {
|
|
|
|
// String [
|
|
|
|
// len: int,
|
|
|
|
// data: byte[int], # utf-8
|
|
|
|
// ]
|
|
|
|
const strByteLen = this.readInt();
|
|
|
|
const data = this.readBytes(strByteLen);
|
|
|
|
return textDecoder.decode(data).normalize();
|
|
|
|
}
|
|
|
|
|
|
|
|
public readVariantString() {
|
|
|
|
// Container [
|
|
|
|
// len: int,
|
|
|
|
// data: variant
|
|
|
|
// ]
|
|
|
|
const containerLen = this.readInt();
|
|
|
|
const newOffset = this.offset + containerLen;
|
|
|
|
const result = this.readString();
|
|
|
|
if (newOffset !== this.offset) {
|
2023-06-15 18:30:33 +00:00
|
|
|
const expected = formatHex(newOffset);
|
|
|
|
const actual = formatHex(this.offset);
|
|
|
|
throw new Error(`readVariantString failed: offset does mismatch (expect: ${expected}, actual: ${actual})`);
|
2023-06-12 23:26:30 +00:00
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
public static toStringMap(view: DataView): Map<string, string> {
|
|
|
|
const mmkv = new MMKVParser(view);
|
|
|
|
const result = new Map<string, string>();
|
|
|
|
while (!mmkv.eof) {
|
|
|
|
const key = mmkv.readString();
|
|
|
|
const value = mmkv.readVariantString();
|
|
|
|
result.set(key, value);
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
2023-06-17 01:45:31 +00:00
|
|
|
|
|
|
|
public static parseKuwoEKey(_view: DataView): unknown[] {
|
|
|
|
return [];
|
|
|
|
}
|
2023-06-12 23:26:30 +00:00
|
|
|
}
|