silverbullet/plug-api/lib/json.ts

142 lines
3.3 KiB
TypeScript

/**
* Performs a deep comparison of two objects, returning true if they are equal
* @param a first object
* @param b second object
* @returns
*/
export function deepEqual(a: any, b: any): boolean {
if (a === b) {
return true;
}
if (typeof a !== typeof b) {
return false;
}
if (a === null || b === null) {
return false;
}
if (a === undefined || b === undefined) {
return false;
}
if (typeof a === "object") {
if (Array.isArray(a) && Array.isArray(b)) {
if (a.length !== b.length) {
return false;
}
for (let i = 0; i < a.length; i++) {
if (!deepEqual(a[i], b[i])) {
return false;
}
}
return true;
} else {
const aKeys = Object.keys(a);
const bKeys = Object.keys(b);
if (aKeys.length !== bKeys.length) {
return false;
}
for (const key of aKeys) {
if (!deepEqual(a[key], b[key])) {
return false;
}
}
return true;
}
}
return false;
}
/**
* Converts a Date object to a date string in the format YYYY-MM-DD if it just contains a date (and no significant time), or a full ISO string otherwise
* @param d the date to convert
*/
export function cleanStringDate(d: Date): string {
// If no significant time, return a date string only
if (
d.getUTCHours() === 0 && d.getUTCMinutes() === 0 && d.getUTCSeconds() === 0
) {
return d.getFullYear() + "-" +
String(d.getMonth() + 1).padStart(2, "0") + "-" +
String(d.getDate()).padStart(2, "0");
} else {
return d.toISOString();
}
}
/**
* Processes a JSON (typically coming from parse YAML frontmatter) in two ways:
* 1. Expands property names in an object containing a .-separated path
* 2. Converts dates to strings in sensible ways
* @param a
* @returns
*/
export function cleanupJSON(a: any): any {
if (!a) {
return a;
}
if (typeof a !== "object") {
return a;
}
if (Array.isArray(a)) {
return a.map(cleanupJSON);
}
// If a is a date, convert to a string
if (a instanceof Date) {
return cleanStringDate(a);
}
const expanded: any = {};
for (const key of Object.keys(a)) {
const parts = key.split(".");
let target = expanded;
for (let i = 0; i < parts.length - 1; i++) {
const part = parts[i];
if (!target[part]) {
target[part] = {};
}
target = target[part];
}
target[parts[parts.length - 1]] = cleanupJSON(a[key]);
}
return expanded;
}
/**
* Performs a deep merge of two objects, with b taking precedence over a
* @param a
* @param b
* @returns
*/
export function deepObjectMerge(a: any, b: any, reverseArrays = false): any {
if (typeof a !== typeof b) {
return b;
}
if (a === undefined || a === null) {
return b;
}
if (b === undefined || b === null) {
return a;
}
if (typeof a === "object") {
if (Array.isArray(a) && Array.isArray(b)) {
if (reverseArrays) {
return [...b, ...a];
} else {
return [...a, ...b];
}
} else {
const aKeys = Object.keys(a);
const bKeys = Object.keys(b);
const merged = { ...a };
for (const key of bKeys) {
if (aKeys.includes(key)) {
merged[key] = deepObjectMerge(a[key], b[key], reverseArrays);
} else {
merged[key] = b[key];
}
}
return merged;
}
}
return b;
}