YORM is a no-nonsense, super-simple and light-weight ORM built on top of Knex and inspired by Laravel's Eloquent ORM.
$ npm install yorm.js
// CommonJS
const { Model } = require('yorm.js')
// ESM
import { Model } from 'yorm.js'
class Example extends Model {
hello = 'Hello'
world = 'World'
toString() {
return `${this.hello}, ${this.world}!`
}
}
const example = Example.make()
console.log(`${example}`) // Hello, World!
Imagine writing migrations like...
function up(knex) {
return knex.schema.createTable('users', table => {
table.uuid('id').defaultTo(knex.fn.uuid()).primary()
table.string('email').unique()
table.string('name')
table.timestamps(false, true)
})
}
function down(knex) {
return knex.schema.dropTable('users')
}
Then... imagine writing a class like...
class User extends Model {
id
email
name
created_at
updated_at
}
... and then, imagine you could just start CRUD-ing from the users
table ... 🦄
// Add a new user to the database!
const user = await User.create({
email: '[email protected]',
name: 'Susan Example',
})
// Make a change!
user.name = 'S. Example'
// Persist the change!
await user.save()
// Fetch by primary key!
await User.find(user.id)
// Fetch by whatever you want!
await User.where({ email: '[email protected]' })
// Delete that jank!
await user.delete()
// Count and stuff!
await User.count()
What if folks start adding random properties here and there? HOW IS YORM GONNA HANDLE THAT?!
const user = await User.create({
email: '[email protected]',
name: 'Susan Example',
// This will throw an error...
badProperty: true
})
// ... as will this.
user.badPropertyAfterTheFact = true
Before any instance of a model is returned, we seal the object to prevent addition (or removal) of properties from the object. Only properties explicitly declared on the model are allowed... for now.
YORM models don't have constructors... or rather... they do... but they're private and throw a TypeError
when you try to use them. 99% of the time, you'll just User.create(...)
and be on your way. If you want an instance of a user without saving to the database, call User.make(...)
instead.
By default, model table names are computed to be a pluralization of the model name:
- A class named
User
will map to a table namedusers
Comment
will map tocomments
BirdOfPrey
will map tobirdofpreys
However, you can always override this in your own model by adding a tableName
accessor:
class User extends Model {
id
username
email
get tableName() {
return 'maj' // Klingon for "Well Done!"
}
}
Well good freakin' news... YORM lets you do whatever you want... seriously. By default, we delegate to your DBMS of choice to do the right thing when it comes to auto-incrementing or database-generated UUIDs and things like that. However, there are times where you'll want to generate an ID before persistance to the database. For that, we have the newUniqueId
accessor.
class Example extends Model {
id
get newUniqueId() {
return 'foo'
}
}
Identifiers must be returned as strings. You have control over generation of identifiers. That means it's up to you to make sure they're unique!
To make things simpler, we have some out-of-the-box support for UUID, ULID (which are lexicographically sortable), and nanoid. To use these, just return uuid
, ulid
, or nanoid
, respectively.
class Example extends Model {
id
get newUniqueId() {
return 'uuid' // '9b1deb4d-3b7d-4bad-9bdd-2b0d7b3dcb6d'
}
}
It's very, very common to want to have two models that relate to one another. Imagine a "user has posts", but a "post belongs to a user". Unfortunately, in Node, you run into all sorts of funky issues when you have two modules require one another. Instead, YORM advises to create an index.js
file wherever you keep your models at. This file will re-export all of your models and at the same time, register
all of the models with one another.
By doing this, we're allowing all the modules to load, THEN we're iterating through each of them and "telling" the others about the full collection. Check it out:
const { Model } = require('yorm.js')
// None of these modules should `require(...)` one another.
const User = require('./User.js')
const Post = require('./Post.js')
const Comment = require('./Comment.js')
module.exports = { User, Post, Comment }
Model.register(module.exports)
Of course, you are completely free to ignore this altogether and do whatever you do to manage circular dependencies. Each of the relationship mapping functions expect to be given a class reference and they don't care how that happens. As long as they can call .name
on what you pass in and new
it up, it's all good. For example, if I know I won't be putting a .belongsTo(...)
relationship from Post
to User
, I could just do this:
const { Model } = require('yorm.js')
const Post = require('./Post.js')
class User extends Model {
id
username
email
posts() {
// SELECT * FROM posts WHERE user_id = '${this.id}'
return this.hasMany(Post)
}
}
Wherever you set up your application's shared knex
instance, just call Model.bind(knex)
when it's ready. This tells YORM to use that instance of knex
.
const { knex } = require('knex')
const knexfile = require('./knexfile.js')
const { Model } = require('yorm')
Model.bind(knex(knexfile))
module.exports = knex
Under the hood, YORM models are just a set of utility functions on top of a Knex instance. We use Knex to implement transactions. Normally, this means that we would have to create a transaction context and pass that around to every model that needs to take part in the transaction.
It's going to feel weird (and we're looking for a better interface), but to avoid having to remember to bind (and unbind) the transaction context to specific models, we took the following approach:
const { transaction } = require('yorm')
/**
* Use `transaction` to start a transaction. Pass in each Model
* class you want to be available to participate in the transaction.
*
* The last argument is always a callback that receives Model classes
* (in the order you provided them) that have been bound to the
* transaction. In this way, the Model classes are able to participate
* in the transaction within the scope of the provided callback.
*/
await transaction(User, Post, async (User, Post) => {
// User inside the closure is not the same as User outside
// the closure. The inside User has the Knex transaction bound.
const user = await User.create()
await Post.create({ user_id: user.id, title: 'Created if the User is successfully created.' })
// Any exception / error thrown inside the closure will rollback. Otherwise,
// the transaction is implicitly committed.
})
It's somewhat common to support features for "undo"-ing deletes. This is usually accomplished by replacing DELETE FROM {table} WHERE ...
statements with an UPDATE SET deleted_at = NOW() WHERE ...
and then having every query function account for this field. If deleted_at
is NULL
, the record exists. Otherwise, you have the date and time that the record was deleted.
class SoftDelete extends Model {
id
deleted_at
get softDeletes() { return true }
}
const model = await SoftDelete.create()
await model.delete() // UPDATE softdeletes SET deleted_at = NOW() WHERE id = 'foo'
Customizing the field name
You can override the default deleted_at
field name can be done globally or per-model. You can also set a default name globally and then override in specific models.
To override globally, use the deletedAtColumn
option when you call Model.bind(...)
:
Model.bind(knex, {
deletedAtColumn: 'deletedAt'
})
To set this value per-model, override the deletedAtColumn
accessor:
get deletedAtColumn() {
return 'deleted_date'
}
Restoring deleted models
If you have an instance of a model that was just deleted, you can call its .restore()
method and it will be restored.
More commonly, you'll be recovering a model that was deleted in the past where you do not have an instance. In these cases, you can use the static version of the same method to restore ALL models matching specific criteria:
// Restore all users deleted on or after Jan 1st, 2022
await User.restore(query => {
query.where('deleted_at', '>=', '2022-01-01')
})
Model.restore(...)
accepts a callback which is given the current Knex query builder instance. Anything you can do with Knex as far as querying goes can be done here. Just don't do weird stuff like query.where(...).delete().select('*')
. I'm not going to stop you, but that probably blows chunks.
Thought you'd never ask! YORM supports the usual relationship types:
- One to One
- One to Many
- Belongs To
YORM does not support "Many to Many" relationships yet because honestly... haven't needed em' in a while and I'm just slapping this together. It'll be added for sure.
class User extends Model {
id
username
email
posts() {
// Normally, hasMany would compute that the foreign key on
// the posts table should be `user_id`, referencing this model's
// singularized table name. However, we can also override that
// behaviour like so.
return this.hasMany(this.models.Post, 'author_id')
}
comments() {
// SELECT * FROM posts WHERE author_id = '${this.id}'
return this.hasMany(this.models.Post, 'author_id')
}
}
There's a couple things going on up there. First, we're modeling two relationships: A user has-many posts and a user has-many comments. You see two different flavors for how this can be done. In the simplest case (not shown), you only have to return this.hasMany(...)
with a single argument that references the related model. The ORM will inspect that model for its computed table name and follow a convention that the foreign-key on the related table will point back to {tableName}_id
. So, in this case, it would NORMALLY point back to users.user_id
. However, in our example, the posts
and comments
tables named that field author_id
. The second argument to hasMany
allows you to override that foreign key name.
class User extends Model {
id
username
email
photo() {
// SELECT * FROM photos WHERE photos.user_id = '${this.id}' LIMIT 1
return this.hasOne(this.models.Photo)
}
}
class Post extends Model {
id
author_id
title
content
created_at
updated_at
user() {
// SELECT * FROM users WHERE id = '${this.author_id}'
return this.belongsTo(this.models.User, 'author_id')
}
}
A "Belongs To" relationship is the inverse of a One-to-Many relationship. If a User has-many Posts, then this is a way to get an individual Post's author through that same relationship.
There are scenarios where you need to have a property on a model that shouldn't be shown in your API. For example, the password
field on a User
model.
You can already override a model's toJSON()
method to support this use-case, but you have to remember to do it and when you're only hiding a single property, it feels like a lot of boilerplate.
If you define a hidden
accessor on your model that returns an array of field names, they will automatically be omitted from JSON output.
class User extends Model {
id
username
password
get hidden() {
return ['password']
}
}
const user = User.make({ username: 'user', password: 'super.secret' })
JSON.stringify(user) // { "username": "user" }
Imagine two requests modifying the same property on a model at the same time. Which one wins? How do you prevent this?
YORM provides an easy-to-follow optimistic locking strategy through the use of ETag (Entity Tag). An ETag is an identifier that represents a specific version (or state) of a model. This means that if you query the ETag of a model instance, change a property, then query the ETag again, that you will have two different ETags. This property allows us to know whether or not a model has changed since the client last queried an API.
You can communicate a resource's ETag with a client by sending the ETag
response header. Clients can make conditional requests against a resource by sending an If-Match
header set to the ETag
value.
// controllers/users.js
// GET /users/{id}
handlers.get = async (req, res) => {
const user = await User.find(req.params.id)
res.header('ETag', user.etag)
res.json(user)
}
// PUT /users/{id}
handlers.update = async (req, res) => {
const user = await User.find(req.params.id)
if (req.headers['If-Match'] !== user.etag) {
// 412 Precondition Failed
throw new errors.PreconditionFailed(`User has been updated since last seen.`)
}
// Eehhhhhh, maybe don't do this. Mass assignment es peligrosa.
// But... I'm not going to stop you.
Object.assign(user, req.body)
await user.save()
}