Modern JavaScript ODM for Couchbase ποΈ
Settee is a modern ODM for Couchbase (and Node.js), featuring a custom query builder, and fast, easy-to-use code.
- Robust, well-documented source code written in TypeScript (compiled to ES6)
- 100% test coverage with over 100 tests
- Includes type definitions (IDE typehint support!)
- Custom query builder with chainable statements and paginator
- Custom type definitions for schemas with default values
- Unlimited nesting within schemas
- Referenced Models
- Powerful schema layouts with awesome Joi powered validation
- Indexer
- Fully Promise based code with async-await support (no more callbacks!)
To install Settee, just run the following:
$ yarn add settee
#... or npm
$ npm install settee --save
Note: This package was only tested on Node.js versions 6 and 7 or higher.
First, you need to setup your connection to Couchbase. We have included that functionality with Settee.
import { settee, Schema, Type } from 'settee'
// Connect to the cluster.
await settee.connect('127.0.0.1', 'default')
// Create a new schema for a Car.
const CarSchema = new Schema('Car', {
brand: Type.string(),
color: Type.string()
})
// Build a Model
const Car = settee.buildModel(CarSchema)
// Create a new Car entry in the database.
let audi = await Car.create({
brand: 'Audi',
color: 'red'
})
This section will cover most of Settee's use cases. For more detailed API reference, click here. In our usage examples, we use car, engine and user examples to demonstrate the usage in real-world applications.
Note: Most of these examples require a connection to the Bucket. Check out
settee.connect()
in the API reference.
List of examples:
- Defining a Schema
- Creating a new entry in the database
- Modifying an entry
- Creating indexes
- Queries
- Nested data
- Referenced Models
- Accessing registered models
Schema is the core entity of Settee. It describes the structure of mapped instances and provides useful checks to ensure that only the correct entries are persisted in the cluster. Individual items in the Schema are referred to as Types. Types are mostly basic data types, such as number, object, string, etc. Check out the Types API reference for detailed overview.
Example: We want to create a schema for a car. Our car will have 3 properties: brand
, topSpeed
and taxPaid
.
const CarSchema = new Schema('Car', {
brand: Type.string(),
topSpeed: Type.number(),
taxPaid: Type.boolean(false)
})
Car's brand is a string, like Audi
. Top speed is defined as number, like 192
(kph) and tax Paid is a boolean, with a default value of false
. You can pass default values to any of the Types. Let's say we want all cars to be created with a top speed of a 100 kph, and we will modify them later.
const CarSchema = new Schema('Car', {
brand: Type.string(),
topSpeed: Type.number(100),
taxPaid: Type.boolean(false)
})
After you set your schema, build a model from it.
const Car = settee.buildModel(CarSchema)
Car
is now a Model.
To create a new entry in the database, first you have to define a Schema. We have already done that in the previous example. We now have a Car
constant which is a Model. Let's create a new car:
let audi = await Car.create({
brand: 'Audi',
topSpeed: 220,
taxPaid: true
})
This will store a new entry in our database. Behind the scenes, we add a docType
and a docId
to the entry, so it's possible to find it, edit and more. You can now access the individual properties of the entry like this:
audi.brand // "Audi"
audi.topSpeed // 220
audi.taxPaid // true
Note: Create runs Couchbase Insert behind the scenes.
We have already created a car entry in the database. Let's say we didn't really pay tax and the top speed is 250. Let's edit the entry!
// audi is an Instance
audi.topSpeed = 250
audi.taxPaid = false
await audi.save()
Model.save()
updates an existing entry. Please note that Settee will validate the data you input, and will be rejected with an error if the data types are incorrect:
audi.topSpeed = false
audi.taxPaid = false
await audi.save() // gets rejected
Indexes are very useful for faster lookups. Let's say we have a lot of cars in our database, and we usually look them up by brand
. Let's add an index for it, so the search is faster!
// define our schema
const CarSchema = new Schema('Car', {
brand: Type.string(),
topSpeed: Type.number(),
taxPaid: Type.boolean()
})
CarSchema.addIndexes({
findByBrand: { by: 'brand' }
})
After that, we need to make sure that we build a model from our Schema, and the indexes are ready to use. We can achieve that like this:
// Build a Model from our Schema
const Car = settee.buildModel(CarSchema)
// ensure that the indexes are ready
await settee.buildIndexes()
Settee also supports multi-key index. Let's say we'd like to have faster lookups for both brand
and taxPaid
:
const CarSchema = new Schema('Car', {
brand: Type.string(),
topSpeed: Type.number(),
taxPaid: Type.boolean()
})
CarSchema.addIndexes({
findByBrand: { by: 'brand' },
findByBrandAndTax: { by: ['brand', 'taxPaid'] }
})
const Car = settee.buildModel(CarSchema)
await settee.buildIndexes()
Note: Settee creates N1QL indexes behind the scenes.
Settee features a powerful QueryBuilder. You can do a lot with it, and to check out the full functionality, check out the QueryBuilder API reference.
Here are a few use cases of the QB. Note that calling settee.buildModel()
is required for these operations.
// our Model is defined as 'Car'
// This query will get you all Audi cars,
// where top speed is greater or equal to 190 kph,
// and tax is paid.
let results = Car.query()
.where('brand', 'Audi')
.where('topSpeed', '>=', 190)
.whereNot('taxPaid', false)
.get()
// but if you only want one car entry retrieved
let results = Car.query()
.where('brand', 'Audi')
.where('topSpeed', '>=', 190)
.whereNot('taxPaid', false)
.first()
// if you want to know the count of the cars matching the criteria
let results = Car.query()
.where('brand', 'Audi')
.where('topSpeed', '>=', 190)
.whereNot('taxPaid', false)
.count()
// if you want to retrieve ALL cars, regardless of the criteria
let results = Car.all()
Nested data is useful when you want a little bit more structure with your data. In our example, we have a User schema, that contains Profile information for the user. With Settee, this can be achieved like this:
const UserSchema = {
email: Type.string(),
password: Type.string(),
profile: {
age: Type.integer(),
names: {
first: Type.string(),
last: Type.string()
}
}
}
const User = settee.buildModel(UserSchema)
let joe = await User.create({
email: 'joe@limedeck.io',
password: 'hashedString',
profile: {
age: 26,
names: {
first: 'Joe',
last: 'Persson'
}
}
})
// and you can access all the properties like usual
joe.email // 'joe@limedeck.io'
joe.profile.age // 26
joe.profile.name.first // 'Joe'
joe.profile.name.last // 'Persson'
Referenced models are useful when nesting data is not an option. Let's say you have multiple cars in your database, and their engines are relatively similar. The point of a reference is that you can reuse it in multiple different entries.
Let's build our example:
// Create our Engine Schema first.
const EngineSchema = new Schema('Engine', {
make: Type.string(),
power: Type.number()
})
const Engine = settee.buildModel(EngineSchema)
// we define the Engine as Type.reference(Engine)
const CarSchema = new Schema('Car', {
brand: Type.string(),
topSpeed: Type.number(),
taxPaid: Type.boolean(),
engine: Type.reference(Engine)
})
const Car = settee.buildModel(CarSchema)
let bmwEngine = await Engine.create({
power: 150,
make: 'Bayerische Motoren Werke AG'
})
let bmw = await Car.create({
brand: 'BMW',
engine: bmwEngine
})
// you can now access the engine directly
bmw.brand // 'BMW'
bmw.engine.power // 150
bmw.engine.make // 'Bayerische Motoren Werke AG'
// and you can also update the engine individually
bmw.engine.power = 200
await bmw.engine.save()
Settee gives you access to your models from any file in your source code. There are two approaches of accessing them.
Let's say we have an index.js file:
// Create our Engine Schema first.
const EngineSchema = new Schema('Engine', {
make: Type.string(),
power: Type.number()
})
const Engine = settee.buildModel(EngineSchema)
// we define the Engine as Type.reference(Engine)
const CarSchema = new Schema('Car', {
brand: Type.string(),
topSpeed: Type.number(),
taxPaid: Type.boolean(),
engine: Type.reference(Engine)
})
const Car = settee.buildModel(CarSchema)
export { Car, Engine }
In another file, you can access the models either by importing them, or through settee.getModel(modelName)
:
// using direct import
import { Car } from './index'
// using settee.getModel()
let Car = settee.getModel('Car')
// you now have access to all Car model methods. Both approaches are equivalent.
This is our public API reference. Arguments wrapped in square brackets like [argument]
are required arguments. Methods that return Promises are noted with async before the method name. The return type is noted after the method's name.
Contents:
Main settee file.
Imported via:
import { settee } from 'settee'
Available Methods:
Establishes the connection to the bucket. Sets an active bucket.
Arguments:
clusterUrl
(string) - URL that points to the cluster.bucketName
(string) - Name of the bucket you are connecting to.
Terminates the connection to the bucket.
Sets the active bucket.
Arguments:
bucket
(Bucket) - Resolved and connected bucket. See typings.d.ts for reference.
Provides the active bucket instance.
Provides the active storage instance.
Provides a model based on the schema.
Arguments:
schema
(Schema) - instance of Schema.
Registers a set of provided models. Useful when you have to reference the models after the settee.connect()
method, e.g. when you have a dedicated DB class which uses models specified in separate files/modules.
Arguments:
Models
(Model) - Array of Models
Example:
const EngineSchema = new Schema('Engine', {
make: Type.string(),
power: Type.number()
})
const Engine = settee.buildModel(EngineSchema)
settee.registerModels([Engine])
Builds deferred indexes.
Available properties:
Helps you retrieve the Couchbase consistency without having to remember all the different values. Useful for N1QL operations.
Usage:
settee.consistency.NOT_BOUND
settee.consistency.REQUEST_PLUS
settee.consistency.STATEMENT_PLUS
Retrieves all registered schemas.
Provides a registered model. Accessible from anywhere where settee
instance is available.
Arguments:
name
(string) - Model name
Schema class.
Imported via:
import { Schema } from 'settee'
Available Methods:
Schema constructor.
Arguments:
name
(string) - Name of the schema.layout
(object) - Object containing Types.
Sets the active storage for the schema.
Arguments:
storage
(Storage) - Storage instance
Provides the active storage.
Provides the validator instance.
Adds indexes to the list.
Arguments:
indexes
(object) - Object containing indexes. Needs to contain theby
property, and it can be either a string, or an array of strings (if you want an index containing multiple keys)
let indexes = {
findByColor: {
by: 'color'
},
findByColorAndBrand: {
by: ['color', 'brand']
}
}
Verifies if the index is present in the database.
Arguments:
name
(string) - Name of the index you are looking up.type
(string) - Index type. By default, 'GSI' is used.
Drops index by name.
Arguments:
name
(string) - Name of the index.options
(object) - Additional options.
Imported via:
import { Type } from 'settee'
Types are data types, used by Schema for ensuring data and types in the database entries are valid. The validation behind the scenes ensures you can't override a Number
with a boolean
by mistake, for example.
Types have an optional default value. This value will be used when creating a new entry without having to supply the value again when creating entries.
The methods for Type
are static, so you can call them without having to instantiate Type
, i.e.:
Type.boolean(false)
Type.string('foo')
Type.integer(42)
Type.number(3.14)
Type.array(Type.string(), ['foo', 'bar'])
These are the basic Types.
Defaults to null
when no defaultValue is supplied.
Defaults to null
when no defaultValue is supplied.
Defaults to null
when no defaultValue is supplied.
Defaults to null
when no defaultValue is supplied.
Arguments:
defaultValue
(any|Function) - Value for the corresponding data type. If the argument passed is a function, the function will be evaluated as well.
These are the special Types:
Defaults to []
when no defaultValue is supplied.
Arguments:
type
(Type) - Any of the available Types.defaultValue
(any|Function) - Value for the corresponding data type. If the argument passed is a function, the function will be evaluated as well.
Defaults to null
when no defaultValue is supplied.
If a defaultValue is supplied, it returns a moment instance, so you can process it however you'd like (parse, manipulate, add, localize, etc).
Arguments:
defaultValue
(any|Function) - Can be a ms timestamp, ISO date, or a moment instance, although using a moment instance is advised for consistency reasons.
Note: We return a moment instance in the UTC timezone.
Reference entry type. Provides an object when saving referenced models.
Arguments:
model
(Model) - A referred Model.
Not exported directly. You have to register a Schema first, in order to retrieve the Model, where the Query Builder (QB) is exposed:
import { settee, Schema } from 'settee'
const CarSchema = new Schema('Car', {
brand: Type.string(),
topSpeed: Type.number(),
taxPaid: Type.boolean()
})
const Car = settee.buildModel(CarSchema)
Now you can instantiate the QB directly like Schema.query()
or Schema.q()
as shorthand.
Car.query() // or Car.q()
Note: You have to call either
get()
orfirst()
after your query chain. Otherwise, you will not be able to retrieve any results.
Afterwards, you will have access to these methods:
Adds a WHERE
clause. You can chain however many wheres you want.
Arguments:
field
(string) - The field you are looking to match the value.operator
(any) - Can either be a logical operator, or a value, if you are trying to matchfield === value
value
(any) - Value you're looking up to match the logical operator and the field
Examples:
Car.q().where('brand', '=', 'Audi')
// is the same as
Car.q().where('brand', 'Audi')
Car.q().where('topSpeeed', '<=', 190)
// And you can chain multiple wheres as well
Car.q()
.where('brand', 'Audi')
.where('topSpeeed', '<=', 190)
.get()
Allowed operators:
'=', // equal
'==', // equal
'<', // less than
'<=', // less or equal
'>', // greater than
'>=', // greater or equal
'!=', // not equal
'<>', // not equal
'LIKE',
'like',
'NOT LIKE',
'not like'
Adds a WHERE NOT
clause. You can chain however many whereNots you want.
Arguments:
field
(string) - The field you are looking to match the value.value
(any) - Value you're looking up to match the whereNot clause
Example:
Car.q().whereNot('brand', 'BMW')
Adds a WHERE IN
clause. You can chain however many whereIns you want.
Arguments:
field
(string) - The field you are looking to match the value.value
(any[]) - Value you're looking up to match the whereIn clause
Example:
Car.q().whereIn('brand', ['BMW', 'Honda'])
Adds a WHERE NOT IN
clause. You can chain however many whereNotIns you want.
Arguments:
field
(string) - The field you are looking to match the value.value
(any[]) - Value you're looking up to match the whereNotIn clause
Example:
Car.q().whereNotIn('brand', ['Mercedes-Benz', 'Toyota'])
Adds a WHERE BETWEEN min AND max
clause. You can chain however many whereBetweens you want.
Arguments:
field
(string) - The field you are looking to match the values.min
(any[]) - first (lowest => min) value of the whereBetweenmax
(any[]) - last (highest => max) value of the whereBetween
Example:
Car.q().whereBetween('topSpeed', 190, 250)
Adds a WHERE NOT BETWEEN min AND max
clause. You can chain however many whereNotBetweens you want.
Arguments:
field
(string) - The field you are looking to match the values.min
(any[]) - first (lowest => min) value of the whereNotBetweenmax
(any[]) - last (highest => max) value of the whereNotBetween
Example:
Car.q().whereNotBetween('topSpeed', 180, 199)
Adds a WHERE NULL
clause. You can chain however many whereNulls you want.
Arguments:
field
(string) - The field you are looking to match the null values.
Example:
Car.q().whereNull('topSpeed')
Adds a WHERE NOT NULL
clause. You can chain however many whereNotNulls you want.
Arguments:
field
(string) - The field you are looking to match the null values.
Example:
Car.q().whereNotNull('topSpeed')
Offsets the results.
Arguments:
count
(integer) - Number of objects to be skipped.
Example:
// will start from the 11th car
Car.q().where('brand', 'BMW').offset(10)
Limits the count of the results selected by the query.
Arguments:
count
(integer) - Number of objects to retrieve.
Example:
// will only return 5 results
Car.q().where('brand', 'BMW').limit(5)
Adds an order by clause to the query. You can add multiple orderBys to your query.
Arguments:
field
(string) - Field you are ordering bydirection
(string) -ASC
for ascending,DESC
for descending.ASC
is the default value.
Example:
// ... ORDER BY brand ASC
Car.q().orderBy('brand')
// ... ORDER BY brand ASC
Car.q().orderBy('brand', 'ASC')
// ... ORDER BY brand DESC
Car.q().orderBy('brand', 'DESC')
Returns the first entry from the query results.
Arguments:
fields
(string|string[]) - Defaults to*
. You can specify the fields you want to retrieve, such as:
// will run SELECT * FROM ... LIMIT 1
let firstCar = Car.q().first()
// will run SELECT brand FROM ... LIMIT 1
let firstCar = Car.q().first('brand')
// will run SELECT brand, topSpeed as maxSpeed FROM ... LIMIT 1
let firstCar = Car.q().first(['brand', 'topSpeed as maxSpeed'])
// will run SELECT * FROM ... WHERE brand IN ['BMW', 'Honda'] ... LIMIT 1
let firstCar = Car.q().whereIn('brand', ['BMW', 'Honda']).first()
Returns the count of entries matching the query results.
Arguments:
field
(string|string[]) - Defaults todocType
. You can specify the fields you want to retrieve, such as:
// will run SELECT COUNT(docType) as count FROM ...
Car.q().count()
// will run SELECT COUNT(brand) as count FROM ...
Car.q().count('brand')
Executes a query. Call this method at the end of your query chain. Returns a promise with an array of mapped instances.
Arguments:
fields
(string|string[]) - Defaults to*
. You can specify the fields you want to retrieve, such as:
// will run SELECT * FROM ...
let results = Car.q().get()
// will run SELECT * FROM ... WHERE brand IN ['BMW', 'Honda'] ...
let results = Car.q().whereIn('brand', ['BMW', 'Honda']).get()
Provides all entries.
Arguments:
fields
(string|string[]) - Defaults to*
. You can specify the fields you want to retrieve, such as:
// will run SELECT * FROM ...
Car.q().all()
// will run SELECT brand FROM ...
Car.q().all('brand')
// will run SELECT brand, topSpeed as maxSpeed FROM ...
Car.q().all(['brand', 'topSpeed as maxSpeed'])
Paginates the results depending on the perPage count and page number.
Arguments:
perPage
(number) - Number of entries to be retrieved per page. Defaults to15
pageNumber
(number) - Page number. Defaults to1
fields
(string|string[]) - Defaults to*
. You can specify the fields you want to retrieve, such as:
// will run SELECT * FROM ... OFFSET 0 LIMIT 15
let results = Car.q().paginate()
// will run SELECT brand, topSpeed FROM ... OFFSET 10 LIMIT 10
let results = Car.q().paginate(10, 2, ['brand', 'topSpeed'])
To create a Model, you need to have a defined Schema first.
// Define a new schema for a Car.
const CarSchema = new Schema('Car', {
brand: Type.string(),
color: Type.string()
})
// Build a Model from the Schema.
const Car = settee.buildModel(CarSchema)
The Car
constant can now utilize Model methods scoped to the Car document type.
Available methods:
Creates a new mapped instance, and inserts the data
as a single entry in the database.
Arguments:
data
(Object) - Data to be inserted. The data will be validated against the Schema layout to prevent Type mismatch and invalid values.
Example:
let bmw = Car.create({
brand: 'BMW',
color: 'blue'
})
Adds custom methods to the model.
Arguments:
methods
(Object) - Object containing custom functions to be added.
Example:
Car.addMethods({
findFastestCars () {
// 'this' context is bound to the model
return this.q()
.where('topSpeed', '>', 190)
.orderBy('topSpeed', 'DESC')
.limit(10)
.get()
}
})
// now you can call
let fastestCars = await Car.findFastestCars()
Adds custom methods to the instance.
Arguments:
methods
(Object) - Object containing custom functions to be added.
Example:
// assume we have a person model, with firstName and lastName
Person.addInstanceMethods({
getFullName () {
// 'this' context is bound to the instance
return this.firstName + ' ' + this.lastName
}
})
// now you can call this
let joe = await Person.create({
firstName: 'Joe',
lastName: 'Something'
})
joe.getFullName() // Joe Something
Executes a raw query.
Arguments:
query
(string) - N1QL querybindings
(Object) - an object containing bindingsoptions
(Object) - an object containing additional options
Example:
let entries = await Car.rawQuery(
'SELECT * FROM `default` WHERE `brand` = $brand',
bindings: {
brand: 'BMW'
}
)
Provides a single Couchbase entry by its key.
Arguments:
key
(string) - Raw key of the entry. Consists ofdocType
,separator (::)
anddocId
Example:
let car = await Car.findRawByKey('Car::123')
Provides a mapped instance by ID.
Arguments:
id
(string) -docId
of the entry
Example:
let car = await Car.findById('123')
Provides multiple Couchbase entries by their keys.
Arguments:
key
(string[]) - Array of raw keys of the entries. The key for individual entries consists ofdocType
,separator (::)
anddocId
Example:
let cars = await Car.findMatchingKeys(['Car::123', 'Car::456'])
Provides multiple mapped instances by their IDs.
Arguments:
id
(string[]) - Array of entries with theirdocId
Example:
let cars = await Car.findMatchingIds(['123', '456'])
Deletes a Couchbase entry by ID.
Arguments:
id
(string) -docId
of the entry
Example:
await Car.deleteById('123')
To access the Instance methods, you need to have access to a model:
// Create a new schema for a Car.
const CarSchema = new Schema('Car', {
brand: Type.string(),
color: Type.string()
})
// Build a Model from Schema.
const Car = settee.buildModel(CarSchema)
let bmw = Car.create({
brand: 'BMW',
color: 'blue'
})
The bmw
variable can now utilize an instance class methods.
Note: QueryBuilder methods
get()
andall()
also return an array of instances, whilefirst()
will return a single instance.
Available methods:
Provides instance ID (UUID).
Example:
bmw.getId() // 123
Provides instance type (derived from Schema name).
Example:
bmw.getType() // 'Car'
Provides instance key.
Example:
bmw.getKey() // 'Car::123'
Provides instance CAS.
Saves the changed instance. Changes on the instance will be validated against the Schema layout to prevent Type mismatch and invalid values.
Example:
bmw.color // 'blue'
// respray to a hot pink color
bmw.color = 'hot pink'
await bmw.save()
bmw.color // 'hot pink'
Deletes the instance, and respective entry bound to the instance from the database.
Example:
await bmw.delete() // deleted!
Provides the latest data of the instance.
Example:
// Will be an object with these properties
// docId: 123
// docType: 'Car'
// brand: 'BMW'
// color: 'blue'
let bmwData = bmw.getData()
Rudolf Halas | Jakub Homoly |
Thanks for your interest in Settee! If you'd like to contribute, please read our contribution guide.
Settee is open-sourced software licensed under the ISC license. If you'd like to read the license agreement, click here.