UNPKG

@harishreddym/baqend

Version:

Baqend JavaScript SDK

329 lines (285 loc) 8.96 kB
'use strict'; const ManagedType = require('./ManagedType'); const EntityType = require('./EntityType'); const Enhancer = require('../binding/Enhancer'); const ModelBuilder = require('./ModelBuilder'); const DbIndex = require('./DbIndex'); const Lockable = require('../util/Lockable'); const deprecated = require('../util/deprecated'); const StatusCode = require('../connector/Message').StatusCode; const message = require('../message'); /** * @alias metamodel.Metamodel * @extends util.Lockable */ class Metamodel extends Lockable { /** * Constructs a new metamodel instance which represents the complete schema of one baqend app * @param {EntityManagerFactory=} entityManagerFactory */ constructor(entityManagerFactory) { super(); /** * Defines if the Metamodel has been finalized * @type boolean */ this.isInitialized = false; /** * @type EntityManagerFactory */ this.entityManagerFactory = entityManagerFactory; /** * @type Object<string,metamodel.EntityType> */ this.entities = null; /** * @type Object<string,metamodel.EmbeddableType> */ this.embeddables = null; /** * @type Object<string,metamodel.BasicType> */ this.baseTypes = null; /** * @type {binding.Enhancer} */ this.enhancer = new Enhancer(); } /** * Prepare the Metamodel for custom schema creation * @param {Object=} jsonMetamodel initialize the metamodel with the serialized json schema * @return {void} */ init(jsonMetamodel) { if (this.isInitialized) { throw new Error('Metamodel is already initialized.'); } this.fromJSON(jsonMetamodel || []); this.isInitialized = true; } /** * @param {(Class<binding.Managed>|string)} arg * @return {string} */ getRef(arg) { let ref; if (Object(arg) instanceof String) { ref = arg; if (ref.indexOf('/db/') !== 0) { ref = '/db/' + arg; } } else { ref = Enhancer.getIdentifier(arg); } return ref; } /** * Return the metamodel entity type representing the entity. * * @param {(Class<binding.Entity>|string)} typeConstructor - the type of the represented entity * @return {metamodel.EntityType} the metamodel entity type */ entity(typeConstructor) { const ref = this.getRef(typeConstructor); return ref ? this.entities[ref] : null; } /** * Return the metamodel basic type representing the native class. * @param {(Class<*>|string)} typeConstructor - the type of the represented native class * @return {metamodel.BasicType} the metamodel basic type */ baseType(typeConstructor) { let ref = null; if (Object(typeConstructor) instanceof String) { ref = this.getRef(typeConstructor); } else { const baseTypesNames = Object.keys(this.baseTypes); for (let i = 0, len = baseTypesNames.length; i < len; i += 1) { const name = baseTypesNames[i]; const type = this.baseTypes[name]; if (!type.noResolving && type.typeConstructor === typeConstructor) { ref = name; break; } } } return ref ? this.baseTypes[ref] : null; } /** * Return the metamodel embeddable type representing the embeddable class. * @param {Class<binding.Managed>|string} typeConstructor - the type of the represented embeddable class * @return {metamodel.EmbeddableType} the metamodel embeddable type */ embeddable(typeConstructor) { const ref = this.getRef(typeConstructor); return ref ? this.embeddables[ref] : null; } /** * Return the metamodel managed type representing the entity, mapped superclass, or embeddable class. * * @param {(Class<binding.Managed>|string)} typeConstructor - the type of the represented managed class * @return {metamodel.Type} the metamodel managed type */ managedType(typeConstructor) { return this.baseType(typeConstructor) || this.entity(typeConstructor) || this.embeddable(typeConstructor); } /** * @param {metamodel.Type} type * @return {metamodel.Type} the added type */ addType(type) { let types; if (type.isBasic) { types = this.baseTypes; } else if (type.isEmbeddable) { type.init(this.enhancer); types = this.embeddables; } else if (type.isEntity) { type.init(this.enhancer); types = this.entities; if (type.superType === null && type.ref !== EntityType.Object.ref) { type.superType = this.entity(EntityType.Object.ref); } } if (types[type.ref]) { throw new Error('The type ' + type.ref + ' is already declared.'); } types[type.ref] = type; return type; } /** * Load all schema data from the server * @return {Promise<metamodel.Metamodel>} */ load() { if (!this.isInitialized) { return this.withLock(() => { const msg = new message.GetAllSchemas(); return this.entityManagerFactory.send(msg).then((response) => { this.init(response.entity); return this; }); }); } throw new Error('Metamodel is already initialized.'); } /** * Store all local schema data on the server, or the provided one * * Note: The schema must be initialized, by init or load * * @param {metamodel.ManagedType=} managedType The specific type to persist, if omitted the complete schema * will be updated * @return {Promise<metamodel.Metamodel>} */ save(managedType) { return this.sendUpdate(managedType || this.toJSON()).then(() => this); } /** * Update the metamodel with the schema * * The provided data object will be forwarded to the UpdateAllSchemas resource. * The underlying schema of this Metamodel object will be replaced by the result. * * @param {json} data The JSON which will be send to the UpdateAllSchemas resource. * @return {Promise<metamodel.Metamodel>} */ update(data) { return this.sendUpdate(data).then((response) => { this.fromJSON(response.entity); return this; }); } sendUpdate(data) { return this.withLock(() => { let msg; if (data instanceof ManagedType) { msg = new message.UpdateSchema(data.name, data.toJSON()); } else { msg = new message.UpdateAllSchemas(data); } return this.entityManagerFactory.send(msg); }); } /** * Get the current schema types as json * @return {json} the json data */ toJSON() { if (!this.isInitialized) { throw new Error('Metamodel is not initialized.'); } return [].concat( Object.keys(this.entities).map(ref => this.entities[ref].toJSON()), Object.keys(this.embeddables).map(ref => this.embeddables[ref].toJSON()) ); } /** * Replace the current schema by the provided one in json * @param {json} json The json schema data * @return {void} */ fromJSON(json) { const builder = new ModelBuilder(); const models = builder.buildModels(json); this.baseTypes = {}; this.embeddables = {}; this.entities = {}; Object.keys(models).forEach(ref => this.addType(models[ref])); } /** * Creates an index * * @param {string} bucket Name of the Bucket * @param {metamodel.DbIndex} index Will be applied for the given bucket * @return {Promise<*>} */ createIndex(bucket, index) { index.drop = false; const msg = new message.CreateDropIndex(bucket, index.toJSON()); return this.entityManagerFactory.send(msg); } /** * Drops an index * * @param {string} bucket Name of the Bucket * @param {metamodel.DbIndex} index Will be dropped for the given bucket * @return {Promise<*>} */ dropIndex(bucket, index) { index.drop = true; const msg = new message.CreateDropIndex(bucket, index.toJSON()); return this.entityManagerFactory.send(msg); } /** * Drops all indexes * * @param {string} bucket Indexes will be dropped for the given bucket * @return {Promise<*>} */ dropAllIndexes(bucket) { const msg = new message.DropAllIndexes(bucket); return this.entityManagerFactory.send(msg); } /** * Loads all indexes for the given bucket * * @param {string} bucket Current indexes will be loaded for the given bucket * @return {Promise<Array<metamodel.DbIndex>>} */ getIndexes(bucket) { const msg = new message.ListIndexes(bucket); return this.entityManagerFactory.send(msg) .then(response => response.entity.map(el => new DbIndex(el.keys, el.unique))) .catch((e) => { if (e.status === StatusCode.BUCKET_NOT_FOUND || e.status === StatusCode.OBJECT_NOT_FOUND) { return null; } throw e; }); } } deprecated(Metamodel.prototype, '_enhancer', 'enhancer'); deprecated(Metamodel.prototype, '_send', 'sendUpdate'); deprecated(Metamodel.prototype, '_getRef', 'getRef'); module.exports = Metamodel;