DEV Community

Diyorbek Sobirov
Diyorbek Sobirov

Posted on

Library for storing graphs in files and making queries to them.πŸ”—

The file-graph class provides an abstract implementation for graph operations. It supports graph manipulation, including creating, updating, and deleting vertices, creating arcs and edges, and traversing or querying the graph structure.
https://github.com/DIY0R/file-graph


Installation

To install file-graph from npm, run the following command:

npm install file-graph
Enter fullscreen mode Exit fullscreen mode

Alternatively, you can install it from the GitHub registry.

Creating a Graph Instance

const graph = FileGraph('graph.txt');
Enter fullscreen mode Exit fullscreen mode

FileGraph accepts a path where the graph data will be stored.The returned object graph is an instance of the FileGraphAbstract class, which provides methods for working with the graph.

File Structure

After initializing the graph at the specified path, a file will be created where each line represents a vertex. It is recommended not to modify the file contents manually.

Example of a vertex record: graph.txt

{
  "id": "33ef29be-adaa-4509-b306-62a32db7310e", // UUID - unique identifier of the vertex
  "data": { ... }, // The data associated with the vertex
  "links": [] // List of UUIDs of vertices that this vertex is connected to
}
Enter fullscreen mode Exit fullscreen mode

Public API - Asynchronous Methods

Below is an overview of the methods, their parameters, return values, and examples of how to use them.

createVertex<T extends object>(data: T): Promise<IVertex<T>>

Creates a vertex (node) in the graph with the provided data.

  • Parameters:
    • data: T - An object representing the vertex data.
  • Returns:
    • A promise that resolves to the created vertex object { id, data, links }.

Example:

const data = { name: 'Alice', age: 30 };
const createdVertex = await graph.createVertex(data);
console.log(createdVertex);
// Output: { id: 'some-unique-id', data: { name: 'Alice', age: 30 }, links: [] }
Enter fullscreen mode Exit fullscreen mode

createVertices<T extends object>(data: T[]): Promise<IVertex<T>[]>

Creates multiple vertices at once.

  • Parameters:
    • data: T[] - An array of objects representing the data for each vertex.
  • Returns:
    • A promise that resolves to an array of created vertex objects.

Example:

const data = [{ name: 'Alice' }, { name: 'Bob' }];
const createdVertices = await graph.createVertices(data);
console.log(createdVertices);
// Output: [{ id: 'id1', data: { name: 'Alice' }, links: [] }, { id: 'id2', data: { name: 'Bob' }, links: [] }]
Enter fullscreen mode Exit fullscreen mode

updateVertex<T extends object>(updater: IUpdater<T>): Promise<boolean>

Updates a vertex that matches the given condition. The updater function defines the logic for finding and modifying the vertex.

  • Parameters:
    • updater: IUpdater<T> - A function that takes a vertex and returns an updated vertex if it matches the condition.
  • Returns:
    • A promise that resolves to true if the update was successful, otherwise false.

Example:

const isUpdated = await graph.updateVertex(
  vertex =>
    vertex.id === 'some-unique-id' && { data: { name: 'Alice Updated' } },
);
console.log(isUpdated); // true
Enter fullscreen mode Exit fullscreen mode

deleteVertex<T extends object>(predicate: IPredicate<T>): Promise<boolean>

Deletes a vertex that matches the given condition.

  • Parameters:
    • predicate: IPredicate<T> - A function that returns true if the vertex should be deleted.
  • Returns:
    • A promise that resolves to true if the deletion was successful, otherwise false.

Example:

const isDeleted = await graph.deleteVertex(
  vertex => vertex.id === 'some-unique-id',
);
console.log(isDeleted); // true
Enter fullscreen mode Exit fullscreen mode

findOne<T extends object>(predicate: IPredicate<T>): Promise<IVertex<T> | null>

Finds a single vertex that matches the given condition.

  • Parameters:
    • predicate: IPredicate<T> - A function that returns true if the vertex matches the search condition.
  • Returns:
    • A promise that resolves to the matching vertex object, or null if no match is found.

Example:

const foundVertex = await graph.findOne(vertex => vertex.data.name === 'Alice');
console.log(foundVertex);
// Output: { id: 'some-unique-id', data: { name: 'Alice', age: 30 }, links: [] }
Enter fullscreen mode Exit fullscreen mode

findAll<T extends object>(predicate: IPredicate<T>): Promise<IVertex<T>[]>

Finds all vertices that match the given condition.

  • Parameters:
    • predicate: IPredicate<T> - A function that returns true for each vertex that matches the search condition.
  • Returns:
    • A promise that resolves to an array of matching vertex objects.

Example:

const foundVertices = await graph.findAll(
  vertex => vertex.data.name === 'Alice',
);
console.log(foundVertices);
// Output: [{ id: 'id1', data: { name: 'Alice', age: 30 }, links: [] }, { id: 'id2', data: { name: 'Alice', age: 25 }, links: [] }]
Enter fullscreen mode Exit fullscreen mode

forEachVertex<T extends object>(callbackVertex: ICallbackVertex<T>): Promise<void>

Iterates over each vertex in the graph and applies the provided callback function.

  • Parameters:
    • callbackVertex: ICallbackVertex<T> - A callback function that is invoked for each vertex. If it returns true, the iteration stops.
  • Returns:
    • A promise that resolves when the iteration is complete.

Example:

await graph.forEachVertex(vertex => {
  console.log(vertex);
  // Stop iteration if the vertex's name is 'Alice'
  return vertex.data.name === 'Alice';
});
Enter fullscreen mode Exit fullscreen mode

createEdge(ids: IUuidArray): Promise<boolean>

Creates edges (links) between the specified vertices.

  • Parameters:
    • ids: IUuidArray - An array of vertex IDs between which to create edges.
  • Returns:
    • A promise that resolves to true if the edge creation was successful.

Example:

const isEdgeCreated = await graph.createEdge(['id1', 'id2', 'id3']);
console.log(isEdgeCreated); // true
Enter fullscreen mode Exit fullscreen mode

removeEdge(ids: IUuidArray): Promise<boolean>

Removes the edges (links) between the specified vertices in the graph.

  • Parameters

    • ids: IUuidArray - An array of vertex IDs. The edges between the vertices specified by these IDs will be removed.
  • Returns

    • A promise that resolves to true if the edge removal was successful.

Example:

const isEdgeRemoved = await graph.removeEdge(['id1', 'id2', 'id3']);
console.log(isEdgeRemoved); // true
Enter fullscreen mode Exit fullscreen mode

createArc(sourceVertexId: uuidType, targetVertexId: uuidType): Promise<boolean>

Creates an arc (directed edge) between two vertices.

  • Parameters:
    • sourceVertexId: uuidType - The ID of the source vertex.
    • targetVertexId: uuidType - The ID of the target vertex.
  • Returns:
    • A promise that resolves to true if the arc creation was successful.

Example:

const isArcCreated = await graph.createArc('id1', 'id2');
console.log(isArcCreated); // true
Enter fullscreen mode Exit fullscreen mode

createArcs(ids: IUuidArray): Promise<boolean>

Creates arcs (directed edges) between multiple vertices in the specified order.

  • Parameters:
    • ids: IUuidArray - An array of vertex IDs.
  • Returns:
    • A promise that resolves to true if the arcs creation was successful.

Example:

const isArcsCreated = await graph.createArcs(['id1', 'id2', 'id3']);
console.log(isArcsCreated); // true
Enter fullscreen mode Exit fullscreen mode

removeArc(sourceVertexId: uuidType, targetVertexId: uuidType): Promise<boolean>

Removes an arc (edge) between two vertices.

  • Parameters:
    • sourceVertexId: uuidType - The ID of the source vertex.
    • targetVertexId: uuidType - The ID of the target vertex.
  • Returns:
    • A promise that resolves to true if the arc was successfully removed.

Example:

const isArcRemoved = await graph.removeArc('id1', 'id2');
console.log(isArcRemoved); // true
Enter fullscreen mode Exit fullscreen mode

hasArc(sourceVertexId: uuidType, targetVertexId: uuidType): Promise<boolean>

Checks if an arc exists between two vertices.

  • Parameters:
    • sourceVertexId: uuidType - The ID of the source vertex.
    • targetVertexId: uuidType - The ID of the target vertex.
  • Returns:
    • A promise that resolves to true if the arc exists, otherwise false.

Example:

const hasArc = await graph.hasArc('id1', 'id2');
console.log(hasArc); // true or false
Enter fullscreen mode Exit fullscreen mode

findUpToLevel<T extends object>(vertexId: uuidType, maxLevel?: number): Promise<IVertexTree<T>[]>

Retrieves vertices up to a specified depth level from a starting vertex.

  • Parameters:
    • vertexId: uuidType - The ID of the starting vertex.
    • maxLevel?: number - (Optional) The depth level to limit the search.
  • Returns:
    • A promise that resolves to an array of vertices up to the specified level.

Example:

const graphTree = await graph.findUpToLevel('id1', 2);
console.log(graphTree);
/* Output: [
{ id: 'id1',data:{...},links:[...],level: 0 }, 
{ id: 'id2',data:{...},links:[...], level: 1 }, 
{ id: 'id3', data:{...},links:[...],level: 2 }
 ];  */
Enter fullscreen mode Exit fullscreen mode

searchVerticesFrom<T extends object>(vertexId: uuidType, predicate: IPredicate<T>): Promise<IVertex<T>[]>

Performs a search starting from the given vertex and returns vertices that match the predicate.

  • Parameters:
    • vertexId: uuidType - The ID of the starting vertex.
    • predicate: IPredicate<T> - A function used to evaluate each vertex. Only vertices that satisfy the predicate will be returned.
  • Returns:
    • A promise that resolves to an array of matching vertices.

Example:

const matchingVertices = await graph.searchVerticesFrom(
  'id1',
  vertex => vertex.data.age > 25,
);
console.log(matchingVertices);
// Output: [{ id: 'id2', data: { name: 'Alice', age: 30 }, links: [] }]
Enter fullscreen mode Exit fullscreen mode

hasPath(sourceVertexId: uuidType, targetVertexId: uuidType): Promise<boolean>

Checks if a path exists between two vertices using Depth-First Search (DFS).

  • Parameters:
    • sourceVertexId: uuidType - The ID of the source vertex.
    • targetVertexId: uuidType - The ID of the target vertex.
  • Returns:
    • A promise that resolves to true if a path exists between the vertices, otherwise false.

Example:

const pathExists = await graph.hasPath('id1', 'id3');
console.log(pathExists); // true or false
Enter fullscreen mode Exit fullscreen mode

Top comments (0)