** Important this package just
Very begining of development phase
it have alot of limitation to use, becareful if you plan to use on production **
The Node.Js Firestore ORM inspire by laravel eloquent, extremely easy and simple to use, super lightweight package.
and sorry about my english.
- This package support
orWhere()
, but it takeextended price cost
because firestore not have operator 'or' in where clause. every time your use 'orWhere()' the library must split request query to normal APIwhere()
and then merge union the data on every split request on your nodejs server. - This package not support firestore sub collection at all. the concept of this library your must imagine to
RDB (Relational Data Base
it justDB->TABLE->COLUMN
schema andRelational
data withforeign key
, yes no need sub collection forRDB
. - This Package not require a
composite or single field index
when your useorderBY()
to sort your query, because the library use your nodejs server to do this. - Currently version not support difining
field type
, you can workaround to edit field type manualy on gcp firestore page. - Right now support
LIKE
forwhere()
only, fororwhere()
please wait to update next version. - Support
==
,>
,<
,>=
,<=
,!=
,array-contains
,array-contains-any
,in
,not-in
onwhere()
operator same of gcp firestore. - Currently version suport only
limit()
not supportstartAt()
,endAt()
. - Currently version not support realtime snapshot but you can workaround to get model instant property
.activeTable
it will returninstant of firestore target collection
then you can follow to use firestore official nude.js api sdk like realtime snapshot or other default official api. - Support
MVC
structure project as well, you can buildModel
with this flarestore package.
To authenticate a service account and authorize it to access Firebase services, you must generate a private key file in JSON format.
To generate a private key file for your service account:
-
In the Firebase console, open Settings > Service Accounts.
-
Click Generate New Private Key, then confirm by clicking Generate Key.
-
The JSON key file will download to your computer, This key must use for .env file to nearly next step.
npm i @acetex/flarestore
Create the .env file to root of your node.js project.
#.env
FS_KEY_FILENAME=[A path/to/file.json service account JSON key file on your computer]
FS_DB_NAME=[The name of firestore root collection for your database name]
The model it mean table in RDB
for example we try to create 'users' table in models
directory
# /models/User.js
var FlareStore = require('@acetex/flarestore')
class User extends FlareStore {
table = 'users'; // define the name of table
// fillable field list
fields = {
'name': 'any', // any use for ignore validate data type for the future version
'address': '5-gram' // use n-gram field type if you want to grant "Full Text Search" for this field, the perfix number tell the library split keyword with n digit, the small prefix number grant user countless typing frequency to get search result but bad system performance coz it make document size to large and using search time result more than higher prefix number. yes... becreaful to use n-gram number keep high prefix number as posible for performance.
};
}
module.exports = User // export User model
Create the user
# /index.js
var User = require('./models/User.js');
async function createUser(){
const user = new User;
const params = {
name: 'firstname lastname',
address: 'example to use n-gram data type for Full Text Search'
};
await user.create(params); // when you create a new data record, the '__id' and '__timestamp' is automatic generated.
}
createUser();
Get the user and where query then update current user
# /index.js
var User = require('./models/User.js');
async function getUser(){
const user = new User;
user.where('name', '==', 'firstname lastname');
user.orWhere('address', 'customer address');
user.orderBy('__timestamp', 'desc');
let users = await user.get();
users[0].name = 'firstname lastname changed!';
users[0].save();
}
getUser();
Search like
operator with FULL TEXT SEARCH
this ability only use when the field type set to n-gram
eg. 2-gram
, 3-gram
...
this version support where()
only, for orWhere()
wait next update version.
# /index.js
var User = require('./models/User.js');
async function getUser(){
const user = new User;
user.where('address', 'like', 'n-gram data type for Full Text search'); // no need type % block
console.log(await user.get());
}
getUser();
Delete Record
# /index.js
var User = require('./models/User.js');
async function deleteUser(){
const user = new User;
let users = await user.limit(3).get(); // limit only 3 record
users[2].delete(); // delete user record by index position 3
}
deleteUser();
Relational Data
First: modify User
model by adding post()
with User->hasMany->Post
and create new Post
model with Post->belongTo->User
# /models/User.js
var FlareStore = require('@acetex/flarestore')
class User extends FlareStore {
table = 'users'; // define the name of table
// fillable field list
fields = {
'name': 'string',
'address': '5-gram'
};
post(){
return this.hasMany('models/Post', 'users_id', '__id');
// you can type just 'this.hasMany('models/Post')' if your foreignKey is {table name}_id
}
}
module.exports = User // export User model
# /models/Post.js
var FlareStore = require('@acetex/flarestore')
class Post extends FlareStore {
table = 'posts'; // define the name of table
fields = ['title', 'description', 'uses_id'];
fields = {
'title': '3-gram',
'description': '4-gram', // can use 2-gram, 3-gram, 4-gram ... n-gram
'uses_id': 'string' //assume this key reference to 'user.__id' and the 'user.__id' auto generate to string data type by default.
};
user(){
this.belongTo('models/User');
}
}
module.exports = Post // export Post model
Secord: create new 'Post' record with 'users_id' foreignKey
# /index.js
var Post = require('./models/Post.js');
async function createPost(){
const user = new User;
let userData = await user.first(); // 'first()' return only first one record with 'object' not 'array of object'
const post = new Post;
const params = {
title: 'test title',
description: 'test description',
users_id: userData.__id
};
await post.create(params); // when you create a new data record, the '__timestamp' field is automatic generated.
}
createPost();
Third: try to retrieving relation model;
# /index.js
var User = require('./models/User.js');
var Post = require('./models/Post.js');
async function getUser(){
//example with hasMany
const user = new User;
let userData = await user.first();
let posts = await userData.post().get();
console.log(posts);
// '.post()' is relation method to 'Post' model it will return 'Post' model object, of course you can use all property/method of 'Post' model like where(), orWhere(), orderBy(), etc. just like example below.
let posts = await userData.post().where('description', 'test description').get();
console.log(posts);
//example with belongTo
const postBelongTo = new Post;
let postBelongToData = await postBelongTo.first();
let belongToUser = postBelongToData.user().first();
console.log(belongToUser);
}
getUser();
Note!
hasOne()
use the same mechanic ashasMany()
but have some differencereturn type
withget()
method,hasOne()
return single object andhasMany()
return array of object