JavaScript In-memory Query Language.
npm install @jsiqle/core
You would typically use this library much like a traditional ORM, but without the data bindings to interface with a database. In such a scenario, data would be provided by loading data from a filesystem-based data storage such as JSON files.
// Suppose data for yur schema is appropriately stored in data.json
import data from './data.json';
import jsiqle from '@jsiqle/core';
const Ledger = jsiqle.create({
models: [
{
name: 'Person',
fields: {
username: 'string',
role: 'string',
firstName: 'string',
lastName: 'string'
},
properties: {
fullName: rec => `${rec.firstName} ${rec.lastName}`
}
},
{
name: 'Transaction',
fields: {
time: 'date',
amount: 'number'
}
}
],
relationships: [
{
from: { model: 'Transaction', name: 'payer' },
to: { model: 'Person', name: 'outgoingTransactions' },
type: 'manyToOne'
},
{
from: { model: 'Transaction', name: 'payee' },
to: { model: 'Person', name: 'incomingTransactions' },
type: 'manyToOne'
}
]
});
const Person = Ledger.getModel('Person');
const Transaction = Ledger.getModel('Transaction');
// Load some data from storage
data.people.forEach(
personData => Person.createRecord(personData)
);
data.transactions.forEach(
transactionData => Transaction.createRecord(transactionData)
);
// Get the name of the first person that has at least one outgoing transaction
Person.records.where(p => p.outgoingTransactions.length).first.fullName;
Note: Data bindings for a standardized JSON-based data storage may be coming soon, either as part of this package or a complementary one.
A schema instance can be created using jsiqle.create()
:
import jsiqle from '@jsiqle/core';
const MySchema = jsiqle.create({});
Schema definition requires an object argument with the following attributes:
models
: (Optional) An array of models that are part of the schema. More information about model definitions can be found in the next section.relationships
: (Optional) An array of relationships between models. More information about relationship definitions can be found in one of the following sections.serializers
: (Optional) An array of serializers for the schema. More information about serializer definitions can be found in one of the following sections.config
: (Optional) A configuration object that supports the following attributes:experimentalAPIMessages
: One of'warn'
,'error'
or'off'
. Depending on this flag, experimental API messages can either be logged as warnings, throw an error or be turned off entirely.
Models can be defined as part of the schema definition.
import jsiqle from '@jsiqle/core';
const MySchema = jsiqle.create({
models: [{ name: 'MyModel' }]
});
const MyModel = MySchema.getModel('MyModel');
Model definition options require an object argument with the following attributes:
name
: The name of the model. By convention, model names and variables should be title-cased (i.e.MyModel
instead ofmyModel
). Model names must be globally unique.fields
: (Optional) An object containing key-value pairs for fields that make up the model. More information about field definitions can be found in the next section.properties
: (Optional) An object containing key-value pairs for getter properties to be defined on the model. All properties expect a single argument representing a record of the given model. More information about property definitions can be found in one of the following sections.scopes
: (Optional) An object containing key-value pairs for getter properties to be defined on the record set of the model. All scopes expect a single argument representing the record set or a subset of records from the current model. Alternatively, an object with amatcher
andsorter
key can be supplied for ordered scopes. More information about scope definitions can be found in one of the following sections.
You can retrieve an already defined model by calling Schema.prototype.getModel()
with the model name:
const MyModel = MySchema.getModel('MyModel');
Fields can be defined as part of a model definition.
import jsiqle from '@jsiqle/core';
const MySchema = jsiqle.create({
models: [
{
name: 'MyModel',
fields: {
firstName: 'string',
age: 'number',
username: 'string',
role: 'string'
}
}
]
});
Field definition options require an object argument with the following attributes:
name
: The name of the field. By convention, field names should be camel-cased (i.e.myField
). Field names must be unique for each model.type
: The type of the field. Read below for more information on types and validation.
In the case of defining the field in the model definition, the field name
should be defined as the key that the type string corresponds to.
There are a few standard field types corresponding to primitives and certain common field type values:
boolean string number date object
booleanArray numberArray stringArray dateArray
Properties can be defined as part of a model definition.
import jsiqle from '@jsiqle/core';
const MySchema = jsiqle.create({
models: [
{
name: 'MyModel',
fields: {
firstName: 'string',
lastName: 'string'
},
properties: {
fullName: record => `${record.firstName} ${record.lastName}`,
formalName: {
body: record => `${record.lastName} ${record.firstName}`,
cache: true
}
}
}
]
});
Properties are specified as key-value pairs. The value can either be a function or an object containing a body
and optional cache
.
Property functions can expect up to two arguments. If they do not expect any arguments or only expect a single argument, they are called with the current record as their sole argument. If they expect two arguments, they are considered "lazy" properties and their second argument is automatically bound to the current schema object representation ({ models, serializers }
). These can only be defined as part of the model definition, are added to the model post initialization and are useful if you need access to other models or serializers.
Properties can receive an additional boolean key, cache
, indicating if the property should be cached. Property caches are persisted as long as there are no field changes for a given record and cannot be specified for relationships. This means that properties that depend on other properties, methods or external values are not good candidates for caching. If a cached property is stale, yo can force a recalculation is via updating any field on the record manually. Additionally, all cached properties across all models can be cleared via Schema.prototype.clearPropertyCache()
.
Boolean properties can receive an additional string key, inverse
, defining the name of the inverse boolean property. Inverse boolean properties are automatically updated when the property is updated and vice versa and can be called like regular properties on any records of the model. Caching is also applied to inverse boolean properties, if specified via the cache
boolean key.
import jsiqle from '@jsiqle/core';
const MySchema = jsiqle.create({
models: [
{
name: 'MyModel',
fields: {
firstName: 'string',
lastName: 'string'
},
properties: {
fullName: record => `${record.firstName} ${record.lastName}`,
formalName: {
body: record => `${record.lastName} ${record.firstName}`,
cache: true
},
hasLongLastName: {
body: record => record.lastName.length > 10,
inverse: 'hasShortLastName'
},
}
},
{
name: 'AnotherModel',
properties: {
myModelName: (record, { models: { myModel }}) => myModel.name
}
}
]
});
Methods can be defined as part of a model definition.
import jsiqle from '@jsiqle/core';
const MySchema = jsiqle.create({
models: [
{
name: 'MyModel',
fields: {
firstName: 'string',
lastName: 'string'
},
methods: {
prefixedName: (record, prefix) => `${prefix} ${record.lastName}`,
suffixedName: (record, suffix) => `${record.firstName} ${suffix}`,
isModelNameCorrect: (record, { models: { myModel }}, valey) => value === myModel.name
}
}
]
});
Methods definition are specified as key-value pairs.
Methods expect any number of arguments, the current record and any arguments passed to them when called, and may return any type of value. They are called with the current record as their first argument, any other arguments passed to them at call time and the current schema object representation ({ models, serializers }
) as their last argument. This means that methods can access other models and serializers.
Scopes can be defined as part of a model definition.
import jsiqle from '@jsiqle/core';
const MySchema = jsiqle.create({
models: [
{
name: 'MyModel',
fields: {
firstName: 'string',
lastName: 'string'
},
scopes: {
smiths: record => record.lastName === 'Smith'
does: record => record.lastName === 'Doe'
orderedSmiths: {
matcher: record => record.lastName === 'Smith',
sorter: (a, b) => a.firstName.localeCompare(b.firstName)
}
}
}
]
});
Scopes definition are specified as key-value pairs.
Scopes expect one argument, the current record, and must return a boolean indicating if the scope should include the record or not. Alternatively, scopes can be specified as objects when defined as part of the model definition with a matcher
function and a sorter
function. This will create an ordered scope that will always apply the sorter
to matched records before returning them.
Note: The relationships API is not currently stable and is considered experimental. While no major changes are expected in the future, it might not be fit for use in production just yet.
Relationships can be defined as part of the schema definition.
import jsiqle from '@jsiqle/core';
const MySchema = jsiqle.create({
models: [
{
name: 'Person',
fields: {
username: 'string',
}
},
{
name: 'Transaction',
fields: {
amount: 'number',
},
}
],
relationships: [
{
from: { model: 'Transaction', name: 'payer' },
to: { model: 'Person', name: 'outgoingTransactions' },
type: 'manyToOne'
},
{
from: { model: 'Transaction', name: 'payee' },
to: { model: 'Person', name: 'incomingTransactions' },
type: 'manyToOne'
}
]
});
A relationship definition is an object with the following keys:
from
: Either a string representing the name of a model or an object with amodel
key and aname
key. In the latter case, thename
key is the name that will be given to the field in the specifiedmodel
.to
: Either a string representing the name of a model or an object with amodel
key and aname
key. In the latter case, thename
key is the name that will be given to the field in the specifiedmodel
.type
: One ofoneToOne
,oneToMany
,manyToOne
,manyToMany
depending on the type of relationship.
When a relationship is defined between to models, the model specified as from
will receive a new field named accordingly. Similarly, the to
model will receive a new property instead that performs the reverse operation. Only the field on the from
model is writeable.
The names for the field and property are automatically generated if not specified, must be valid names and not already exist in the model. For singular relationships, the name of the other model is used, whereas for plural the name of the model followed by Set
. For example, a manyToOne
relationship between two models, Person
and Transaction
, would be named transactionSet
on Person
and person
on Transaction
.
Relationships between records of the same model are allowed. The only caveat is that symemtric (i.e. oneToOne
and manyToMany
) relationships in the same model need to be named on both sides.
Serializers can be defined as part of the schema definition.
import jsiqle from '@jsiqle/core';
const MySchema = jsiqle.create({
models: [
{
name: 'MyModel',
fields: {
firstName: 'string',
lastName: 'string'
}
}
],
serializers: [
{
name: 'MySerializer',
attributes: [
'firstName,
['lastName', 'surname'],
['fullName', 'name']
'truncatedName'
],
methods: {
fullName: (record) => `${record.firstName} ${record.lastName}`,
truncatedName: (record) => record.fullName.slice(0, 20)
}
}
]
});
Serializer definition options require an object argument with the following attributes:
name
: The name of the serializer. Must be unique.attributes
: An array of strings or arrays of strings. Each string represents the name of a model field or property or a serializer method. Each array represents the name of a model field or property or a serializer method and the name that should be used in the serializer. For example,['lastName', 'surname']
would create a field namedsurname
in the serializer that would be populated with the value of thelastName
field on the model.methods
: An object with key-value pairs. Each key represents the name of a serializer method and each value a function that will be called with the current record as its first argument and any arguments passed to the serializer at call time as an object as its second argument. The method can return any value.
At the heart of datasets are records, which represent individual data points. Each record belongs to a model and must comply with the model's definition.
Records can be created using the Model.prototype.createRecord()
method:
import jsiqle from '@jsiqle/core';
const MySchema = jsiqle.create({
models: [
{
name: 'MyModel',
fields: {
firstName: 'string',
lastName: 'string'
},
properties: {
fullName: record => `${record.firstName} ${record.lastName}`
}
}
]
});
const MyModel = MySchema.getModel('MyModel');
MyModel.createRecord({ id: 'jsmith', firstName: 'John', lastName: 'Smith' });
MyModel.createRecord({ id: 'jdoe', firstName: 'John', lastName: 'Doe' });
Each record definition consists of an object with the appropriate key-value pairs. Fields without a value will be automatically set to null
. All records must contain an id
key with a string value that is unique within the model. Key-value pairs that do not match a field definition will be stored in the record. This can be useful for fields that might be added in later operations (e.g. adding relationships to a populated model).
Records can be updated in place (i.e. retrieving them from the record set and updating their attributes). They can also be updated by calling Model.prototype.updateRecord()
:
MyModel.records.first.firstName = 'Jim';
MyModel.updateRecord('jdoe', { firstName: 'Josh' });
When calling Model.prototype.updateRecord()
, the given object will be merged into the existing record, allowing for partial record updating.
Records can be removed using Model.prototype.removeRecord()
. Note that record removal does not currently run any validations or perform any cascade operations to ensure relationship stability.
MyModel.removeRecord('jdoe');
Data from a model is stored in records that make up the model's record set. Records and record sets can be queried in various ways.
To query the record set of an individual model, use the Model.prototype.records
getter. This returns the record set along with all the tools neccessary to filter and transform results.
import jsiqle from '@jsiqle/core';
const MySchema = jsiqle.create({
models: [
{
name: 'MyModel',
fields: {
firstName: 'string',
lastName: 'string'
},
properties: {
fullName: record => `${record.firstName} ${record.lastName}`
}
}
]
});
const MyModel = MySchema.getModel('MyModel');
MyModel.createRecord({ id: 'jsmith', firstName: 'John', lastName: 'Smith' });
MyModel.createRecord({ id: 'jdoe', firstName: 'John', lastName: 'Doe' });
const records = MyModel.records;
Record sets can be filtered, mapped and sorted much like regular arrays. Here's a list of operations:
RecordSet.prototype.forEach()
: Executes a provided function once for every element in the calling record set. This method takes a callback function as an argument that expects three arguments (record
,id
,recordSet
), similar toArray.prototype.forEach()
. The method does not return a result.RecordSet.prototype.map()
: Creates an array or object populated with the results of calling a provided mapping function on every element in the calling record set. This method takes a mapping callback function as an argument that expects three arguments (record
,id
,recordSet
), similar toArray.prototype.map()
. The result is an array object with each id mapped to the result of the mapping function. Pass the{ flat: true }
option to return an array instead of an object.RecordSet.prototype.reduce()
: Executes a user-supplied reducer callback function on each element of the record set, passing in the return value from the calculation on the preceding element. This method takes a reducer callback function as an argument that expects four arguments (accumulator
,record
,id
,recordSet
) and an initial value, similar toArray.prototype.reduce()
. The final result of running the reducer across all elements of the record set is a single value.RecordSet.prototype.filter()
: Creates a new record set or array with all elements that pass the test implemented by the provided filtering function. This method takes a filtering callback function as an argument that expects three arguments (record
,id
,recordSet
), similar toArray.prototype.filter()
. The result is a record set or array containing only the records that pass the test. Pass the{ flat: true }
option to return an array instead of a record set.RecordSet.prototype.find()
: Retrieves the first record matching the condition implemented by the provided testing function. This method takes a testing callback function as an argument that expects three arguments (record
,id
,recordSet
), similar toArray.prototype.find()
. The result is a record orundefined
if none match the condition.RecordSet.prototype.findId()
: Same asRecordSet.prototype.find()
except that the resulting value is the record's id instead of the record itself.RecordSet.prototype.only()
: Returns a new record set containing only objects that match the id/ids provided. Records are returned in order of appearance in the provided ids. Expects any number of ids as arguments.RecordSet.prototype.except()
: Returns a new record set containing only objects that don't match the id/ids provided. Expects any number of ids as arguments.RecordSet.prototype.every()
: Returns a boolean indicating if all the records in the record set pass the test implemented by the provided testing function. This method takes a testing callback function as an argument that expects three arguments (record
,id
,recordSet
), similar toArray.prototype.every()
.RecordSet.prototype.some()
: Returns a boolean indicating if any of the records in the record set pass the test implemented by the provided testing function. This method takes a testing callback function as an argument that expects three arguments (record
,id
,recordSet
), similar toArray.prototype.some()
.RecordSet.prototype.where()
: Creates a new record set with all elements that pass the test implemented by the provided filtering function. This method takes a filtering callback function as an argument that expects three arguments (record
,id
,recordSet
), similar toArray.prototype.filter()
. The result is a record set containing only the records that pass the test.RecordSet.prototype.whereNot()
: Creates a new record set with all elements that fail the test implemented by the provided filtering function. This method takes a filtering callback function as an argument that expects three arguments (record
,id
,recordSet
), similar toArray.prototype.filter()
. The result is a record set containing only the records that fail the test.
Specific attributes can be selected from records via the following properties:
RecordSet.prototype.select()
: Expects any number of field names in a record. Returns an array of objects with only the selected fields.RecordSet.prototype.pluck()
: Expects any number of field names in a record. Returns an array of arrays with only the selected field values. If only one key is provided, an array of individual attributes will be returned instead.
Record sets can be grouped or sorted via the following properties:
RecordSet.prototype.groupBy()
: Expects a field name and groups the records based on its value. Returns an object with value-based keys containing record sets.RecordSet.prototype.sort()
: Sorts the elements of the record set and returns a new sorted record set. Expects a comparator callback function as an argument that takes three arguments (firstValue
,secondValue
,firstId
,secondId
) and returns an appropriate value for sorting similar toArray.prototype.sort()
.
Record sets are iterable, meaning you can use for
loops to iterate over them, similar to a regular ES6 Map
. Additionally, RecordSet.prototype.batchIterator()
is available expecting a batchSize
numeric argument and allowing for the records in a record set to be iterated in batches. An additional { flat: true }
argument can be passed to return an array of records instead of a record set for each batch.
You can acces the first record of a record set using RecordSet.prototype.first
. Similarly, you can acces the last record of a record set using RecordSet.prototype.last
.
Additionally, you can get the first n
elements of a record set using RecordSet.prototype.limit()
with an appropriate numeric argument or skip over them and get all other records using RecordSet.prototype.offset()
with an appropriate numeric argument. These properties can be combined to get specific records in a record set based on the order of insertion.
Scopes defined on a model are defined as getters on its record set. Thus, they can be used by calling them on the record set:
import jsiqle from '@jsiqle/core';
const MySchema = jsiqle.create({
models: [
{
name: 'MyModel',
fields: {
firstName: 'string',
lastName: 'string'
},
scopes: {
smiths: record => record.lastName === 'Smith'
}
}
]
});
const MyModel = MySchema.getModel('MyModel');
MyModel.createRecord({ id: 'jsmith', firstName: 'John', lastName: 'Smith' });
MyModel.createRecord({ id: 'jdoe', firstName: 'John', lastName: 'Doe' });
const smithsFamily = MyModel.records.smiths;
Relationships can be queried from either side of the relationship using the field/property name added to the model. For more information refer to the section about relationship definitions and how they are represented in models.
The schema provides a quick way to access a model, record or even an attribute using the Schema.prototype.get()
method. This method takes a .
-separated path and returns the value at that particular location:
import jsiqle from '@jsiqle/core';
const MySchema = jsiqle.create({
models: [
{
name: 'MyModel',
fields: {
firstName: 'string',
lastName: 'string'
},
}
]
});
const MyModel = MySchema.getModel('MyModel');
MyModel.createRecord({ id: 'jsmith', firstName: 'John', lastName: 'Smith' });
MyModel.createRecord({ id: 'jdoe', firstName: 'John', lastName: 'Doe' });
const myModel = MySchema.get('MyModel');
const johnSmith = MySchema.get('MyModel.jsmith');
const johnSmithsName = MySchema.get('MyModel.jsmith.firstName');
Records and record sets can be serialized to regular objects, arrays or JSON. Calling JSON.stringify()
will suffice in most cases, as all records and record sets have appropriate properties to handle serialization. Apart from that, Record.prototype.toObject()
can be called for individual records to convert them into regular objects.
Additionally, record sets implement the following serialization properties:
RecordSet.prototype.toArray()
: Returns an array of records contained in the record set.RecordSet.prototype.toObject()
: Returns an object of records representing the key-value pairs of the records in the record set.
Both of these methods can be called with an optional { flat: true }
options argument to convert records into objects.
Serializers can be used to serialize records and record sets into custom formats. They are defined on the schema level and can be used by calling one of the methods available as part of the individual serializer:
Serializer.prototype.serialize()
: Serializes a record in the format defined by the serializer. Expects a record as the first argument and an optional options object as the second argument.Serializer.prototype.serializeArray()
: Serializes an array of records in the format defined by the serializer. Expects an array of records as the first argument and an optional options object as the second argument.Serializer.prototype.serializeRecordSet()
: Serializes a record set in the format defined by the serializer. Expects a record set as the first argument and an optional options object as the second argument. A third argument can be passed to specify a function mapping each record to a key in the serialized object.
Certain naming conventions and rules are in place.
Generally, names can only include alphanumeric characters and underscores and are expected to be unique in their scope. There are a few exceptions to this rule that are not enforced (e.g. scope names), but it should be followed across as part of the convention.
Additionally, conventions dictate that model and schema names are title-cased, and that all other names are camel-cased. Variables should also match this convention.
To clear up any confusion, here are the names of the definition types and the names of the corresponding data objects they create:
- A schema is a set of definitions that contain models, fields, relationships etc. The data contained within a schema is called a dataset.
- A model is a set of field, property and scope definitions. The data contained within a model is called a record set and each individual item within it is called a record.
- A record is a set of values corresponding to different keys. Each of these values is called an attribute.
This project is licensed under the MIT license.