Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/gabrielcsapo/node-flat-db

🚫 a nodejs embedded db
https://github.com/gabrielcsapo/node-flat-db

deprecated

Last synced: 20 days ago
JSON representation

🚫 a nodejs embedded db

Awesome Lists containing this project

README

        

> DEPRECATED: THIS WILL NO LONGER BE UNDER DEVELOPMENT

# node-flat-db

[![Build Status](https://travis-ci.org/gabrielcsapo/node-flat-db.svg?branch=master)](https://travis-ci.org/gabrielcsapo/node-flat-db)
[![Npm Version](https://img.shields.io/npm/v/node-flat-db.svg)](https://www.npmjs.com/package/node-flat-db)
[![Dependency Status](https://david-dm.org/gabrielcsapo/node-flat-db.svg)](https://david-dm.org/gabrielcsapo/node-flat-db)
[![devDependency Status](https://david-dm.org/gabrielcsapo/node-flat-db/dev-status.svg)](https://david-dm.org/gabrielcsapo/node-flat-db#info=devDependencies)
[![npm](https://img.shields.io/npm/dt/node-flat-db.svg)]()
[![npm](https://img.shields.io/npm/dm/node-flat-db.svg)]()

> This is a fork of [lowdb](https://github.com/typicode/lowdb) before they broke the api and no longer is as stable as before (fork of 12.5 lowdb)

> Need a quick way to get a local database for a CLI, an Electron app, a small server or the browser?

## Example

```js
var flat = require('node-flat-db')
var storage: storage = require('node-flat-db/file-sync')

var db = flat('db.json', { storage: storage })

db('posts').push({ title: 'node-flat-db is awesome'})
```

Database is __automatically__ saved to `db.json`.

```js
{
"posts": [
{ "title": "node-flat-db is awesome" }
]
}
```

You can query and manipulate it using __any__ [lodash](https://lodash.com/docs) __method__.

```js
db('posts').find({ title: 'node-flat-db is awesome' })
```

And access underlying database object any time.

```js
db.object.posts
```

## Installation

Using npm:

```bash
npm install node-flat-db --save
```

A standalone UMD build is also available on [unpkg](https://unpkg.com/):

```html

var db = flat() // in-memory
var db = flat('db', { storage: flat.localStorage }) // localStorage

```

## Features

* Very small (~100 lines for core)
* lodash API
* Extendable:
* __Custom storage__ (file, browser, in-memory, ...)
* __Custom format__ (JSON, BSON, YAML, ...)
* __Mixins__ (id support, ...)
* __Encryption__

node-flat-db is also very easy to learn since it has __only a few methods and properties__.

_node-flat-db powers [json-server](https://github.com/typicode/json-server) package, [jsonplaceholder](http://jsonplaceholder.typicode.com/) website and [many other great projects](https://www.npmjs.com/browse/depended/node-flat-db)._

## Usage examples

Depending on the context, you can use different storages and formats.

node-flat-db comes bundled with `file-sync`, `file-async` and `browser` storages, but you can also write your own if needed.

### CLI

For CLIs, it's easier to use `node-flat-db/file-sync` synchronous file storage: storage .

```js
var flat = require('node-flat-db')
var storage: storage = require('node-flat-db/file-sync')

var db = flat('db.json', { storage: storage })

db('users').push({ name: 'typicode' })
var user = db('users').find({ name: 'typicode' })
```

### Server

For servers, it's better to avoid blocking requests. Use `node-flat-db/file-async` asynchronous file storage.

__Important__

* When you modify the database, a Promise is returned.
* When you read from the database, the result is immediately returned.

```js
var flat = require('node-flat-db').
var storage: storage = require('node-flat-db/file-async')

var db = flat('db.json', { storage: storage })

app.get('/posts/:id', function(req, res) {
// Returns a post
var post = db('posts').find({ id: req.params.id })
res.send(post)
})

app.post('/posts', function(req, res) {
// Returns a Promise that resolves to a post
db('posts')
.push(req.body)
.then(function(post) {
res.send(post)
})
})
```

### Browser

In the browser, `node-flat-db/browser` will add `localStorage` support.

```js
var flat = require('node-flat-db')
var storage: storage = require('node-flat-db/browser')

var db = flat('db', { storage: storage })

db('users').push({ name: 'typicode' })
var user = db('users').find({ name: 'typicode' })
```

### In-memory

For the best performance, use node-flat-db in-memory storage.

```js
var flat = require('node-flat-db')
var db = flat()

db('users').push({ name: 'typicode' })
var user = db('users').find({ name: 'typicode' })
```

Please note that, as an alternative, you can also disable `writeOnChange` if you want to control when data is written.

## Compression

> compresses 1000 record db from `77kb` to `7kb`

The best way to use node-flat-db if space constraint is an issue, use `lzstring` adapter

```js
var flat = require('node-flat-db')
var storage: storage = require('node-flat-db/file-compress-sync')

var db = flat('db', { storage: storage })

db('users').push({ name: 'typicode' })
var user = db('users').find({ name: 'typicode' })
```

## API

__flat([filename, [storage, [writeOnChange = true]]])__

Creates a new database instance. Here are some examples:

```js
flat() // in-memory
flat('db.json', { storage: /* */ }) // persisted
flat('db.json', { storage: /* */ }, false) // auto write disabled

// To create read-only or write-only database,
// set only storage.read or storage.write
var fileSync = require('node-flat-db/file-sync')

// write-only
flat('db.json', {
storage: { write: fileSync.write }
})

// read-only
flat('db.json', {
storage: { read: fileSync.read }
})
```

You can also define custom storages and formats:

```js
var myStorage = {
read: (source, deserialize) => // obj or a Promise
write: (dest, obj, serialize) => // undefined or a Promise
}

var myFormat = {
format: {
deserialize: (data) => // obj
serialize: (obj) => // data
}
}

flat(source, { storage: myStorage, format: myFormat }, writeOnChange)
```

__db.___

Database lodash instance. Use it to add your own utility functions or third-party mixins like [underscore-contrib](https://github.com/documentcloud/underscore-contrib) or [underscore-db](https://github.com/typicode/underscore-db).

```js
db._.mixin({
second: function(array) {
return array[1]
}
})

var post1 = db('posts').first()
var post2 = db('posts').second()
```

__db.object__

Use whenever you want to access or modify the underlying database object.

```js
db.object // { posts: [ ... ] }
```

If you directly modify the content of the database object, you will need to manually call `write` to persist changes.

```js
// Delete an array
delete db.object.posts
db.write()

// Drop database
db.object = {}
db.write()
```

__db.write([source])__

Persists database using `storage.write` method. Depending on the storage, it may return a promise.

Note: by default, node-flat-db automatically calls it when database changes.

```js
var db = flat('db.json', { storage: storage })
db.write() // writes to db.json
db.write('copy.json') // writes to copy.json
```

__db.read([source])__

Reads source using `storage.read` method. Depending on the storage, it may return a promise.

```js
var db = flat('db.json', { storage: storage })
db.read() // re-reads db.json
db.read('copy.json') // reads copy.json
```

## Guide

### How to query

With node-flat-db, you get access to the entire [lodash API](http://lodash.com/), so there is many ways to query and manipulate data. Here are a few examples to get you started.

Please note that data is returned by reference, this means that modifications to returned objects may change the database. To avoid such behaviour, you need to use `.cloneDeep()`.

Also, the execution of chained methods is lazy, that is, execution is deferred until `.value()` is called.

#### Examples

Sort the top five posts.

```js
db('posts')
.chain()
.filter({published: true})
.sortBy('views')
.take(5)
.value()
```

Retrieve post titles.

```js
db('posts').map('title')
```

Get the number of posts.

```js
db('posts').size()
```

Make a deep clone of posts.

```js
db('posts').cloneDeep()
```

Update a post.

```js
db('posts')
.chain()
.find({ title: 'flat!' })
.assign({ title: 'hi!'})
.value()
```

Remove posts.

```js
db('posts').remove({ title: 'flat!' })
```

### How to use id based resources

Being able to retrieve data using an id can be quite useful, particularly in servers. To add id-based resources support to node-flat-db, you have 2 options.

[underscore-db](https://github.com/typicode/underscore-db) provides a set of helpers for creating and manipulating id-based resources.

```js
var db = flat('db.json')

db._.mixin(require('underscore-db'))

var postId = db('posts').insert({ title: 'flat!' }).id
var post = db('posts').getById(postId)
```

[uuid](https://github.com/broofa/node-uuid) is more minimalist and returns a unique id that you can use when creating resources.

```js
var uuid = require('uuid')

var postId = db('posts').push({ id: uuid(), title: 'flat!' }).id
var post = db('posts').find({ id: postId })
```

### How to use custom format

By default, node-flat-db storages will use `JSON` to `parse` and `stringify` database object.

But it's also possible to specify custom `format.serializer` and `format.deserializer` methods that will be passed by node-flat-db to `storage.read` and `storage.write` methods.

For example, if you want to store database in `.bson` files ([MongoDB file format](https://github.com/mongodb/js-bson)):

```js
var flat = require('node-flat-db')
var storage: storage = require('node-flat-db/file-sync')
var bson = require('bson')
var BSON = new bson.BSONPure.BSON()

flat('db.bson', { storage, format: {
serialize: BSON.serialize,
deserialize: BSON.deserialize
}})

// Alternative ES2015 short syntax
var bson = require('bson')
var format = new bson.BSONPure.BSON()
flat('db.bson', { storage, format })
```

### How to encrypt data

Simply `encrypt` and `decrypt` data in `format.serialize` and `format.deserialize` methods.

For example, using [cryptr](https://github.com/MauriceButler/cryptr):

```js
var Cryptr = require("./cryptr"),
var cryptr = new Cryptr('my secret key')

var db = flat('db.json', {
format: {
deserialize: function(str) {
var decrypted = cryptr.decrypt(str)
var obj = JSON.parse(decrypted)
return obj
},
serialize: function(obj) {
var str = JSON.stringify(obj)
var encrypted = cryptr.encrypt(str)
return encrypted
}
}
})
```

## Changelog

See changes for each version in the [release notes](https://github.com/typicode/node-flat-db/releases).

## Limits

node-flat-db is a convenient method for storing data without setting up a database server. It is fast enough and safe to be used as an embedded database.

However, if you seek high performance and scalability more than simplicity, you should probably stick to traditional databases like MongoDB.

## Benchmarks

> `npm install microtime`

`npm run benchmarks`

## License

MIT - [Gabriel J. Csapo](https://github.com/gabrielcsapo)