A minimal Graph database backed by IndexedDB modeled after Neo4j's with a Cypher Query Language (CQL) like syntax.
This project does not intend to implement a complete set of the CQL, with some additions specific to this library. Just a subset that is just enough to get the job done.
While working on a project, I needed a store resources in the browser, the kind of data that can't be stored with localStorage
. I looked out for packages and found levelgraph and Dexie, but they were not a good fit for the kind of data I was dealing with.
I was dealing with connected peices of data, it was like squeezing a square peg into a round hole. I soon found myself fighting with the DB and I felt like I was just hacking my way around it.
What other way to handle storing of data that's connected like a graph, other than a Graph database. Having worked with Neo4j on the backend, I found it difficult doing things that would be trivial if I were using a graph database. For example, I have a list of users - A, B, C, I want to be able to say:
- User A is a friend of User B and C.
- User B is a friend of User A and C.
To be able to handle such a scenario using the existing solutions in the market, I'd have to save an array/table with the id of each target user tied to the main user I want to relate them with, e.g The above example would be handled like so:
- User A: [User B, User, C].
- User B: [User A, User, C].
Notice the duplicating of reference, which would require array manipulations just to get something that trivial done. But you might say, why not use a relational database like lovefield. You see, the thing is, after using a Graph database for most of my projects on the backend, I don't think I can ever get to make my brain think in relational databases again.
Once you see it, you can't unsee it.
pnpm add mercurydb
import { q, assign, Mercury } from "mercurydb";
const db = new Mercury("test", 1);
const Employees = db.model("Employee", {
age: "number",
name: "string",
email: {
unique: true,
indexed: true,
type: "string",
default: "[email protected]",
},
});
const Employers = db.model("Employer", {
address: "number",
name: {
unique: true,
indexed: true,
type: "string",
},
regNo: {
unique: true,
indexed: true,
type: "string",
default: () => uuid(),
},
});
db.onUpgrade(async ({ schema }) => {
await schema.install();
});
await db.connect();
- The brackets denote nodes, while the square brackets denote a relationship.
- The
e
is optional, but is neccessary if you want to do things like RETURN, DELETE, WHERE, SET and ORDERBY. - Specify the first/start node (e.g (e:Employee)) is neccessary, while the rest are optional.
// This query matches all `Employee`s.
const createQuery = q`CREATE``(e:Employee)``[]``()`;
// This query matches only `Employee`s that have a relationship of type employed.
const createQuery = q`CREATE``(e:Employee)``[:EMPLOYED_BY]``()`;
/**
* This query matches only nodes labeled `Employee`, that have a relationship
* of `EMPLOYED_BY` to another node labeled `Employer`. In other words, this
* query matches only `Employee`s that are employed by a certain employer.
*/
const createQuery = q`CREATE``(e:Employee ${employee})``[:EMPLOYED_BY]``(:Employer ${employer})`;
The above concepts apply to other query types, with just a difference in operator. So lets take a look at other query types.
const createQuery = q`CREATE``(e:Employee)``[:EMPLOYED_BY]``(:Employer)`;
const matchQuery = q`MATCH``(e:Employee)``[:EMPLOYED_BY]``(:Employer)`;
Create a relationship between two existing nodes.
const relateQuery = q`RELATE``(e:Employee)``[:EMPLOYED_BY]``(:Employer)`;
Merge tries to match the full pattern and merges its contents, and also creates the full pattern if no match is found.
const mergeQuery = q`MERGE``(e:Employee)``[:EMPLOYED_BY]``(:Employer)`;
const createQuery = q`CREATE``(e:Employee)``[:EMPLOYED_BY]``(:Employer)`;
The relationship goes from Employee
to Employer
. Then doing something like this:
const matchQuery = q`MATCH``(e:Employer)``[:EMPLOYED_BY]``(:Employee)`;
won't work. The match query relationship goes from Employer
to Employee
which is not the form in which it was created.
- set
- skip
- limit
- where
- delete
- return
- orderBy
- rawLimit (limit the cursor over the database)
const createQuery = q` CREATE``(e:Employee)``[:EMPLOYED_BY]``(emp:Employer) `;
const { e } = await db.exec(createQuery, {
return: "e",
return: ["e"],
return: ["e", "emp.regNo AS regNo"],
return: ["e", ["emp.regNo", "regNo"]],
skip: 2,
limit: 10,
rawLimit: 5,
delete: ["e", "emp"],
orderBy: {
type: "DESC",
key: "e.name",
},
where: ({ name }) => name.startsWith("Segun"),
set: {
e: assign({ name: "Arinze" }),
e: assign({ name: () => "Arinze" }),
e: assign(() => ({ name: "Arinze" })),
},
// specific to merge queries
onMatch: {
e: assign({ name: "Arinze" }),
},
onCreate: {
e: assign({ name: "Segun Arinze" }),
},
});
const employee = {
name: "Segun Arinze",
age: Math.floor(Math.random() * 10),
};
const employer = {
name: "Self",
address: "Earth",
};
const createQuery = q`CREATE``(e:Employee ${employee})``[:EMPLOYED_BY]``(:Employer ${employer})`;
const { e } = await db.exec(createQuery, { return: ["e"] });
Equivalent of an update query
const mergeQuery = q`MERGE``(e:Employee ${{ ...e, age: 50 }})``[]``()`;
const mergeRes = await db.exec(mergeQuery, {
onCreate: {
e: assign({ age: 70 }),
},
});
There are two options for a match query. This, but this would only work if the objects have an indexed field. So its advisable to index searchable fields. And this also makes the query faster.
const matchQuery = q`MATCH``(e:Employee ${employee})``[:EMPLOYED_BY]``(:Employer ${employer})`;
const matchRes = await db.exec(matchQuery, { return: ["e"] });
OR
This approach can get quite slow as the number of items in the database increases, because every item with the label Employee
and Employer
will be traversed.
const matchQuery = q`MATCH``(e:Employee)``[:EMPLOYED_BY]``(:Employer)`;
const matchRes = await db.exec(matchQuery, {
where({ name }) {
return name === "John Doe";
},
return: ["e"],
});
const createQuery1 = q`CREATE``(e:Employee ${employee})``[]``()`;
const createQuery2 = q`CREATE``(e:Employer ${employer})``[]``()`;
const [createRes1, createRes2] = await db.batch([createQuery1, createQuery2], {
return: "e",
});
const relateQuery = q`RELATE``(:Employee ${createRes1.e})``[:EMPLOYED_BY]``(:Employer ${createRes2.e})`;
await db.exec(relateQuery);
- Validation.
- Better query mechanism.
- Relationship direction.
- DELETE operator.
- OPTIONAL MATCH operator.