From c5d411489f5923ca4fb4d0a6f7a88470363bff48 Mon Sep 17 00:00:00 2001 From: Kevin Delisle Date: Wed, 20 Dec 2017 10:44:36 -0500 Subject: [PATCH] feat(repository): helper function for getting Model metadata --- .../repository/src/decorators/metadata.ts | 39 +++++++++++++ packages/repository/src/index.ts | 1 + .../test/unit/decorator/metadata.ts | 55 +++++++++++++++++++ 3 files changed, 95 insertions(+) create mode 100644 packages/repository/src/decorators/metadata.ts create mode 100644 packages/repository/test/unit/decorator/metadata.ts diff --git a/packages/repository/src/decorators/metadata.ts b/packages/repository/src/decorators/metadata.ts new file mode 100644 index 000000000000..9b8634527bb2 --- /dev/null +++ b/packages/repository/src/decorators/metadata.ts @@ -0,0 +1,39 @@ +import {InspectionOptions, MetadataInspector} from '@loopback/context'; +import {MODEL_KEY, MODEL_PROPERTIES_KEY} from '../'; +import {ModelDefinition, PropertyDefinition} from '../../index'; + +export class ModelMetadataHelper { + /** + * A utility function to simplify retrieving metadata from a target model and + * its properties. + * @param target The class from which to retrieve metadata. + * @param options An options object for the MetadataInspector to customize + * the output of the metadata retrieval functions. + */ + static getModelMetadata(target: Function, options?: InspectionOptions) { + const classDef = MetadataInspector.getClassMetadata( + MODEL_KEY, + target, + options, + ); + const meta = new ModelDefinition( + Object.assign( + { + name: target.name, + properties: {}, + }, + classDef, + ), + ); + + meta.properties = Object.assign( + {}, + MetadataInspector.getAllPropertyMetadata( + MODEL_PROPERTIES_KEY, + target.prototype, + options, + ), + ); + return meta; + } +} diff --git a/packages/repository/src/index.ts b/packages/repository/src/index.ts index 3207690eac54..b1069dca6a7f 100644 --- a/packages/repository/src/index.ts +++ b/packages/repository/src/index.ts @@ -7,6 +7,7 @@ export * from './common-types'; export * from './decorators/model'; export * from './decorators/repository'; export * from './decorators/relation'; +export * from './decorators/metadata'; export * from './types'; export * from './model'; export * from './query'; diff --git a/packages/repository/test/unit/decorator/metadata.ts b/packages/repository/test/unit/decorator/metadata.ts new file mode 100644 index 000000000000..4fa3248b4a2c --- /dev/null +++ b/packages/repository/test/unit/decorator/metadata.ts @@ -0,0 +1,55 @@ +import {ModelMetadataHelper} from '../../../src'; +import {property, model, ModelDefinition} from '../../..'; +import {expect} from '@loopback/testlab'; + +describe('Repository', () => { + describe('getAllClassMetadata', () => { + @model() + class Colour { + @property({}) + rgb: string; + } + @model() + class Widget { + @property() id: number; + @property.array(Colour) colours: Colour[]; + } + + @model() + class Samoflange { + id: number; + name: string; + canRotate: boolean; + } + + it('retrieves metadata for classes with @model', () => { + const meta = ModelMetadataHelper.getModelMetadata(Samoflange); + expect(meta).to.deepEqual( + new ModelDefinition({ + name: 'Samoflange', + properties: {}, + settings: new Map(), + }), + ); + }); + + it('retrieves metadata for classes with @model and @property', () => { + const meta = ModelMetadataHelper.getModelMetadata(Widget); + expect(meta).to.deepEqual( + new ModelDefinition({ + properties: { + id: { + type: Number, + }, + colours: { + array: true, + type: Colour, + }, + }, + settings: new Map(), + name: 'Widget', + }), + ); + }); + }); +});