SkillAgentSearch skills...

Hedy

Functional ORM

Install / Use

/learn @StephanHoyer/Hedy
About this skill

Quality Score

0/100

Supported Platforms

Universal

README

hedy - a functional ORM

Join the chat at https://gitter.im/StephanHoyer/hedy Build Status

Let's face it: Node's ORMs are crap. Mostly at least. But JavaScript is an awesome language. So why? Because we did it wrong. To much class-ish stuff like backbone. And prototypal inheritance, which is bad.

We want to fix this!

Initialisation

const hedy = require('hedy')
const mem = require('hedy/adapter/mem')

const store = hedy(adapter)

The adapter is a function that is called when a query is fired. It gets the query options-object as parameter. This is a database-specific function. In this case we user the build in memory adapter. In the future we want to create adapter for various databases. The goal is that it's really easy to create one for you specific datastore.

See the existing adapters how this works.

Creating Models/Collections/Whatever-you-call-them

..., we just call it query.

const userQuery = store('user') // 'user' is the tablename

The query is a monad-ish structure. You can call methods on it and it returns another query with the method applied. The original query remains untouched. Unter the hood this is done using patchinko.

Fetch array of things

// Array of POJOs containing user data
const users = await userQuery.where(where).load()

The where can be omitted, in this case all items where fetched. The query will only run, if you call load on it.

There are also some utility functions that might be usefull. They can be called on the query without actually fetching it.

// Array of POJOs containing user data
const users = await userQuery
  .where(where)
  .map(fn1)
  .filter(fn2)
  .load()
})

The map and filter will be applied onto the fetched collection in the order as they are applied to the query. They may return a promise.

Because all that is lazy you might do the following:

const usersWithLongNames = userQuery.filter(user => user.name.length > 10)

const femaleUsersWithLongNames = usersWithLongNames.filter(
  user => user.gender === 'female'
)

const kidsWithLongNames = usersWithLongNames.filter(user => user.age < 10)

const namesOfKidsWithLongNames = kidsWithLongNames
  .columns(['name'])
  .map(user => user.name)

// if you now need the names of the kids:
// there you have it.
const names = await namesOfKidsWithLongNames.load()

Sure, in some of those cases the database might do the heavy lifting, but there are cases where code can express much more then a database query. Also code reuse and composition can lead to great improvements here.

Currently there are only a few methods build in: map, reduce, groupBy, indexBy. It's easy to add your own method there. In the upper example lodashs pluck might be a good choice to get the user name:

const store = hedy(adapter, {
  methods: {
    pluck: require('lodash/collection/pluck'),
  },
})

In this case we add the pluck-function of lodash. Now we can do

const usernames = await userQuery.pluck('name').load()
// usernames = ['heiner', 'klaus', 'birgit']

Fetch one thing by pk

const user = await userQuery.get(id)
// user = { id: 1, name: 'Heiner' }

Fetch one first

const user = await userQuery.where(where).first()
// user = { id: 1, name: 'Heiner' }

select columns (with aliasing)

const user = await userQuery
  .columns({ username: 'name' })
  .where(where)
  .get(1)
// user = { id: 1, username: 'Heiner' }

Counting

const count = await userQuery.count()
// count = 5

Create one thing

const user = await userQuery.post(data)
// user = { id: 1, name: 'Heiner' }

Update one thing

const user = await userQuery.put(id, data)
// user = { id: 1, name: 'Heiner' }

You can also patch stuff:

const user = await userQuery.patch(id, data)
// user = { id: 1, name: 'Heiner' }

delete one thing

await userQuery.del(id)

Relations

Relations are a key part of ORMs. In most ORMs relations can only be in the same database. Hedy as a different approach on this. Relations are defined as querys. Let's look at an example:

const data = {
  user: [
    { id: 1, name: 'heiner' },
    { id: 2, name: 'klaus' },
    { id: 3, name: 'manfred' },
  ],
  friend: [{ user1Id: 1, user2Id: 2 }, { user1Id: 2, user2Id: 3 }],
  comment: [
    { id: 1, userId: 2, text: 'gorgeous' },
    { id: 2, userId: 3, text: 'nice' },
    { id: 4, userId: 1, text: 'splended' },
    { id: 5, userId: 2, text: 'awesome' },
  ],
}
const adapter = memAdapter(data)
const store = hedy(adapter)

const userQuery = store('user')
const commentQuery = store('comment')
const friendQuery = store('friend')

Here we have a memory db containing users and their comments.

To fetch the users with the comments we do:

userQuery.withRelated(hedy.hasMany(commentQuery)).then(users => ...)

As you see, we use a helper to declare a to-many-relation and give a query as parameter. The query does not have to request to the same database, so this is perfectly possible.

const memQuery = hedy(memAdapter(data))
const pgQuery = hedy(pgAdapter(config))

const userQuery = memQuery('user')
const commentQuery = pgQuery('comment')

userQuery.withRelated(hedy.hasMany(commentQuery)).then(users => ...)

For a more advanced example see the examples in the examples folder.

Has-many

const users = await userQuery.withRelated(hedy.hasMany(commentQuery)).load()

Has-one

TODO Add example

Belongs-to

const comments = await commentQuery
  .withRelated(hedy.belongsTo(userQuery))
  .load()

Many-to-many

const users = await user
  .withRelated(hedy.hasManyThrough(userQuery, friendQuery))
  .load()

For many-to-many-Relations there are two helper functions to create and delete them:

const friendship = await friends.link(userA, userB)
await friends.unlink(userA, userB)

Current state

This is pretty much WIP. The basics are done. Now it's time to write the adapters. First one will be a memory-adapter. Then we first add a postgres adapter since this is the DB we're using in our project.

Hope you like it.

If you have any ideas, feel free to create a PR/Issue.

links

  • Pretty similar attempt to hedy by the bookshelf maintainer: https://github.com/rhys-vdw/data-mapper
View on GitHub
GitHub Stars7
CategoryDevelopment
Updated4y ago
Forks1

Languages

JavaScript

Security Score

65/100

Audited on Feb 23, 2022

No findings