Skip to content

Latest commit

 

History

History
229 lines (153 loc) · 6.06 KB

README.md

File metadata and controls

229 lines (153 loc) · 6.06 KB

JavaScript Entity Graph

An in-memory graph database for JavaScript data.

Status

Moderately used in at least one real product. Expected to become more widely deployed and battle tested real soon.

Inspiration

The most direct inspiration is DataScript, which is in turn inspired by Datomic. Like DataScript, but unlike Datomic, this "database" does not offer durability of any kind.

Further inspiration comes from Facebook's Relay and Netflix's Falcor. Unlike either of these, this project does not attempt to address any networking and service challenges.

Lastly, this project is inspired by Om Next and discussions with its creator, David Nolen.

Background

We've already got a growing set of JSON/REST APIs, so we can't easily switch everything to a Relay or Falcor style service endpoint overnight.

Our frontend is already written in JavaScript, utilizing React.js; ClojureScript is too large of a leap for our team at this time.

We need something that's, above all else, simple, but acts as a stepping stone along the path towards frontend nirvana.

Goals

  • Client-side, in-memory only.
    • Assume dataset is small enough to traverse portions of it many times.
  • Plain-old JavaScript objects.
    • Hierarchical flattening of the graph.
    • Not necessarily just JSON (allow dates, etc).
    • Encourages use via destructuring.
    • All the standard debugging, printing, etc tools should work.
  • Graph-based information model.
    • Strong support for relationships between entities.
    • Allow navigation from any node as a root, in any direction.
  • No spooky action at a distance.
    • Every database operation makes an implicit defensive copy.
    • Good enough compared to real immutability.
    • Safe to directly return to callers of your store API.

Non-Goals

  • Be a Flux "store" by itself.
    • No one general purpose information model can address all domain needs.
    • You still need an API for mutations anyway.
  • Persistent Data Structure
    • We don't need undo or anything like that.
    • The debugging benefits are nice, but just gravy.
  • Serializablity
    • We already need to reshape data from our APIs.
    • Solve durability, caching, and transmission at another layer.
  • Query
    • Since dataset is small, assume it's OK to aggressively over-satisfy gets.
    • If you need filtering, use alternative or additional data structure in your store.
  • Change Notifications
    • Wrap with domain-level store API and implement your own coarse-grain notifications for key entities.

API

import Database from 'jseg';
let db = new Database(schema);

The only field provided by the default schema, lid, is required. It is short for "Local ID" and is named such to differentiate it from other application specific identifiers. The recommended name for server-specified identifiers is "gid", short for "Global ID".

See below for methods of db and schema details.

See the test file for many concrete examples.

get(lid, options)

Gets a whole tree of related objects by lid. The options parameter may be omitted.

Does not traverse in to cycles. If the maxDepth option is specified, will not recurse more than that many levels.

Null field values and empty collections are omitted.

Always returns an object, with at least a lid field.

put(entity)

Puts a whole tree of related objects. Properties are merged in to existing objects with matching lid fields. Collection properties are set-unioned.

Fields set to null are deleted from entities.

lookup(field, value, options)

Gets an object by a unique field string value. See schema.

The options parameter may be omitted and is delegated to get.

Returns null if no entity exists.

destroy(lid)

Removes an object from the database by lid. Recurses as per schema.

remove(parentLid, field, childLid)

Removes a related object from a reference collection field.

Also works on non-collection reference fields. Treats the field as a collection with a max size of one. Equivalent to setting the field to null.

lids()

Returns an array of all lids for all objects in the database.

Schema

Just a map of named fields to config.

Entity Identity

The lid property is required for all get/put operations. It's just a string.

Scalar Fields

By default, fields may contain scalar values. These are typically strings and numbers, but any JavaScript non-null, non-undefined object is allowed.

Unique Lookup

unique: true

Use on string fields to enable O(1) indexing for use by lookup.

Validation

The validate property specifies a function to validate and transform a scalar value. Throw an exception to report a validation error or return the transformed value.

Validation errors are logged and invalid fields are discarded.

validate: function(value) {
  if (!valid(value)) {
    throw validationError;
  }
  return coerce(value);
}

Entity References

ref: 'reverse'

Specifies which fields are references to other objects, and those object's reverse relationship field name. Neither, either, or both ends of the relationship may be collections.

For example:

let schema = {
  owner: {
    ref: 'tickets',
    collection: true,
  },
  tickets: {
    ref: 'owner',
  },
};

Use field value of {lid: ...} for related objects:

db.put({lid: 'ticket1', owner: 'user1'});
db.put({lid: 'user1', tickets: [{lid: 'ticket2'}]});

Collections

ref: 'reverse',
collection: true,
sort: function compare(x, y) {
  ...
}

An array field value adds entities in to a set. The sort comparator is optional. To remove entities, see remove.

For non-reference collections, simply put an array or other collection in a scalar field. Note that puts to scalar fields perform a complete value replacement, not set union.

Cascading Delete

destroy: true

Use on ref fields to recursively call destroy.