elastic-lucid-provider v3.0.1
ELASTIC LUCID PROVIDER
Gives nice lucid-like way for handling elasticsearch queries in adonis framework.
It is built upon official elastic npm client and uses bodybuilder for query building.
While elasticsearch is awesome, when dealing with it you will realise that API it provides is sometimes weird and requires a lot of work.
This provider aims to provider better and easier API for dealing with elasticsearch API
INSTALLATION
- Install package
npm i --save elastic-lucid-provider
- Register provider
const providers = [
'elastic-lucid-provider/Providers/ElasticLucidProvider',
]
const aceProviders = [
'elastic-lucid-provider/Providers/CommandsProvider',
]
- Add
config/elasticLucid.js
const Env = use('Env')
const Helpers = use('Helpers')
module.exports = {
connection: {
host: Env.get('ELASTIC_HOST'),
log: Env.get('ELASTIC_LOG', null),
vpcActive: Env.get('ELASTIC_VPC_ACTIVE', false),
username: Env.get('ELASTIC_USER', null),
password: Env.get('ELASTIC_PASS', null)
},
migrationsDirectory: Helpers.appRoot('Elastic/Schema'),
schemaIndex: 'gb_schema'
}
Note: You don't need to provide all this info. If you have elastic running on local machine accessible from specific ip, no need to supply username or password.
MIGRATIONS
The provider stores information about run migrations in separate index called app_schema
Migrations need to be stored in directory specified in config file. (will be configurable in the future). If you registered commands provider, make new schema by running command:
adonis make:elastic_migration {name: migration name} { --model: create a model also }
Migrations look like this:
module.exports = {
index: 'users',
mappings: {
dynamic: false,
properties: {
username: 'text',
posts: {
dynamic: true,
properties: {}
}
}
},
settings: {}
}
Read more about what can be put in settings and mappings on elasticsearch documentation.
Order of each migration is:
create index if it does not exist
close index
run settings
open index
run mapping
Notice that provider gives you a little shortcut, you can simply set field type as string. It will be converted to object {type: 'string'}
. If you need anything custom, you need to pass object as specified by elasticsearch documentation.
Note Transactions are not supported, so if you have an error in migrations, part that executed will stay there. You have to revert it manually.
MODELS
Unlike migrations, there is no required folder for models. They are just classes that extend BaseModel class.
They are built to have as similar API to lucid as possible.
Defining models
Model files look like this:
const ElasticModel = use('ElasticLucid/Model')
class User extends ElasticModel{
static get index() {
return 'users'
}
static get resourceType() {
return 'User' //defaults to model name
}
static get resourceTypeField() {
return 'resourceType' //defaults to resourceType
}
}
module.exports = User
They need to have index getter on them and extend BaseModel which is bound to ElasticLucid/Model
Use it as any other adonis entity.
Model instances
Get new instance simply by new Model(body, id)
. If id is provided, it is automatically assumed that record with that id exists in elastic.
Much like with lucid, you can Model.create({...body})
, Model.find(id)
, Model.findOrFail(id)
, Model.findBy(fieldInBody, value)
, Model.findByOrFail(id)
When you have instance, you can edit its attributes and run instance.save()
. It will create new record or update existing (depending if it has id or not).
Also, you can run instance.delete()
to delete instance from elastic
Also like in lucid, you can run instance.merge(body)
to merge object into body.
Main difference between lucid and elastic lucid api:
Attributes are nested inside body object.
So unlike lucid, where you would do something like user.name = 'joe'
, here you would do it like
user.body.name = 'Joe'
console.log(user.body.lastname)
user.id //id attribute is special as it is not nested in body
This change needed to be made, because unlike most sql databases, elastic does not treat id same as other attributes
Static getters
You can define static getters for resourceType
and resourceTypeField
, and those will be implicitly added when creating elastic documents. Set any to null if you don't want to have them. However, resourceType is useful when building global search, and searching over multiple indexes.
Serializers
Like in lucid, you can use Model.all()
and various query builder methods to get multiple instances
Those methods will return instance of serializer defined on model. You can defined your own (see below) or use base serializer. It will look like this
{
pages: {total: 20, isOne: false},
aggregations: {}, //aggregation results, if any
rows: [] //array of models
}
On any model or serializer instance you can apply .toJSON method to get serialized data.
Query builder
Access query builder simply by Model.query()
method. It will return instance of bodybuilder.
Learn more about bodybuilder methods here. Everything used by bodybuilder can be used here
So query would look something like:
User.query().query('match', 'name', 'Joe')
Note Notice this weird query().query? First one is model method to get bodybuilder instance. Second one comes from bodybuilder.
Even thou it is confusing, nomenclature is like this to be as similar to lucid as possible.
Terminating methods
On query builder instance you can use many terminating functions that will apply some action to records that match query.
Those are:
- fetch
- paginate
- first
- nativeResult
- update
- iterate
fetch
, paginate
and first
, update
and delete
work much the same way as they do in lucid.
nativeResult
will return result as returned by npm elastic adapter without any modification.
So, the code would look much like in lucid:
let users = await User.query().query('match', 'name', 'Joe').paginate(1, 10)
for(let user of users.rows) {
user.body.name = 'Pete'
await user.save() //this is just for showcase, updating inside loop is bad idea
}
return users.toJSON()
await User.query().query('match', 'name', 'Joe').update({
surname: 'Doe',
"something.nested": "anotherValue"
})
iterate
method
Iterating over many documents in elasticsearch can be a lot of trouble.
On small number of documents, you can simply .fetch()
query and iterate over result.rows
However, elasticsearch will never return to you more than 10 000 results, and if you need to get that many documents, you would need to use elastic's scroll api. That is where this method comes in.
.iterate will do everything about scroll api for you and run provided iteratee for every record matching the query, even if there are more than 10 000
await User.query().query('match', 'name', 'joe').iterate(async modelInstance => {
modelInstance.body.name = _.upperFirst(modelInstance.body.name)
await modelInstance.save()
})
Note This operations are usually memory and processor heavy and require a lot of time. If you can avoid this approach you should.
Model.raw
If you get into situation that bodybuilder cannot build body you need, you can supply raw body. It will still return model the usual model holder from matching results
let models = Model.raw({
query: {
match_all:{}
}
})
return models.toJSON()
MULTI SEARCH
Model.rawMany()
runs elastic's msearch for provided query bodies - runs multiple search queries in one elastic call. Returns arrays of model holders for each of those search.
No need to build weird object that elastic accepts for msearch - just pass actual search body objects, and provider will handle the rest.
Accepts raw search bodies - if you want to use bodybuilder you still can, just .build()
on bodybuilder.
const differentSearchesOnSameIndex = await User.rawMany([
{
query: {match_all: {}}
},
{
query: {
term: {
type: {
value: 'admin'
}
}
}
},
User.query().query('match', 'name', 'joe').build()
//or build this by bodybuilder
])
const admins = differentSearchesOnSameIndex[1]
for(let admin of admins.rows) {
sendEmail(admin.body.email)
}
BULK ACTION
Runs elastic's bulkAction, but with cleaner api and help of elastic models.
Model.bulkAction(body, ...pluckFields)
accepts body of bulk action. It is different from elastic bulkAction body, will be shown in example
If action is update, and you want to update only some params, passing those params as second argument will speed up the process (but it is optional).
const john = new User({name: 'Jon'}) // this one will be created
const jane = await User.find('ejkjdsks')
jane.body.name = 'Jane' //this one deleted
const paul = await User.query('match', 'name', 'paul').first()//this one will be deleted
await User.rawMany([
{action: 'create', modelInstance: john},
{action: 'update', modelInstance: jane},
{action: 'delete', modelInstance: paul},
])
CUSTOM SERIALIZERS
You can define your serializer per model. Serializer would look like this:
class MySerializer {
constructor(rows, aggregations, pages) {
this.pages = pages
this.aggregations = aggregations
this.rows = rows
}
toJSON() {
//return serialized object
}
}
module.exports = MySerializer
Then, in model you would define static getter like:
static get Serializer() {
return 'App/Serializers/MySerializer'
}
TRAITS
Traits work in much the same way as in lucid. Trait is class with register method on it, which accepts model and configuration. You can apply it inside boot on model. Also, you can extend query builder with your own methods with queryMacro.
class MyTrait {
register(Model, options={}) {
Model.queryMacro('whereUser', function(user) {
this.filter('term', 'user', user)
return filter
})
Model.prototype.methodOnInstance = function() {
//
}
Model.staticMethod = function() {
//
}
}
}
module.exports = MyTrait
Inside model, you can get call trait
static boot() {
super.boot()
this.addTrait('App/Traits/MyTrait', {/*options*/})
}
You can supply full path, path relative to App/Models/Traits/Elastic/
or pointer to trait class.
USING OUTSIDE OF ADONIS
This provider is not dependent on adonis framework. However, there are couple of things to keep in mind
- You need to call server init function on server init and pass configuration (configuration is like config file you would make in adonis)
const elasticLucid = require('elastic-lucid-provider')
elasticLucid({
//same configuration as you would add to config/elasticLucid.js
})
Get BaseModel by
require('elastic-lucid-provider/src/Models/Base')
Ace commands won't work outside of adonis. Run migrations like this:
const runMigrations = require('elastic-lucid-provider/src/RunMigrations')
runMigrations()
- If using traits, you need to call boot method on your own.
Like in lucid, this provider makes use of iocHooks to run boot code. If outside of adonis, you need to call it some other way. Maybe easiest would be to call it before module.exports
const Model = require('elastic-lucid-provider/src/Models/Base')
class MyModel extends Model {
static boot() {
//add trait or something
}
}
MyModel._bootIfNotBooted()
module.exports = MyModel
- Where you would provide ioc path, provide pointer instead. For example
const Model = require('elastic-lucid-provider/src/Models/Base')
const MyTrait = require('./MyTrait')
class MyModel extends Model {
static boot() {
this.addTrait(MyTrait)
}
}
MyModel._bootIfNotBooted()
module.exports = MyModel
Thanks
Special thanks to creators of all the frameworks and libraries used in this library
4 years ago
4 years ago
4 years ago
4 years ago
4 years ago
4 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
6 years ago
6 years ago
6 years ago
6 years ago
6 years ago
6 years ago
6 years ago
6 years ago
6 years ago
6 years ago
6 years ago
6 years ago
6 years ago
6 years ago