All original tests working

main
Jackson Morgan 8 months ago
parent 563a0976ad
commit 2655acd893
  1. 3
      packages/jsonld-dataset-proxy/src/ProxyContext.ts
  2. 7
      packages/jsonld-dataset-proxy/src/setProxy/ObjectSetProxy.ts
  3. 20
      packages/jsonld-dataset-proxy/src/setProxy/SetProxy.ts
  4. 21
      packages/jsonld-dataset-proxy/src/setProxy/SubjectSetProxy.ts
  5. 102
      packages/jsonld-dataset-proxy/src/setProxy/WildcardObjectSetProxy.ts
  6. 24
      packages/jsonld-dataset-proxy/src/setProxy/WildcardSubjectSetProxy.ts
  7. 2
      packages/jsonld-dataset-proxy/test/jsonldDatasetProxy.test.ts

@ -68,7 +68,7 @@ export class ProxyContext {
public createSetProxy( public createSetProxy(
quadMatch: QuadMatch, quadMatch: QuadMatch,
isSubjectOriented?: boolean, isSubjectOriented?: boolean,
_isLangStringSet?: boolean, isLangStringSet?: boolean,
): SetProxy { ): SetProxy {
const key = this.getSetKey(...quadMatch); const key = this.getSetKey(...quadMatch);
if (!this.setMap.has(key)) { if (!this.setMap.has(key)) {
@ -76,6 +76,7 @@ export class ProxyContext {
quadMatch, quadMatch,
isSubjectOriented ?? false, isSubjectOriented ?? false,
this, this,
isLangStringSet,
); );
this.setMap.set(key, proxy); this.setMap.set(key, proxy);
} }

@ -59,9 +59,8 @@ export class ObjectSetProxy<
*/ */
delete(value: T): boolean { delete(value: T): boolean {
const { dataset } = this.context; const { dataset } = this.context;
const { subject, predicate, object, graph } = this.getSPOG(value); const quads = this.getQuads(value);
const didDelete = dataset.match(subject, predicate, object, graph).size > 0; quads.forEach((quad) => dataset.delete(quad));
dataset.deleteMatches(subject, predicate, object, graph); return quads.size > 0;
return didDelete;
} }
} }

@ -6,7 +6,6 @@ import type { Dataset, Quad } from "@rdfjs/types";
import type { import type {
GraphNode, GraphNode,
ObjectNode, ObjectNode,
PredicateNode,
QuadMatch, QuadMatch,
SubjectNode, SubjectNode,
} from "@ldo/rdf-utils"; } from "@ldo/rdf-utils";
@ -43,12 +42,7 @@ export abstract class SetProxy<
/** /**
* Gets the subject, predicate and object for this set * Gets the subject, predicate and object for this set
*/ */
protected abstract getSPOG(value?: T): { protected abstract getQuads(value?: T): Dataset<Quad, Quad>;
subject?: SubjectNode;
predicate?: PredicateNode;
object?: ObjectNode;
graph?: GraphNode;
};
protected abstract getNodeOfFocus(quad: Quad): SubjectNode | ObjectNode; protected abstract getNodeOfFocus(quad: Quad): SubjectNode | ObjectNode;
@ -86,15 +80,11 @@ export abstract class SetProxy<
} }
has(value: T): boolean { has(value: T): boolean {
const { dataset } = this.context; return this.getQuads(value).size > 0;
const { subject, predicate, object, graph } = this.getSPOG(value);
return dataset.match(subject, predicate, object, graph).size > 0;
} }
get size() { get size() {
const { dataset } = this.context; return this.getQuads().size;
const { subject, predicate, object, graph } = this.getSPOG();
return dataset.match(subject, predicate, object, graph).size;
} }
entries(): IterableIterator<[T, T]> { entries(): IterableIterator<[T, T]> {
@ -114,9 +104,7 @@ export abstract class SetProxy<
} }
[Symbol.iterator](): IterableIterator<T> { [Symbol.iterator](): IterableIterator<T> {
const { dataset } = this.context; const quads = this.getQuads();
const { subject, predicate, object, graph } = this.getSPOG();
const quads = dataset.match(subject, predicate, object, graph);
const collection: T[] = quads.toArray().map((quad) => { const collection: T[] = quads.toArray().map((quad) => {
const quadSubject = this.getNodeOfFocus(quad); const quadSubject = this.getNodeOfFocus(quad);
return nodeToJsonldRepresentation(quadSubject, this.context) as T; return nodeToJsonldRepresentation(quadSubject, this.context) as T;

@ -86,4 +86,25 @@ export class SubjectSetProxy<
}); });
return this; return this;
} }
/**
* Clears the set of all values
*/
clear(): void {
for (const value of this) {
this.delete(value);
}
}
/**
* Deletes an item for the set
* @param value the item to delete
* @returns true if the item was present before deletion
*/
delete(value: T): boolean {
const { dataset } = this.context;
const quads = this.getQuads(value);
quads.forEach((quad) => dataset.delete(quad));
return quads.size > 0;
}
} }

@ -10,6 +10,7 @@ import { SetProxy } from "./SetProxy";
import type { ProxyContext } from "../ProxyContext"; import type { ProxyContext } from "../ProxyContext";
import { getNodeFromRawValue } from "../util/getNodeFromRaw"; import { getNodeFromRawValue } from "../util/getNodeFromRaw";
import { _isSubjectOriented } from "../types"; import { _isSubjectOriented } from "../types";
import { filterQuadsByLanguageOrdering } from "../language/languageUtils";
export type WildcardObjectSetProxyQuadMatch = [ export type WildcardObjectSetProxyQuadMatch = [
SubjectNode | undefined | null, SubjectNode | undefined | null,
@ -27,25 +28,22 @@ export class WildcardObjectSetProxy<
T extends NonNullable<RawValue>, T extends NonNullable<RawValue>,
> extends SetProxy<T> { > extends SetProxy<T> {
protected quadMatch: WildcardObjectSetProxyQuadMatch; protected quadMatch: WildcardObjectSetProxyQuadMatch;
protected isLangSet: boolean; protected isLangStringSet: boolean;
constructor( constructor(
context: ProxyContext, context: ProxyContext,
quadMatch: WildcardObjectSetProxyQuadMatch, quadMatch: WildcardObjectSetProxyQuadMatch,
isLangSet?: boolean, isLangStringSet?: boolean,
) { ) {
super(context, quadMatch); super(context, quadMatch);
this.quadMatch = quadMatch; this.quadMatch = quadMatch;
this.isLangSet = isLangSet ?? false; this.isLangStringSet = isLangStringSet ?? false;
} }
protected getSPOG(value?: T | undefined): { protected getQuads(value?: T | undefined): Dataset<Quad, Quad> {
subject?: SubjectNode; const { dataset } = this.context;
predicate?: PredicateNode; let quads: Dataset<Quad, Quad>;
object?: ObjectNode; // Get the RDF Node that represents the value, skip if no value
graph?: GraphNode;
} {
// Get the RDF Node that represents the value, skip is no value
const subject = this.quadMatch[0] ?? undefined; const subject = this.quadMatch[0] ?? undefined;
const predicate = this.quadMatch[1] ?? undefined; const predicate = this.quadMatch[1] ?? undefined;
const graph = this.quadMatch[3] ?? undefined; const graph = this.quadMatch[3] ?? undefined;
@ -58,81 +56,31 @@ export class WildcardObjectSetProxy<
datatype = this.context.contextUtil.getDataType(key, rdfType); datatype = this.context.contextUtil.getDataType(key, rdfType);
} }
const valueNode = getNodeFromRawValue(value, this.context, datatype); const valueNode = getNodeFromRawValue(value, this.context, datatype);
return { quads = dataset.match(subject, predicate, valueNode, graph);
subject, // If there is no valueNode, we must filter by value manually as we
predicate, // weren't able to deduce the datatype.
object: valueNode, if (!valueNode) {
graph, quads = quads.filter(
};
}
// SPO for no value
return {
subject,
predicate,
object: undefined,
graph,
};
}
protected getNodeOfFocus(quad: Quad): ObjectNode {
return quad.object as ObjectNode;
}
private manuallyMatchWithUnknownObjectNode(
subject: SubjectNode | undefined,
predicate: PredicateNode | undefined,
graph: GraphNode | undefined,
value: T,
): Dataset<Quad, Quad> {
// If there's not an object, that means that we don't know the object node
// and need to find it manually.
const matchingQuads = this.context.dataset.match(
subject,
predicate,
null,
graph,
);
return matchingQuads.filter(
(quad) => (quad) =>
quad.object.termType === "Literal" && quad.object.value === value, quad.object.termType === "Literal" && quad.object.value === value,
); );
} }
delete(value: T): boolean {
const { dataset } = this.context;
const { subject, predicate, object, graph } = this.getSPOG(value);
if (!object) {
const matchedQuads = this.manuallyMatchWithUnknownObjectNode(
subject,
predicate,
graph,
value,
);
matchedQuads.forEach((quad) => dataset.delete(quad));
return matchedQuads.size > 0;
} else { } else {
const willDelete = // SPO for no value
dataset.match(subject, predicate, object, graph).size > 0; quads = dataset.match(subject, predicate, undefined, graph);
dataset.deleteMatches(subject, predicate, object, graph);
return willDelete;
}
} }
// If this is a langStringSet, filter by language preferences
has(value: T): boolean { if (this.isLangStringSet) {
const { dataset } = this.context; return filterQuadsByLanguageOrdering(
const { subject, predicate, object, graph } = this.getSPOG(value); quads,
if (!object) { this.context.languageOrdering,
return (
this.manuallyMatchWithUnknownObjectNode(
subject,
predicate,
graph,
value,
).size > 0
); );
} else {
return dataset.match(subject, predicate, object, graph).size > 0;
} }
return quads;
}
protected getNodeOfFocus(quad: Quad): ObjectNode {
return quad.object as ObjectNode;
} }
get [_isSubjectOriented](): false { get [_isSubjectOriented](): false {

@ -4,7 +4,7 @@ import type {
ObjectNode, ObjectNode,
GraphNode, GraphNode,
} from "@ldo/rdf-utils"; } from "@ldo/rdf-utils";
import type { Quad } from "@rdfjs/types"; import type { Dataset, Quad } from "@rdfjs/types";
import type { RawObject } from "../util/RawObject"; import type { RawObject } from "../util/RawObject";
import { SetProxy } from "./SetProxy"; import { SetProxy } from "./SetProxy";
import type { ProxyContext } from "../ProxyContext"; import type { ProxyContext } from "../ProxyContext";
@ -34,32 +34,18 @@ export class WildcardSubjectSetProxy<T extends RawObject> extends SetProxy<T> {
this.quadMatch = quadMatch; this.quadMatch = quadMatch;
} }
protected getSPOG(value?: T | undefined): { protected getQuads(value?: T | undefined): Dataset<Quad, Quad> {
subject?: SubjectNode; const { dataset } = this.context;
predicate?: PredicateNode;
object?: ObjectNode;
graph?: GraphNode;
} {
// Get the RDF Node that represents the value, skip is no value // Get the RDF Node that represents the value, skip is no value
const predicate = this.quadMatch[1] ?? undefined; const predicate = this.quadMatch[1] ?? undefined;
const object = this.quadMatch[2] ?? undefined; const object = this.quadMatch[2] ?? undefined;
const graph = this.quadMatch[3] ?? undefined; const graph = this.quadMatch[3] ?? undefined;
if (value) { if (value) {
const valueNode = getNodeFromRawObject(value, this.context.contextUtil); const valueNode = getNodeFromRawObject(value, this.context.contextUtil);
return { return dataset.match(valueNode, predicate, object, graph);
subject: valueNode,
predicate,
object,
graph,
};
} }
// SPO for no value // SPO for no value
return { return dataset.match(undefined, predicate, object, graph);
subject: undefined,
predicate,
object,
graph,
};
} }
protected getNodeOfFocus(quad: Quad): SubjectNode { protected getNodeOfFocus(quad: Quad): SubjectNode {

@ -1264,7 +1264,7 @@ const testJsonldDatasetProxy = (patientContext: LdoJsonldContext) => () => {
it("Removes an object completely when using the delete method", async () => { it("Removes an object completely when using the delete method", async () => {
const firstPatient = patients.toArray()[0]; const firstPatient = patients.toArray()[0];
patients.delete(firstPatient); patients.delete(firstPatient);
expect(patients.has(firstPatient)).toBe(true); expect(patients.has(firstPatient)).toBe(false);
}); });
it("creates a collection that matches only collections in a certain graph", async () => { it("creates a collection that matches only collections in a certain graph", async () => {

Loading…
Cancel
Save