commit
e9a3b2fe88
@ -0,0 +1,340 @@ |
|||||||
|
import { describe, it, expect } from "vitest"; |
||||||
|
import { deepSignal, DeepPatch, DeepSignalOptions } from "../deepSignal"; |
||||||
|
import { watch } from "../watch"; |
||||||
|
|
||||||
|
describe("deepSignal options", () => { |
||||||
|
describe("custom ID generator", () => { |
||||||
|
it("uses custom ID generator for objects without @id", async () => { |
||||||
|
let counter = 1000; |
||||||
|
const options: DeepSignalOptions = { |
||||||
|
idGenerator: () => `custom-${counter++}`, |
||||||
|
addIdToObjects: true, |
||||||
|
}; |
||||||
|
|
||||||
|
const state = deepSignal({ data: {} as any }, options); |
||||||
|
const patches: DeepPatch[][] = []; |
||||||
|
const { stopListening: stop } = watch(state, ({ patches: batch }) => |
||||||
|
patches.push(batch) |
||||||
|
); |
||||||
|
|
||||||
|
state.data.user = { name: "Alice" }; |
||||||
|
await Promise.resolve(); |
||||||
|
|
||||||
|
// Check that @id was assigned
|
||||||
|
expect((state.data.user as any)["@id"]).toBe("custom-1000"); |
||||||
|
|
||||||
|
// Check that patch was emitted for @id
|
||||||
|
const flat = patches.flat().map((p) => p.path.join(".")); |
||||||
|
expect(flat).toContain("data.user.@id"); |
||||||
|
|
||||||
|
stop(); |
||||||
|
}); |
||||||
|
|
||||||
|
it("respects existing @id on objects", async () => { |
||||||
|
const options: DeepSignalOptions = { |
||||||
|
idGenerator: () => "should-not-be-used", |
||||||
|
addIdToObjects: true, |
||||||
|
}; |
||||||
|
|
||||||
|
const state = deepSignal({ items: [] as any[] }, options); |
||||||
|
|
||||||
|
state.items.push({ "@id": "existing-123", value: 42 }); |
||||||
|
|
||||||
|
// Should use the existing @id
|
||||||
|
expect((state.items[0] as any)["@id"]).toBe("existing-123"); |
||||||
|
}); |
||||||
|
|
||||||
|
it("uses @id property from objects added to Sets", async () => { |
||||||
|
const options: DeepSignalOptions = { |
||||||
|
idGenerator: () => "fallback-id", |
||||||
|
addIdToObjects: true, |
||||||
|
}; |
||||||
|
|
||||||
|
const state = deepSignal({ s: new Set<any>() }, options); |
||||||
|
const patches: DeepPatch[][] = []; |
||||||
|
const { stopListening: stop } = watch(state, ({ patches: batch }) => |
||||||
|
patches.push(batch) |
||||||
|
); |
||||||
|
|
||||||
|
const obj = { "@id": "set-entry-1", data: "test" }; |
||||||
|
state.s.add(obj); |
||||||
|
|
||||||
|
await Promise.resolve(); |
||||||
|
|
||||||
|
const flat = patches.flat().map((p) => p.path.join(".")); |
||||||
|
// Path should use the @id as synthetic key
|
||||||
|
expect(flat.some((p) => p.startsWith("s.set-entry-1"))).toBe(true); |
||||||
|
|
||||||
|
stop(); |
||||||
|
}); |
||||||
|
}); |
||||||
|
|
||||||
|
describe("addIdToObjects option", () => { |
||||||
|
it("adds @id to all nested objects when enabled", async () => { |
||||||
|
const options: DeepSignalOptions = { |
||||||
|
addIdToObjects: true, |
||||||
|
}; |
||||||
|
|
||||||
|
const state = deepSignal({ root: {} as any }, options); |
||||||
|
const patches: DeepPatch[][] = []; |
||||||
|
const { stopListening: stop } = watch(state, ({ patches: batch }) => |
||||||
|
patches.push(batch) |
||||||
|
); |
||||||
|
|
||||||
|
state.root.level1 = { |
||||||
|
level2: { |
||||||
|
level3: { value: "deep" }, |
||||||
|
}, |
||||||
|
}; |
||||||
|
|
||||||
|
await Promise.resolve(); |
||||||
|
|
||||||
|
// Check all levels have @id
|
||||||
|
expect((state.root.level1 as any)["@id"]).toBeDefined(); |
||||||
|
expect((state.root.level1.level2 as any)["@id"]).toBeDefined(); |
||||||
|
expect( |
||||||
|
(state.root.level1.level2.level3 as any)["@id"] |
||||||
|
).toBeDefined(); |
||||||
|
|
||||||
|
// Check patches were emitted for all @id fields
|
||||||
|
const flat = patches.flat().map((p) => p.path.join(".")); |
||||||
|
expect(flat).toContain("root.level1.@id"); |
||||||
|
expect(flat).toContain("root.level1.level2.@id"); |
||||||
|
expect(flat).toContain("root.level1.level2.level3.@id"); |
||||||
|
|
||||||
|
stop(); |
||||||
|
}); |
||||||
|
|
||||||
|
it("does not add @id when option is false", () => { |
||||||
|
const state = deepSignal({ data: { nested: {} } }); |
||||||
|
|
||||||
|
// Should not have @id
|
||||||
|
expect("@id" in (state.data as any)).toBe(false); |
||||||
|
expect("@id" in (state.data.nested as any)).toBe(false); |
||||||
|
}); |
||||||
|
|
||||||
|
it("adds @id to objects in arrays", async () => { |
||||||
|
const options: DeepSignalOptions = { |
||||||
|
addIdToObjects: true, |
||||||
|
}; |
||||||
|
|
||||||
|
const state = deepSignal({ items: [] as any[] }, options); |
||||||
|
const patches: DeepPatch[][] = []; |
||||||
|
const { stopListening: stop } = watch(state, ({ patches: batch }) => |
||||||
|
patches.push(batch) |
||||||
|
); |
||||||
|
|
||||||
|
state.items.push({ name: "Item 1" }, { name: "Item 2" }); |
||||||
|
|
||||||
|
await Promise.resolve(); |
||||||
|
|
||||||
|
// Both items should have @id
|
||||||
|
expect((state.items[0] as any)["@id"]).toBeDefined(); |
||||||
|
expect((state.items[1] as any)["@id"]).toBeDefined(); |
||||||
|
|
||||||
|
// Check patches
|
||||||
|
const flat = patches.flat().map((p) => p.path.join(".")); |
||||||
|
expect(flat).toContain("items.0.@id"); |
||||||
|
expect(flat).toContain("items.1.@id"); |
||||||
|
|
||||||
|
stop(); |
||||||
|
}); |
||||||
|
|
||||||
|
it("adds @id to objects in Sets", async () => { |
||||||
|
const options: DeepSignalOptions = { |
||||||
|
idGenerator: () => |
||||||
|
`gen-${Math.random().toString(36).substr(2, 9)}`, |
||||||
|
addIdToObjects: true, |
||||||
|
}; |
||||||
|
|
||||||
|
const state = deepSignal({ s: new Set<any>() }, options); |
||||||
|
const patches: DeepPatch[][] = []; |
||||||
|
const { stopListening: stop } = watch(state, ({ patches: batch }) => |
||||||
|
patches.push(batch) |
||||||
|
); |
||||||
|
|
||||||
|
const obj1 = { value: 1 }; |
||||||
|
const obj2 = { value: 2 }; |
||||||
|
state.s.add(obj1); |
||||||
|
state.s.add(obj2); |
||||||
|
|
||||||
|
await Promise.resolve(); |
||||||
|
|
||||||
|
// Get proxied objects from Set
|
||||||
|
const proxiedObjs = Array.from(state.s); |
||||||
|
expect((proxiedObjs[0] as any)["@id"]).toBeDefined(); |
||||||
|
expect((proxiedObjs[1] as any)["@id"]).toBeDefined(); |
||||||
|
|
||||||
|
// @id should be used as synthetic key in paths
|
||||||
|
const flat = patches.flat().map((p) => p.path.join(".")); |
||||||
|
const obj1Id = (proxiedObjs[0] as any)["@id"]; |
||||||
|
const obj2Id = (proxiedObjs[1] as any)["@id"]; |
||||||
|
expect(flat.some((p) => p.startsWith(`s.${obj1Id}`))).toBe(true); |
||||||
|
expect(flat.some((p) => p.startsWith(`s.${obj2Id}`))).toBe(true); |
||||||
|
|
||||||
|
stop(); |
||||||
|
}); |
||||||
|
}); |
||||||
|
|
||||||
|
describe("@id property behavior", () => { |
||||||
|
it("makes @id readonly", () => { |
||||||
|
const options: DeepSignalOptions = { |
||||||
|
addIdToObjects: true, |
||||||
|
}; |
||||||
|
|
||||||
|
const state = deepSignal({ obj: {} as any }, options); |
||||||
|
state.obj.data = { value: 1 }; |
||||||
|
|
||||||
|
// Attempting to modify @id should throw
|
||||||
|
expect(() => { |
||||||
|
(state.obj.data as any)["@id"] = "new-id"; |
||||||
|
}).toThrow("Cannot modify readonly property '@id'"); |
||||||
|
}); |
||||||
|
|
||||||
|
it("makes @id enumerable", () => { |
||||||
|
const options: DeepSignalOptions = { |
||||||
|
addIdToObjects: true, |
||||||
|
}; |
||||||
|
|
||||||
|
const state = deepSignal({ obj: {} as any }, options); |
||||||
|
state.obj.data = { value: 1 }; |
||||||
|
|
||||||
|
// @id should show up in Object.keys()
|
||||||
|
const keys = Object.keys(state.obj.data); |
||||||
|
expect(keys).toContain("@id"); |
||||||
|
}); |
||||||
|
|
||||||
|
it("emits patches for @id even on objects with existing @id", async () => { |
||||||
|
const options: DeepSignalOptions = { |
||||||
|
addIdToObjects: true, |
||||||
|
}; |
||||||
|
|
||||||
|
const state = deepSignal({ container: {} as any }, options); |
||||||
|
const patches: DeepPatch[][] = []; |
||||||
|
const { stopListening: stop } = watch(state, ({ patches: batch }) => |
||||||
|
patches.push(batch) |
||||||
|
); |
||||||
|
|
||||||
|
// Object already has @id before being added
|
||||||
|
const objWithId = { "@id": "pre-existing", data: "test" }; |
||||||
|
state.container.item = objWithId; |
||||||
|
|
||||||
|
await Promise.resolve(); |
||||||
|
|
||||||
|
const flat = patches.flat().map((p) => p.path.join(".")); |
||||||
|
// Patch should still be emitted for @id
|
||||||
|
expect(flat).toContain("container.item.@id"); |
||||||
|
|
||||||
|
// Verify the value in the patch
|
||||||
|
const idPatch = patches |
||||||
|
.flat() |
||||||
|
.find((p) => p.path.join(".") === "container.item.@id"); |
||||||
|
expect((idPatch as any).value).toBe("pre-existing"); |
||||||
|
|
||||||
|
stop(); |
||||||
|
}); |
||||||
|
}); |
||||||
|
|
||||||
|
describe("options inheritance", () => { |
||||||
|
it("child objects inherit options from root", async () => { |
||||||
|
let idCounter = 5000; |
||||||
|
const options: DeepSignalOptions = { |
||||||
|
idGenerator: () => `inherited-${idCounter++}`, |
||||||
|
addIdToObjects: true, |
||||||
|
}; |
||||||
|
|
||||||
|
const state = deepSignal({ root: {} as any }, options); |
||||||
|
|
||||||
|
// Add nested structure
|
||||||
|
state.root.child = { |
||||||
|
grandchild: { |
||||||
|
value: "nested", |
||||||
|
}, |
||||||
|
}; |
||||||
|
|
||||||
|
// All should have IDs generated by the custom generator
|
||||||
|
expect((state.root.child as any)["@id"]).toMatch(/^inherited-/); |
||||||
|
expect((state.root.child.grandchild as any)["@id"]).toMatch( |
||||||
|
/^inherited-/ |
||||||
|
); |
||||||
|
}); |
||||||
|
|
||||||
|
it("objects added to Sets inherit options", async () => { |
||||||
|
let counter = 9000; |
||||||
|
const options: DeepSignalOptions = { |
||||||
|
idGenerator: () => `set-child-${counter++}`, |
||||||
|
addIdToObjects: true, |
||||||
|
}; |
||||||
|
|
||||||
|
const state = deepSignal({ s: new Set<any>() }, options); |
||||||
|
|
||||||
|
const obj = { nested: { value: 1 } }; |
||||||
|
state.s.add(obj); |
||||||
|
|
||||||
|
// Iterate to get proxied object
|
||||||
|
const proxied = Array.from(state.s)[0]; |
||||||
|
|
||||||
|
// Object and nested object should have custom IDs
|
||||||
|
expect((proxied as any)["@id"]).toMatch(/^set-child-/); |
||||||
|
expect((proxied.nested as any)["@id"]).toMatch(/^set-child-/); |
||||||
|
}); |
||||||
|
}); |
||||||
|
|
||||||
|
describe("backward compatibility", () => { |
||||||
|
it("still works without options", async () => { |
||||||
|
const state = deepSignal({ data: { value: 1 } }); |
||||||
|
const patches: DeepPatch[][] = []; |
||||||
|
const { stopListening: stop } = watch(state, ({ patches: batch }) => |
||||||
|
patches.push(batch) |
||||||
|
); |
||||||
|
|
||||||
|
state.data.value = 2; |
||||||
|
await Promise.resolve(); |
||||||
|
|
||||||
|
expect(patches.flat().length).toBeGreaterThan(0); |
||||||
|
stop(); |
||||||
|
}); |
||||||
|
|
||||||
|
// TODO: Delete duplicate logic for `id`. Only accept @id.
|
||||||
|
it("objects with id property still work for Sets", async () => { |
||||||
|
const state = deepSignal({ s: new Set<any>() }); |
||||||
|
const patches: DeepPatch[][] = []; |
||||||
|
const { stopListening: stop } = watch(state, ({ patches: batch }) => |
||||||
|
patches.push(batch) |
||||||
|
); |
||||||
|
|
||||||
|
state.s.add({ id: "legacy-id", value: 1 }); |
||||||
|
await Promise.resolve(); |
||||||
|
|
||||||
|
const flat = patches.flat().map((p) => p.path.join(".")); |
||||||
|
// Should use id as synthetic key
|
||||||
|
expect(flat.some((p) => p.startsWith("s.legacy-id"))).toBe(true); |
||||||
|
|
||||||
|
stop(); |
||||||
|
}); |
||||||
|
|
||||||
|
it("@id takes precedence over id property", async () => { |
||||||
|
const state = deepSignal({ s: new Set<any>() }); |
||||||
|
const patches: DeepPatch[][] = []; |
||||||
|
const { stopListening: stop } = watch(state, ({ patches: batch }) => |
||||||
|
patches.push(batch) |
||||||
|
); |
||||||
|
|
||||||
|
state.s.add({ |
||||||
|
id: "should-not-use", |
||||||
|
"@id": "should-use", |
||||||
|
value: 1, |
||||||
|
}); |
||||||
|
await Promise.resolve(); |
||||||
|
|
||||||
|
const flat = patches.flat().map((p) => p.path.join(".")); |
||||||
|
// Should use @id, not id
|
||||||
|
expect(flat.some((p) => p.startsWith("s.should-use"))).toBe(true); |
||||||
|
expect(flat.some((p) => p.startsWith("s.should-not-use"))).toBe( |
||||||
|
false |
||||||
|
); |
||||||
|
|
||||||
|
stop(); |
||||||
|
}); |
||||||
|
}); |
||||||
|
}); |
@ -0,0 +1,217 @@ |
|||||||
|
import type { Diff as Patches, Scope } from "../types.ts"; |
||||||
|
import { applyDiff } from "./applyDiff.ts"; |
||||||
|
|
||||||
|
import * as NG from "@ng-org/lib-wasm"; |
||||||
|
|
||||||
|
import { |
||||||
|
deepSignal, |
||||||
|
watch as watchDeepSignal, |
||||||
|
batch, |
||||||
|
} from "@ng-org/alien-deepsignals"; |
||||||
|
import type { |
||||||
|
DeepPatch, |
||||||
|
DeepSignalObject, |
||||||
|
WatchPatchCallback, |
||||||
|
WatchPatchEvent, |
||||||
|
} from "@ng-org/alien-deepsignals"; |
||||||
|
import type { ShapeType, BaseType } from "@ng-org/shex-orm"; |
||||||
|
|
||||||
|
export class OrmConnection<T extends BaseType> { |
||||||
|
// TODO: WeakMaps?
|
||||||
|
private static idToEntry = new Map<string, OrmConnection<any>>(); |
||||||
|
|
||||||
|
private ng: typeof NG; |
||||||
|
readonly shapeType: ShapeType<T>; |
||||||
|
readonly scope: Scope; |
||||||
|
readonly signalObject: DeepSignalObject<T | {}>; |
||||||
|
private refCount: number; |
||||||
|
/*** Identifier as a combination of shape type and scope. Prevents duplications. */ |
||||||
|
private identifier: string; |
||||||
|
ready: boolean; |
||||||
|
sessionId: number; |
||||||
|
suspendDeepWatcher: boolean; |
||||||
|
readyPromise: Promise<void>; |
||||||
|
// Promise that resolves once initial data has been applied.
|
||||||
|
resolveReady!: () => void; |
||||||
|
|
||||||
|
// FinalizationRegistry to clean up connections when signal objects are GC'd.
|
||||||
|
private static cleanupSignalRegistry = |
||||||
|
typeof FinalizationRegistry === "function" |
||||||
|
? new FinalizationRegistry<string>((connectionId) => { |
||||||
|
// Best-effort fallback; look up by id and clean
|
||||||
|
const entry = this.idToEntry.get(connectionId); |
||||||
|
if (!entry) return; |
||||||
|
entry.release(); |
||||||
|
}) |
||||||
|
: null; |
||||||
|
|
||||||
|
private constructor(shapeType: ShapeType<T>, scope: Scope, ng: typeof NG) { |
||||||
|
this.shapeType = shapeType; |
||||||
|
this.scope = scope; |
||||||
|
this.ng = ng; |
||||||
|
this.refCount = 0; |
||||||
|
this.ready = false; |
||||||
|
this.suspendDeepWatcher = false; |
||||||
|
this.identifier = `${shapeType.shape}::${canonicalScope(scope)}`; |
||||||
|
this.signalObject = deepSignal<T | {}>(new Set(), { |
||||||
|
addIdToObjects: true, |
||||||
|
idGenerator: this.generateSubjectIri, |
||||||
|
}); |
||||||
|
|
||||||
|
// TODO:
|
||||||
|
this.sessionId = 1; |
||||||
|
|
||||||
|
// Schedule cleanup of the connection when the signal object is GC'd.
|
||||||
|
OrmConnection.cleanupSignalRegistry?.register( |
||||||
|
this.signalObject, |
||||||
|
this.identifier, |
||||||
|
this.signalObject |
||||||
|
); |
||||||
|
|
||||||
|
// Add listener to deep signal object to report changes back to wasm land.
|
||||||
|
watchDeepSignal(this.signalObject as T, this.onSignalObjectUpdate); |
||||||
|
|
||||||
|
// Initialize per-entry readiness promise that resolves in setUpConnection
|
||||||
|
this.readyPromise = new Promise<void>((resolve) => { |
||||||
|
this.resolveReady = resolve; |
||||||
|
}); |
||||||
|
|
||||||
|
// Establish connection to wasm land.
|
||||||
|
ng.orm_start(scope, shapeType, this.sessionId, this.onBackendMessage); |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* Get a connection which contains the ORM and lifecycle methods. |
||||||
|
* @param shapeType |
||||||
|
* @param scope |
||||||
|
* @param ng |
||||||
|
* @returns |
||||||
|
*/ |
||||||
|
public static getConnection<T extends BaseType>( |
||||||
|
shapeType: ShapeType<T>, |
||||||
|
scope: Scope, |
||||||
|
ng: typeof NG |
||||||
|
): OrmConnection<T> { |
||||||
|
const scopeKey = canonicalScope(scope); |
||||||
|
|
||||||
|
// Unique identifier for a given shape type and scope.
|
||||||
|
const identifier = `${shapeType.shape}::${scopeKey}`; |
||||||
|
|
||||||
|
// If we already have an object for this shape+scope,
|
||||||
|
// return it and just increase the reference count.
|
||||||
|
// Otherwise, create new one.
|
||||||
|
const connection = |
||||||
|
OrmConnection.idToEntry.get(identifier) ?? |
||||||
|
new OrmConnection(shapeType, scope, ng); |
||||||
|
|
||||||
|
connection.refCount += 1; |
||||||
|
|
||||||
|
return connection; |
||||||
|
} |
||||||
|
|
||||||
|
public release() { |
||||||
|
if (this.refCount > 0) this.refCount--; |
||||||
|
if (this.refCount === 0) { |
||||||
|
OrmConnection.idToEntry.delete(this.identifier); |
||||||
|
|
||||||
|
OrmConnection.cleanupSignalRegistry?.unregister(this.signalObject); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
private onSignalObjectUpdate({ patches }: WatchPatchEvent<T>) { |
||||||
|
if (this.suspendDeepWatcher || !this.ready || !patches.length) return; |
||||||
|
|
||||||
|
const ormPatches = deepPatchesToDiff(patches); |
||||||
|
|
||||||
|
this.ng.orm_update( |
||||||
|
this.scope, |
||||||
|
this.shapeType.shape, |
||||||
|
ormPatches, |
||||||
|
this.sessionId |
||||||
|
); |
||||||
|
} |
||||||
|
|
||||||
|
private onBackendMessage(...message: any) { |
||||||
|
this.handleInitialResponse(message); |
||||||
|
} |
||||||
|
|
||||||
|
private handleInitialResponse(...param: any) { |
||||||
|
console.log("RESPONSE FROM BACKEND", param); |
||||||
|
|
||||||
|
// TODO: This will break, just provisionary.
|
||||||
|
const wasmMessage: WasmMessage = param; |
||||||
|
const { initialData } = wasmMessage; |
||||||
|
|
||||||
|
// Assign initial data to empty signal object without triggering watcher at first.
|
||||||
|
this.suspendDeepWatcher = true; |
||||||
|
batch(() => { |
||||||
|
// Convert arrays to sets and apply to signalObject (we only have sets but can only transport arrays).
|
||||||
|
Object.assign(this.signalObject, recurseArrayToSet(initialData)!); |
||||||
|
}); |
||||||
|
|
||||||
|
queueMicrotask(() => { |
||||||
|
this.suspendDeepWatcher = false; |
||||||
|
// Resolve readiness after initial data is committed and watcher armed.
|
||||||
|
this.resolveReady?.(); |
||||||
|
}); |
||||||
|
|
||||||
|
this.ready = true; |
||||||
|
} |
||||||
|
private onBackendUpdate(...params: any) { |
||||||
|
// Apply diff
|
||||||
|
} |
||||||
|
|
||||||
|
/** Function to create random subject IRIs for newly created nested objects. */ |
||||||
|
private generateSubjectIri(path: (string | number)[]): string { |
||||||
|
// Generate random string.
|
||||||
|
let b = Buffer.alloc(33); |
||||||
|
crypto.getRandomValues(b); |
||||||
|
const randomString = b.toString("base64url"); |
||||||
|
|
||||||
|
if (path.length > 0 && path[0].toString().startsWith("did:ng:o:")) { |
||||||
|
// If the root is a nuri, use that as a base IRI.
|
||||||
|
let rootNuri = path[0] as string; |
||||||
|
|
||||||
|
return rootNuri.substring(0, 9 + 44) + ":q:" + randomString; |
||||||
|
} else { |
||||||
|
// Else, just generate a random IRI.
|
||||||
|
return "did:ng:q:" + randomString; |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
//
|
||||||
|
//
|
||||||
|
|
||||||
|
function escapePathSegment(segment: string): string { |
||||||
|
return segment.replace("~", "~0").replace("/", "~1"); |
||||||
|
} |
||||||
|
|
||||||
|
export function deepPatchesToDiff(patches: DeepPatch[]): Patches { |
||||||
|
return patches.map((patch) => { |
||||||
|
const path = |
||||||
|
"/" + |
||||||
|
patch.path.map((el) => escapePathSegment(el.toString())).join("/"); |
||||||
|
return { ...patch, path }; |
||||||
|
}) as Patches; |
||||||
|
} |
||||||
|
|
||||||
|
const recurseArrayToSet = (obj: any): any => { |
||||||
|
if (Array.isArray(obj)) { |
||||||
|
return new Set(obj.map(recurseArrayToSet)); |
||||||
|
} else if (obj && typeof obj === "object") { |
||||||
|
for (const key of Object.keys(obj)) { |
||||||
|
obj[key] = recurseArrayToSet(obj[key]); |
||||||
|
} |
||||||
|
return obj; |
||||||
|
} else { |
||||||
|
return obj; |
||||||
|
} |
||||||
|
}; |
||||||
|
|
||||||
|
function canonicalScope(scope: Scope | undefined): string { |
||||||
|
if (scope == null) return ""; |
||||||
|
return Array.isArray(scope) |
||||||
|
? scope.slice().sort().join(",") |
||||||
|
: String(scope); |
||||||
|
} |
Loading…
Reference in new issue