LDO for Raw RDF Guide
LDO (Linked Data Objects) is a library that lets you easily manipulate RDF as if it were a standard TypeScript object that follows a ShEx shape you define.
This tutorial will walk you through using LDO on raw RDF. How that raw RDF is fetched is left up the developer.
Setup
Automatic Setup
To setup LDO, cd
into your typescript project and run npx @ldo/cli init
.
Manual Setup
The following is handled by the automatic setup: Install the LDO dependencies. Create a folder to store your ShEx shapes: Create a script to build ShEx shapes and convert them into Linked Data Objects. You can put this script in `package.json`Creating ShEx Schemas
LDO uses ShEx as a schema for the RDF data in your project. To add a ShEx schema to your project, simply create a file ending in .shex
to the shapes
folder.
For more information on writing ShEx schemas see the ShEx Primer.
./shapes/foafProfile.shex
:
PREFIX ex: <http://example.com/>
PREFIX foaf: <http://xmlns.com/foaf/0.1/>
PREFIX rdfs: <http://www.w3.org/2000/01/rdf-schema#>
PREFIX xsd: <http://www.w3.org/2001/XMLSchema#>
ex:FoafProfile EXTRA a {
a [ foaf:Person ]
// rdfs:comment "Defines the node as a Person (from foaf)" ;
foaf:name xsd:string ?
// rdfs:comment "Define a person's name." ;
foaf:img xsd:string ?
// rdfs:comment "Photo link but in string form" ;
foaf:knows @ex:FoafProfile *
// rdfs:comment "A list of WebIds for all the people this user knows." ;
}
To build the shape, run:
This will generate five files:
- ./.ldo/foafProfile.shapeTypes.ts
<-- This is the important file
- ./.ldo/foafProfile.typings.ts
- ./.ldo/foafProfile.schema.ts
- ./.ldo/foafProfile.context.ts
Simple Example
Below is a simple example of LDO in a real use-case (changing the name on a Solid Pod)
import { parseRdf, startTransaction, toSparqlUpdate, toTurtle } from "@ldo/ldo";
import { FoafProfileShapeType } from "./.ldo/foafProfile.shapeTypes";
async function run() {
const rawTurtle = `
<#me> a <http://xmlns.com/foaf/0.1/Person>;
<http://xmlns.com/foaf/0.1/name> "Jane Doe".
`;
/**
* Step 1: Convert Raw RDF into a Linked Data Object
*/
const ldoDataset = await parseRdf(rawTurtle, {
baseIRI: "https://solidweb.me/jane_doe/profile/card",
});
// Create a linked data object by telling the dataset the type and subject of
// the object
const janeProfile = ldoDataset
// Tells the LDO dataset that we're looking for a FoafProfile
.usingType(FoafProfileShapeType)
// Says the subject of the FoafProfile
.fromSubject("https://solidweb.me/jane_doe/profile/card#me");
/**
* Step 2: Manipulate the Linked Data Object
*/
// Logs "Jane Doe"
console.log(janeProfile.name);
// Logs "Person"
console.log(janeProfile.type);
// Logs 0
console.log(janeProfile.knows?.length);
// Begins a transaction that tracks your changes
startTransaction(janeProfile);
janeProfile.name = "Jane Smith";
janeProfile.knows?.push({
"@id": "https://solidweb.me/john_smith/profile/card#me",
type: {
"@id": "Person",
},
name: "John Smith",
knows: [janeProfile],
});
// Logs "Jane Smith"
console.log(janeProfile.name);
// Logs "John Smith"
console.log(janeProfile.knows?.[0].name);
// Logs "Jane Smith"
console.log(janeProfile.knows?.[0].knows?.[0].name);
/**
* Step 3: Convert it back to RDF
*/
// Logs:
// <https://solidweb.me/jane_doe/profile/card#me> a <http://xmlns.com/foaf/0.1/Person>;
// <http://xmlns.com/foaf/0.1/name> "Jane Smith";
// <http://xmlns.com/foaf/0.1/knows> <https://solidweb.me/john_smith/profile/card#me>.
// <https://solidweb.me/john_smith/profile/card#me> a <http://xmlns.com/foaf/0.1/Person>;
// <http://xmlns.com/foaf/0.1/name> "John Smith";
// <http://xmlns.com/foaf/0.1/knows> <https://solidweb.me/jane_doe/profile/card#me>.
console.log(await toTurtle(janeProfile));
// Logs:
// DELETE DATA {
// <https://solidweb.me/jane_doe/profile/card#me> <http://xmlns.com/foaf/0.1/name> "Jane Doe" .
// };
// INSERT DATA {
// <https://solidweb.me/jane_doe/profile/card#me> <http://xmlns.com/foaf/0.1/name> "Jane Smith" .
// <https://solidweb.me/jane_doe/profile/card#me> <http://xmlns.com/foaf/0.1/knows> <https://solidweb.me/john_smith/profile/card#me> .
// <https://solidweb.me/john_smith/profile/card#me> <http://xmlns.com/foaf/0.1/name> "John Smith" .
// <https://solidweb.me/john_smith/profile/card#me> <http://www.w3.org/1999/02/22-rdf-syntax-ns#type> <http://xmlns.com/foaf/0.1/Person> .
// <https://solidweb.me/john_smith/profile/card#me> <http://xmlns.com/foaf/0.1/knows> <https://solidweb.me/jane_doe/profile/card#me> .
// }
console.log(await toSparqlUpdate(janeProfile));
}
run();
Getting an LDO Dataset
An LDO Dataset is a kind of RDF JS Dataset that can create linked data objects.
LDO datasets can be created in two ways:
createLdoDataset(initialDataset?: Dataset<Quad, Quad> | Quad[])
initialDataset
: An optional dataset or array of quads for the new dataset.
parseRdf(data: string, parserOptions?: ParserOptions)
import { parseRdf } from "@ldo/ldo";
const rawTurtle = "...";
const ldoDataset = await parseRdf(rawTurtle, { baseIRI: "http://example.com/" });
data
: The raw data to parse as astring
.options
(optional): Parse options containing the following keys:format
(optional): The format the data is in. The following are acceptable formats:Turtle
,TriG
,N-Triples
,N-Quads
,N3
,Notation3
.baseIRI
(optional): If this data is hosted at a specific location, you can provide the baseIRI of that location.blankNodePrefix
(optional): If blank nodes should have a prefix, that should be provided here.factory
(optional): a RDF Data Factory from@rdfjs/data-model
.
Getting a Linked Data Object
Once you have an LdoDataset we can get a Linked Data Object. A linked data object feels just like a JavaScript object literal, but when you make modifications to it, it will affect the underlying LdoDataset.
Thie first step is defining which Shape Type you want to retrieve from the dataset. We can use the generated shape types and the usingType()
method for this.
import { FoafProfileShapeType } from "./.ldo/foafProfile.shapeTypes.ts";
// ... Get the LdoDataset
ldoDataset.usingType(FoafProfileShapeType);
Next, we want to identify exactly what part of the dataset we want to extract. We can do this in a few ways:
.fromSubject(entryNode)
fromSubject
lets you define a an entryNode
, the place of entry for the graph. The object returned by jsonldDatasetProxy
will represent the given node. This parameter accepts both namedNode
s and blankNode
s. fromSubject
takes a generic type representing the typescript type of the given subject.
const profile = ldoDataset
.usingType(FoafProfileShapeType)
.fromSubject("http://example.com/Person1");
.matchSubject(predicate?, object?, graph?)
matchSubject
returns a Jsonld Dataset Proxy representing all subjects in the dataset matching the given predicate, object, and graph.
const profiles = ldoDataset
.usingType(FoafProfileShapeType)
.matchSubject(
namedNode("http://www.w3.org/1999/02/22-rdf-syntax-ns#type"),
namedNode("http://xmlns.com/foaf/0.1/Person")
);
profiles.forEach((person) => {
console.log(person.fn);
});
.matchObject(subject?, predicate?, object?)
matchObject
returns a Jsonld Dataset Proxy representing all objects in the dataset matching the given subject, predicate, and graph.
const friendsOfPerson1 = ldoDataset
.usingType(FoafProfileShapeType)
.matchSubject(
namedNode("http://example.com/Person1"),
namedNode("http://xmlns.com/foaf/0.1/knows")
);
friendsOfPerson1.forEach((person) => {
console.log(person.fn);
});
.fromJson(inputData)
fromJson
will take any regular Json, add the information to the dataset, and return a Jsonld Dataset Proxy representing the given data.
const person2 = ldoDataset
.usingType(FoafProfileShapeType)
.fromJson({
"@id": "http://example.com/Person2",
fn: ["Jane Doe"],
});
Getting and Setting Data on a Linked Data Object
Once you've created a Linked Data Object, you can get and set data as if it were a normal TypeScript Object. For specific details, see the documentation at JSONLD Dataset Proxy.
import { LinkedDataObject } from "@ldo/ldo";
import { FoafProfileFactory } from "./.ldo/foafProfile.ldoFactory.ts";
import { FoafProfile } from "./.ldo/foafProfile.typings";
aysnc function start() {
const profile: FoafProfile = // Create LDO
// Logs "Aang"
console.log(profile.name);
// Logs "Person"
console.log(profile.type["@id"]);
// Logs 1
console.log(profile.knows?.length);
// Logs "Katara"
console.log(profile.knows?.[0].name);
profile.name = "Bonzu Pippinpaddleopsicopolis III"
// Logs "Bonzu Pippinpaddleopsicopolis III"
console.log(profile.name);
profile.knows?.push({
type: { "@id": "Person" },
name: "Sokka"
});
// Logs 2
console.log(profile.knows?.length);
// Logs "Katara" and "Sokka"
profile.knows?.forEach((person) => console.log(person.name));
}
Converting a Linked Data Object back to RDF
A linked data object can be converted into RDF in multiple ways:
toTurtle(linkedDataObject)
toNTiples(linkedDataObject)
serialize(linkedDataObject, options)
import { serialize } from "@ldo/ldo";
const rawTurtle: string = await serialize({
format: "Turtle",
prefixes: {
ex: "http://example.com/",
foaf: "http://xmlns.com/foaf/0.1/",
}
});
serialize(linkedDataObject, options)
provides general serialization based on provided options:
- foramt
(optional): The format to serialize to. The following are acceptable formats: Turtle
, TriG
, N-Triples
, N-Quads
, N3
, Notation3
.
- prefixes
: The prefixes for those serializations that use prefixes.
Transactions
Sometimes, you want to keep track of changes you make for the object. This is where transactions come in handy.
To start a transaction, use the startTransaction(linkedDataObject)
function. From then on, all transactions will be tracked, but not added to the original ldoDataset. You can view the changes using the transactionChanges(linkedDataObject)
or toSparqlUpdate(linkedDataObject)
methods. When you're done with the transaction, you can run the commitTransaction(linkedDataObject)
method to add the changes to the original ldoDataset.
import {
startTransaction,
transactionChanges,
toSparqlUpdate,
commitTransaction,
} from "@ldo/ldo";
// ... Get the profile linked data object
startTransaction(profile);
profile.name = "Kuzon"
const changes = transactionChanges(profile);
// Logs: <http://example.com/aang> <http://xmlns.com/foaf/0.1/name> "Kuzon"
console.log(changes.added?.toString())
// Logs: <http://example.com/aang> <http://xmlns.com/foaf/0.1/name> "Aang"
console.log(changes.removed?.toString())
console.log(await toSparqlUpdate(profile));
commitTransaction(profile);
Other LDO Helper Functions
getDataset(linkedDataObject)
Returns the Linked Data Object's underlying RDFJS dataset. Modifying this dataset will change the Linked Data Object as well.