Skip to content

Instantly share code, notes, and snippets.

@BruJu
Created May 14, 2021 07:27
Show Gist options
  • Star 0 You must be signed in to star a gist
  • Fork 0 You must be signed in to fork a gist
  • Save BruJu/c1641fc0b1e68a52ec57b7b325ad4f22 to your computer and use it in GitHub Desktop.
Save BruJu/c1641fc0b1e68a52ec57b7b325ad4f22 to your computer and use it in GitHub Desktop.
StoreAlterer.js
"use strict";
//! I was too lazy to learn how to use another Triple Pattern Matching and
//! Replacement library, let alone find one that is not async await heavy,
//! so I built one...
const DataFactory = require('n3').DataFactory;
const variable = DataFactory.variable;
/**
* Converts a term into :
* - Its value if it is a bound variable
* - `undefined` if it is an unbound variable
* - The term itself if every other cases.
*
* Quads that contains variables have their components converted.
*
* @param {*} term The term to convert
* @param {*} binded A mapping variable -> known value
*/
function forMatch(term, binded) {
if (term.termType == "Quad") {
let s = forMatch(term.subject, binded);
let p = forMatch(term.predicate, binded);
let o = forMatch(term.object, binded);
let g = term.graph;
return DataFactory.quad(s, p, o, g);
}
if (term.termType !== "Variable") return term;
return binded[term.value];
}
/**
* If `searched` is an RDF.Variable, binds the value of `searched`
* to `result` in the `bindings` mapping.
* @param {*} bindings The mapping of known bound values.
* @param {*} searched The term from the pattern.
* @param {*} result The term returned by the match operation.
*/
function addBind(bindings, searched, result) {
if (searched.termType === "Variable") {
bindings[searched.value] = result;
}
}
/**
* Replace all quads found using a pattern from the store by a new pattern.
* The new pattern is a list of triple pattern, that can use either fixed terms
* or variables that were present in the request pattern.
*
* @param {*} store The store to modify
* @param {*} foundBindings The result of `matchAndBind`
* @param {*} newPattern The new pattern that is used to replaces the matched
* quads.
*/
function replace(store, foundBindings, newPattern) {
let r = [];
for (let binds of foundBindings) {
r.push(replaceOneBinding(store, binds, newPattern));
}
return r;
}
function replaceOneBinding(store, binds, newPattern) {
store.removeQuads(binds["@quads"]);
const r = { "binds": binds, "quads": [] };
for (const newPattern1 of newPattern) {
const newQuad = DataFactory.quad(
forMatch(newPattern1[0], binds),
forMatch(newPattern1[1], binds),
forMatch(newPattern1[2], binds),
);
r.quads.push(newQuad);
store.addQuad(newQuad);
}
return r;
}
function deleteMatches(store, s, p, o) {
let x = "_";
function m(term) {
if (term == null || term == undefined) {
x = x + "_";
return DataFactory.variable(x);
} else {
return term;
}
}
let request = matchAndBind(store, [[m(s), m(p), m(o)]]);
replace(store, request, []);
}
function _matchAndBind(store, patterns, iPattern, results) {
if (iPattern == patterns.length) {
return results;
}
const pattern = patterns[iPattern];
const newResults = [];
for (const oldResult of results) {
const quads = store.getQuads(
forMatch(pattern[0], oldResult),
forMatch(pattern[1], oldResult),
forMatch(pattern[2], oldResult)
);
for (let quad of quads) {
const r = { "@quads": [...oldResult["@quads"], quad] };
for (let x in oldResult) {
if (x === "@quads") continue;
r[x] = oldResult[x];
}
addBind(r, pattern[0], quad.subject);
addBind(r, pattern[1], quad.predicate);
addBind(r, pattern[2], quad.object);
newResults.push(r);
}
}
return _matchAndBind(store, patterns, iPattern + 1, newResults);
}
/**
* Search the given pattern in the store and returns the list of bindable
* values for each variable.
*
* The pattern is a list of sub patterns. A sub pattern is an array of three
* RDF.JS terms. The terms can be either proper RDF terms or variables.
*
*
* Returns a list of dictionaries in the form:
* {
* "@quad": list of involved quads,
* variableName: the binded quad for each variable
* }
*
* @param {*} store The store
* @param {*} pattern The pattern, a list of arrays of 3 terms.
*/
function matchAndBind(store, pattern) {
return _matchAndBind(store, pattern, 0, [ { "@quads": [] } ]);
}
function mapPattern(bind, patterns) {
let x = patterns.map(
pattern => pattern.map(
term => {
if (term.termType === "Variable") {
const variableName = term.value;
if (bind[variableName] === undefined) {
return term;
} else {
return bind[variableName];
}
} else {
return term;
}
}
)
);
return x;
}
function findFilterReplace(store, source, conditions, destination) {
let binds = matchAndBind(store, source);
binds = binds.filter(bind => {
const mappedConditions = conditions.map(pattern => mapPattern(bind, pattern));
for (let condition of mappedConditions) {
if (matchAndBind(store, condition).length === 0) {
return false;
}
}
return true;
});
replace(store, binds, destination);
return binds;
}
module.exports = {
replaceOneBinding: replaceOneBinding,
deleteMatches: deleteMatches,
matchAndBind: matchAndBind,
findFilterReplace: findFilterReplace,
mapPattern: mapPattern // only used in tests
};
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment