import { Collection, CollectionContext, CollectionOptions } from './collection'; import { default as lodash } from 'lodash'; import { Field } from '.'; export class InheritedCollection extends Collection { parents?: Collection[]; constructor(options: CollectionOptions, context: CollectionContext) { if (!options.inherits) { throw new Error('InheritedCollection must have inherits option'); } options.inherits = lodash.castArray(options.inherits); super(options, context); try { this.bindParents(); } catch (err) { if (err instanceof ParentCollectionNotFound) { const listener = (collection) => { if ( options.inherits.includes(collection.name) && lodash.every(options.inherits, (name) => this.context.database.collections.has(name)) ) { this.bindParents(); this.db.removeListener('afterDefineCollection', listener); } }; this.db.addListener('afterDefineCollection', listener); } else { throw err; } } } protected bindParents() { this.setParents(this.options.inherits); this.setParentFields(); this.setFields(this.options.fields, false); this.db.inheritanceMap.setInheritance(this.name, this.options.inherits); } protected setParents(inherits: string | string[]) { this.parents = lodash.castArray(inherits).map((name) => { const existCollection = this.db.collections.get(name); if (!existCollection) { throw new ParentCollectionNotFound(name); } return existCollection; }); } protected setParentFields() { for (const [name, fieldOptions] of this.parentFields()) { this.setField(name, { ...fieldOptions, inherit: true, }); } } getParents() { return this.parents; } parentFields() { const fields = new Map(); for (const parent of this.parents) { if (parent.isInherited()) { for (const [name, field] of (parent).parentFields()) { fields.set(name, field.options); } } const parentFields = parent.fields; for (const [name, field] of parentFields) { fields.set(name, field.options); } } return fields; } parentAttributes() { const attributes = {}; for (const parent of this.parents) { if (parent.isInherited()) { Object.assign(attributes, (parent).parentAttributes()); } const parentAttributes = (parent.model).tableAttributes; Object.assign(attributes, parentAttributes); } return attributes; } isInherited() { return true; } } class ParentCollectionNotFound extends Error { constructor(name: string) { super(`parent collection ${name} not found`); } }