English | Русский
Repository pattern implementation for Node.js
- Installation
- Import
- Description
- Example
- Schema
- Data Source
- Model
- Properties
- Validators
- Transformers
- Empty Values
- Repository
- Filtering
- Relations
- Extension
- TypeScript
- Tests
- License
npm install @e22m4u/js-repository
Optionally install an adapter.
description | |
---|---|
memory |
in-memory virtual database (no installation required) |
mongodb |
MongoDB - NoSQL database management system (install) |
The module supports both ESM and CommonJS standards.
ESM
import {Schema} from '@e22m4u/js-repository';
CommonJS
const {Schema} = require('@e22m4u/js-repository');
The module provides an abstraction layer over different database interfaces by representing them as named data sources connected to models. A model describes a database table where columns are represented as model properties. Each model property can have a specific type of allowed value, along with validators and transformers that process data before it is written to the database. Additionally, a model can define classic relationship types like "one-to-one", "one-to-many" and others between models.
Data operations are performed using a repository, which is available for each model with a declared data source. The repository can filter requested documents, validate properties according to the model definition, and include related data in query results.
- Data Source - defines database connection settings
- Model - describes document structure and relationships with other models
- Repository - handles read and write operations for model documents
flowchart TD
A[Schema]
subgraph Databases
B[Data Source 1]
C[Data Source 2]
end
A-->B
A-->C
subgraph Collections
D[Model A]
E[Model B]
F[Model C]
G[Model D]
end
B-->D
B-->E
C-->F
C-->G
H[Repository A]
I[Repository B]
J[Repository C]
K[Repository D]
D-->H
E-->I
F-->J
G-->K
Here's how to define a data source, create a model, and add a new document to the collection.
import {Schema} from '@e22m4u/js-repository';
import {DataType} from '@e22m4u/js-repository';
// create Schema instance
const schema = new Schema();
// declare "myMemory" data source
schema.defineDatasource({
name: 'myMemory', // name of new source
adapter: 'memory', // selected adapter
});
// declare "country" model
schema.defineModel({
name: 'country', // name of new model
datasource: 'myMemory', // selected data source
properties: { // model properties
name: DataType.STRING, // "string" type
population: DataType.NUMBER, // "number" type
},
})
// get repository for "country" model
const countryRep = schema.getRepository('country');
// add new document to "country" collection
const country = await countryRep.create({
name: 'Russia',
population: 143400000,
});
// output new document
console.log(country);
// {
// "id": 1,
// "name": "Russia",
// "population": 143400000,
// }
A Schema
class instance stores data source and model definitions.
Methods
defineDatasource(datasourceDef: object): this
- add a data sourcedefineModel(modelDef: object): this
- add a modelgetRepository(modelName: string): Repository
- get a repository
Examples
Import the class and create a schema instance.
import {Schema} from '@e22m4u/js-repository';
const schema = new Schema();
Define a new data source.
schema.defineDatasource({
name: 'myMemory', // name of new source
adapter: 'memory', // selected adapter
});
Define a new model.
schema.defineModel({
name: 'product', // name of new model
datasource: 'myMemory', // selected source
properties: { // model properties
name: DataType.STRING,
weight: DataType.NUMBER,
},
});
Get a repository by model name.
const productRep = schema.getRepository('product');
A data source defines an adapter selection and its configuration
settings. New data sources are added using the defineDatasource
method of a schema instance.
Parameters
name: string
unique nameadapter: string
selected adapter- additional adapter-specific parameters (if any)
Examples
Define a new data source.
schema.defineDatasource({
name: 'myMemory', // name of new source
adapter: 'memory', // selected adapter
});
Pass additional adapter parameters.
schema.defineDatasource({
name: 'myMongodb',
adapter: 'mongodb',
// mongodb adapter parameters
host: '127.0.0.1',
port: 27017,
database: 'myDatabase',
});
A model describes the structure of a collection document and
its relationships with other models. New models are added using
the defineModel
method of a schema instance.
Parameters
name: string
model name (required)base: string
name of parent model to inherit fromtableName: string
collection name in databasedatasource: string
selected data sourceproperties: object
property definitions (see Properties)relations: object
relationship definitions (see Relations)
Examples
Define a model with typed properties.
schema.defineModel({
name: 'user', // name of new model
properties: { // model properties
name: DataType.STRING,
age: DataType.NUMBER,
},
});
The properties
parameter in a model definition accepts an object
where keys are model properties and values are either a property
type or an object with additional parameters.
Data Type
DataType.ANY
any value allowedDataType.STRING
onlystring
type valueDataType.NUMBER
onlynumber
type valueDataType.BOOLEAN
onlyboolean
type valueDataType.ARRAY
onlyarray
type valueDataType.OBJECT
onlyobject
type value
Parameters
type: string
type of allowed value (required)itemType: string
array item type (fortype: 'array'
)model: string
object model (fortype: 'object'
)primaryKey: boolean
mark property as primary keycolumnName: string
override column namecolumnType: string
column type (defined by adapter)required: boolean
mark property as requireddefault: any
default valuevalidate: string | array | object
see Validatorsunique: boolean | string
check value uniqueness
unique
When unique
is set to true
or 'strict'
, strict uniqueness
checking is performed. In this mode, empty values
are also validated, where null
and undefined
cannot appear more
than once.
The 'sparse'
mode only checks non-empty values, excluding
empty values which vary by property type.
For example, for string
type, empty values include undefined
,
null
and ''
(empty string).
unique: true | 'strict'
strict uniqueness checkunique: 'sparse'
exclude empty values from checkunique: false | 'nonUnique'
no uniqueness check (default)
Predefined constants can be used as unique
parameter values,
equivalent to the string values strict
, sparse
and nonUnique
:
PropertyUniqueness.STRICT
PropertyUniqueness.SPARSE
PropertyUniqueness.NON_UNIQUE
Examples
Short model property definition.
schema.defineModel({
name: 'city',
properties: { // model properties
name: DataType.STRING, // "string" property type
population: DataType.NUMBER, // "number" property type
},
});
Full model property definition.
schema.defineModel({
name: 'city',
properties: { // model properties
name: {
type: DataType.STRING, // "string" property type (required)
required: true, // exclude undefined and null values
},
population: {
type: DataType.NUMBER, // "number" property type (required)
default: 0, // default value
},
code: {
type: DataType.NUMBER, // "number" property type (required)
unique: PropertyUniqueness.UNIQUE, // check uniqueness
},
},
});
Factory default values. The function's return value is determined when writing the document.
schema.defineModel({
name: 'article',
properties: { // model properties
tags: {
type: DataType.ARRAY, // "array" property type (required)
itemType: DataType.STRING, // "string" item type
default: () => [], // factory value
},
createdAt: {
type: DataType.STRING, // "string" property type (required)
default: () => new Date().toISOString(), // factory value
},
},
});
In addition to type checking, properties can have validators that process values before they are written to the database. Empty values are exempt from validation.
minLength: number
minimum length for strings or arraysmaxLength: number
maximum length for strings or arraysregexp: string | RegExp
regular expression pattern check
Example
Validators are specified in the model property definition using
the validate
parameter, which accepts an object with validator
names and settings.
schema.defineModel({
name: 'user',
properties: {
name: {
type: DataType.STRING,
validate: { // validators for "name" property
minLength: 2, // minimum string length
maxLength: 24, // maximum string length
},
},
},
});
A validator is a function that receives a property value before
it is written to the database. If validation returns false
,
a standard error is thrown. Custom errors can be thrown directly
within the validator function.
Custom validators are registered using the addValidator
method
of the PropertyValidatorRegistry
service, which accepts a new
validator name and validation function.
Example
// create validator to allow
// only numeric characters
const numericValidator = (input) => {
return /^[0-9]+$/.test(String(input));
}
// register "numeric" validator
schema
.get(PropertyValidatorRegistry)
.addValidator('numeric', numericValidator);
// use validator in "code" property
// definition for new model
schema.defineModel({
name: 'document',
properties: {
code: {
type: DataType.STRING,
validate: 'numeric',
},
},
});
Transformers modify property values before they are written to the database. They define how incoming data should be processed. Empty values are exempt from transformation.
trim
removes whitespace from both ends of stringtoUpperCase
converts string to uppercasetoLowerCase
converts string to lowercasetoTitleCase
converts string to title case
Example
Transformers are specified in the model property definition using
the transform
parameter. It accepts a transformer name as a string.
For multiple transformers, use an array. If a transformer has settings,
use an object where the key is the transformer name and the value contains
its parameters.
schema.defineModel({
name: 'user',
properties: {
name: {
type: DataType.STRING,
transform: [ // transformers for "name" property
'trim', // remove spaces from both ends of string
'toTitleCase', // convert string to title case
],
},
},
});
Different property types have their own sets of empty values. These sets
determine whether a property value has meaningful content. For example,
the default
parameter in a property definition only sets the default
value if the incoming value is empty. The required
parameter excludes
empty values by throwing an error. The unique
parameter in sparse
mode allows duplicate empty values for a unique property.
type | empty values |
---|---|
'any' |
undefined , null |
'string' |
undefined , null , '' |
'number' |
undefined , null , 0 |
'boolean' |
undefined , null |
'array' |
undefined , null , [] |
'object' |
undefined , null , {} |
A repository performs read and write operations on documents
of a specific model. You can get a repository using the schema
instance's getRepository
method.
Methods
create(data, filter = undefined)
add new documentreplaceById(id, data, filter = undefined)
replace entire documentreplaceOrCreate(data, filter = undefined)
replace or create newpatchById(id, data, filter = undefined)
partially update documentpatch(data, where = undefined)
update all documents or by conditionfind(filter = undefined)
find all documents or by conditionfindOne(filter = undefined)
find first document or by conditionfindById(id, filter = undefined)
find document by identifierdelete(where = undefined)
delete all documents or by conditiondeleteById(id)
delete document by identifierexists(id)
check existence by identifiercount(where = undefined)
count all documents or by condition
Arguments
id: number|string
identifier (primary key)data: object
object representing document structurewhere: object
query parameters (see Filtering)filter: object
result parameters (see Filtering)
Examples
Get repository by model name.
const countryRep = schema.getRepository('country');
Add new document to collection.
const res = await countryRep.create({
name: 'Russia',
population: 143400000,
});
console.log(res);
// {
// "id": 1,
// "name": "Russia",
// "population": 143400000,
// }
Find document by identifier.
const res = await countryRep.findById(1);
console.log(res);
// {
// "id": 1,
// "name": "Russia",
// "population": 143400000,
// }
Delete document by identifier.
const res = await countryRep.deleteById(1);
console.log(res); // true
Some repository methods accept a settings object that affects
the returned result. The find
method's first parameter accepts
the widest range of options, which are listed below.
where: object
selection objectorder: string[]
order specificationlimit: number
limit number of documentsskip: number
skip documentsfields: string[]
select required model propertiesinclude: object
include related data in result
The parameter accepts an object with selection conditions and supports a wide range of comparison operators.
{foo: 'bar'}
search by property foo
value
{foo: {eq: 'bar'}}
equality operator eq
{foo: {neq: 'bar'}}
inequality operator neq
{foo: {gt: 5}}
"greater than" operator gt
{foo: {lt: 10}}
"less than" operator lt
{foo: {gte: 5}}
"greater than or equal" operator gte
{foo: {lte: 10}}
"less than or equal" operator lte
{foo: {inq: ['bar', 'baz']}}
equality to one of values inq
{foo: {nin: ['bar', 'baz']}}
exclude array values nin
{foo: {between: [5, 10]}}
range operator between
{foo: {exists: true}}
value existence operator exists
{foo: {like: 'bar'}}
substring search operator like
{foo: {ilike: 'BaR'}}
case-insensitive version ilike
{foo: {nlike: 'bar'}}
substring exclusion operator nlike
{foo: {nilike: 'BaR'}}
case-insensitive version nilike
{foo: {regexp: 'ba.+'}}
regular expression operator regexp
{foo: {regexp: 'ba.+', flags: 'i'}}
regular expression flags
Note: Conditions can be combined with and
, or
and nor
operators.
Examples
Apply selection conditions when counting documents.
const res = await rep.count({
authorId: 251,
publishedAt: {
lte: '2023-12-02T14:00:00.000Z',
},
});
Apply or
operator when deleting documents.
const res = await rep.delete({
or: [
{draft: true},
{title: {like: 'draft'}},
],
});
The parameter orders the selection by specified model properties.
Reverse order can be specified with the DESC
suffix in
the property name.
Examples
Order by createdAt
field.
const res = await rep.find({
order: 'createdAt',
});
Order by createdAt
field in reverse order.
const res = await rep.find({
order: 'createdAt DESC',
});
Order by multiple properties in different directions.
const res = await rep.find({
order: [
'title',
'price ASC',
'featured DESC',
],
});
Note: The ASC
order direction is optional.
The parameter includes related documents in the method result. The included relation names must be defined in the current model (see Relations).
Examples
Include relation by name.
const res = await rep.find({
include: 'city',
});
Include nested relations.
const res = await rep.find({
include: {
city: 'country',
},
});
Include multiple relations using array.
const res = await rep.find({
include: [
'city',
'address',
'employees'
],
});
Use filtering of included documents.
const res = await rep.find({
include: {
relation: 'employees', // relation name
scope: { // filter "employees" documents
where: {hidden: false}, // query conditions
order: 'id', // document order
limit: 10, // limit number
skip: 5, // skip documents
fields: ['name', 'surname'], // only specified fields
include: 'city', // include relations for "employees"
},
},
});
The relations
parameter in a model definition accepts an object where
the key is the relation name and the value is an object with parameters.
Parameters
type: string
relation typemodel: string
target model nameforeignKey: string
current model property for target identifierpolymorphic: boolean|string
declare relation as polymorphic*discriminator: string
current model property for target name*
Note: Polymorphic mode allows dynamically determining the target model by its name, which is stored in the discriminator property.
Relation Type
belongsTo
- current model contains property for target identifierhasOne
- reverse side ofbelongsTo
using "one-to-one" principlehasMany
- reverse side ofbelongsTo
using "one-to-many" principlereferencesMany
- document contains array with target model identifiers
Examples
Declare belongsTo
relation.
schema.defineModel({
name: 'user',
relations: {
role: { // relation name
type: RelationType.BELONGS_TO, // current model references target
model: 'role', // target model name
foreignKey: 'roleId', // foreign key (optional)
// if "foreignKey" is not specified, then foreign key
// property is formed from relation name with "Id" suffix
},
},
});
Declare hasMany
relation.
schema.defineModel({
name: 'role',
relations: {
users: { // relation name
type: RelationType.HAS_MANY, // target model references current
model: 'user', // target model name
foreignKey: 'roleId', // foreign key from target model to current
},
},
});
Declare referencesMany
relation.
schema.defineModel({
name: 'article',
relations: {
categories: { // relation name
type: RelationType.REFERENCES_MANY, // relation through array of identifiers
model: 'category', // target model name
foreignKey: 'categoryIds', // foreign key (optional)
// if "foreignKey" is not specified, then foreign key
// property is formed from relation name with "Ids" suffix
},
},
});
Polymorphic version of belongsTo
schema.defineModel({
name: 'file',
relations: {
reference: { // relation name
type: RelationType.BELONGS_TO, // current model references target
// polymorphic mode allows storing target model name
// in discriminator property, formed from relation name
// with "Type" suffix, so in this case target model name
// is stored in "referenceType" and document identifier
// in "referenceId"
polymorphic: true,
},
},
});
Polymorphic version of belongsTo
with properties specification.
schema.defineModel({
name: 'file',
relations: {
reference: { // relation name
type: RelationType.BELONGS_TO, // current model references target
polymorphic: true, // target model name stored in discriminator
foreignKey: 'referenceId', // property for target identifier
discriminator: 'referenceType', // property for target model name
},
},
});
Polymorphic version of hasMany
with target model relation name.
schema.defineModel({
name: 'letter',
relations: {
attachments: { // relation name
type: RelationType.HAS_MANY, // target model references current
model: 'file', // target model name
polymorphic: 'reference', // target model polymorphic relation name
},
},
});
Polymorphic version of hasMany
with target model property.
schema.defineModel({
name: 'letter',
relations: {
attachments: { // relation name
type: RelationType.HAS_MANY, // target model references current
model: 'file', // target model name
polymorphic: true, // current model name is in discriminator
foreignKey: 'referenceId', // target model property for identifier
discriminator: 'referenceType', // target model property for current name
},
},
});
The getRepository
method of a schema instance checks for
an existing repository for the specified model and returns it.
Otherwise, a new instance is created and cached for subsequent
calls to the method.
import {Schema} from '@e22m4u/js-repository';
import {Repository} from '@e22m4u/js-repository';
// const schema = new Schema();
// schema.defineDatasource ...
// schema.defineModel ...
const rep1 = schema.getRepository('model');
const rep2 = schema.getRepository('model');
console.log(rep1 === rep2); // true
To replace the default repository constructor, use the setRepositoryCtor
method of the RepositoryRegistry
service, which is available in
the schema instance container. After this, all new repositories will
be created using the specified constructor instead of the default one.
import {Schema} from '@e22m4u/js-repository';
import {Repository} from '@e22m4u/js-repository';
import {RepositoryRegistry} from '@e22m4u/js-repository';
class MyRepository extends Repository {
/*...*/
}
// const schema = new Schema();
// schema.defineDatasource ...
// schema.defineModel ...
schema.get(RepositoryRegistry).setRepositoryCtor(MyRepository);
const rep = schema.getRepository('model');
console.log(rep instanceof MyRepository); // true
Note: Since repository instances are cached, constructor replacement
should be done before calling the getRepository
method.
Get a typed repository with model interface specification.
import {Schema} from '@e22m4u/js-repository';
import {DataType} from '@e22m4u/js-repository';
import {RelationType} from '@e22m4u/js-repository';
// const schema = new Schema();
// schema.defineDatasource ...
// schema.defineModel ...
// define "city" model
schema.defineModel({
name: 'city',
datasource: 'myDatasource',
properties: {
title: DataType.STRING,
timeZone: DataType.STRING,
},
relations: {
country: {
type: RelationType.BELONGS_TO,
model: 'country',
},
},
});
// define "city" interface
interface City {
id: number;
title?: string;
timeZone?: string;
countryId?: number;
country?: Country;
}
// get repository by model name
// specifying its type and identifier type
const cityRep = schema.getRepository<City, number>('city');
npm run test
MIT