From da5cb2c11779e8836139ada92343ed0b59034e40 Mon Sep 17 00:00:00 2001 From: Ritchie Martori Date: Tue, 29 Oct 2013 14:12:23 -0700 Subject: [PATCH] Add app.boot() --- docs/api.md | 156 ++++++++++++++- lib/application.js | 221 +++++++++++++++++++++- package.json | 12 +- test/app.test.js | 91 ++++++++- test/fixtures/simple-app/app.json | 4 + test/fixtures/simple-app/datasources.json | 5 + test/fixtures/simple-app/models.json | 5 + test/support.js | 2 +- 8 files changed, 475 insertions(+), 21 deletions(-) create mode 100644 test/fixtures/simple-app/app.json create mode 100644 test/fixtures/simple-app/datasources.json create mode 100644 test/fixtures/simple-app/models.json diff --git a/docs/api.md b/docs/api.md index 3484aabb..4a101155 100644 --- a/docs/api.md +++ b/docs/api.md @@ -21,24 +21,160 @@ app.listen(3000); > - see [express docs](http://expressjs.com/api.html) for details > - supports [express / connect middleware](http://expressjs.com/api.html#middleware) -#### app.model(Model) +#### app.boot([options]) -Expose a `Model` to remote clients. +Initialize an application from an options object or a set of JSON and JavaScript files. + +**What happens during an app _boot_?** + +1. **DataSources** are created from an `options.dataSources` object or `datasources.json` in the current directory +2. **Models** are created from an `options.models` object or `models.json` in the current directory +3. Any JavaScript files in the `./models` and `./datasources` directories are required. + +**Options** + + - `cwd` - _optional_ - the directory to use when loading JSON and JavaScript files + - `models` - _optional_ - an object containing `Model` definitions + - `dataSources` - _optional_ - an object containing `DataSource` definitions + +> **NOTE:** mixing `app.boot()` and `app.model(name, config)` in multiple files may result +> in models being **undefined** due to race conditions. To avoid this when using `app.boot()` +> make sure all models are passed as part of the `models` definition. + + +**Model Definitions** + +The following is an example of an object containing two `Model` definitions: "location" and "inventory". ```js -// create a testing data source -var memory = loopback.memory(); -var Color = memory.createModel('color', {name: String}); +{ + "location": { + // a reference, by name, to a dataSource definition + "dataSource": "my-db", + // the options passed to Model.extend(name, properties, options) + "options": { + "relationships": { + "hasMany": { + "model": "Inventory", "foreignKey": "locationId", "as": "inventory" + } + }, + "remoteMethods": { + "nearby": { + "description": "Find nearby locations around the geo point", + "accepts": [ + {"arg": "here", "type": "GeoPoint", "required": true, "description": "geo location (lat & lng)"} + ], + "returns": {"arg": "locations", "root": true} + } + } + }, + // the properties passed to Model.extend(name, properties, options) + "properties": { + "id": {"id": true}, + "name": "String", + "zip": "Number", + "address": "String" + } + }, + "inventory": { + "dataSource": "my-db" + "options": { + "plural": "inventory" + }, + "properties": { + "id": { + "type": "String", + "required": true, + "id": true, + "length": 20 + }, + "available": { + "type": "Number", + "required": false + }, + "total": { + "type": "Number", + "required": false + } + } + } +} +``` -app.model(Color); -app.use(loopback.rest()); +**Model Definition Properties** + + - `dataSource` - **required** - a string containing the name of the data source definition to attach the `Model` to + - `options` - _optional_ - an object containing `Model` options + - `properties` _optional_ - an object defining the `Model` properties in [LoopBack Definition Language](http://docs.strongloop.com/loopback-datasource-juggler/#loopback-definition-language) + +**DataSource Definition Properties** + + - `connector` - **required** - the name of the [connector](#working-with-data-sources-and-connectors) + +#### app.model(name, definition) + +Define a `Model` and export it for use by remote clients. + +```js +// declare a DataSource +app.boot({ + dataSources: { + db: { + connector: 'mongodb', + url: 'mongodb://localhost:27015/my-database-name' + } + } +}); + +// describe a model +var modelDefinition = {dataSource: 'db'}; + +// create the model +var Product = app.model('product', modelDefinition); + +// use the model api +Product.create({name: 'pencil', price: 0.99}, console.log); ``` > **Note** - this will expose all [shared methods](#shared-methods) on the model. - + +You may also export an existing `Model` by calling `app.model(Model)` like the example below. + +#### app.models.MyModel + +All models are avaialbe from the `loopback.models` object. In the following +example the `Product` and `CustomerReceipt` models are accessed using +the `models` object. + +> **NOTE:** you must call `app.boot()` in order to build the app.models object. + +```js +var loopback = require('loopback'); +var app = loopback(); +app.boot({ + dataSources: { + db: {connector: 'memory'} + } +}); +app.model('product', {dataSource: 'db'}); +app.model('customer-receipt', {dataSource: 'db'}); + +// available based on the given name +var Product = app.models.Product; + +// also available as camelCase +var product = app.models.product; + +// multi-word models are avaiable as pascal cased +var CustomerReceipt = app.models.CustomerReceipt; + +// also available as camelCase +var customerReceipt = app.models.customerReceipt; +``` + #### app.models() -Get the app's exposed models. +Get the app's exported models. Only models defined using `app.model()` will show up in this list. ```js var models = app.models(); @@ -47,7 +183,7 @@ models.forEach(function (Model) { console.log(Model.modelName); // color }); ``` - + #### app.docs(options) Enable swagger REST api documentation. diff --git a/lib/application.js b/lib/application.js index d96d1d59..8ad838d5 100644 --- a/lib/application.js +++ b/lib/application.js @@ -5,8 +5,11 @@ var DataSource = require('loopback-datasource-juggler').DataSource , ModelBuilder = require('loopback-datasource-juggler').ModelBuilder , assert = require('assert') + , fs = require('fs') , RemoteObjects = require('strong-remoting') - , swagger = require('strong-remoting/ext/swagger'); + , swagger = require('strong-remoting/ext/swagger') + , stringUtils = require('underscore.string') + , path = require('path'); /** * Export the app prototype. @@ -52,12 +55,25 @@ app._models = []; * @param Model {Model} */ -app.model = function (Model) { - this.remotes().exports[Model.pluralModelName] = Model; - this._models.push(Model); - Model.shared = true; - Model.app = this; - Model.emit('attached', this); +app.model = function (Model, config) { + if(arguments.length === 1) { + assert(typeof Model === 'function', 'app.model(Model) => Model must be a function / constructor'); + this.remotes().exports[Model.pluralModelName] = Model; + this._models.push(Model); + Model.shared = true; + Model.app = this; + Model.emit('attached', this); + return; + } + var modelName = Model; + assert(typeof modelName === 'string', 'app.model(name, properties, options) => name must be a string'); + + Model = + this.models[modelName] = + this.models[classify(modelName)] = + this.models[camelize(modelName)] = modelFromConfig(modelName, config, this); + + return Model; } /** @@ -120,3 +136,194 @@ app.handler = function (type) { return handler; } +/** + * An object to store dataSource instances. + */ + +app.dataSources = app.datasources = {}; + +/** + * Initialize the app using JSON and JavaScript files. + * + * @throws {Error} If config is not valid + * @throws {Error} If boot fails + */ + +app.boot = function(options) { + var app = this; + options = options || {}; + var cwd = options.cwd = options.cwd || process.cwd(); + var ctx = {}; + var appConfig = options.app; + var modelConfig = options.models; + var dataSourceConfig = options.dataSources; + + if(!appConfig) { + appConfig = tryReadConfig(cwd, 'app') || {}; + } + if(!modelConfig) { + modelConfig = tryReadConfig(cwd, 'models') || {}; + } + if(!dataSourceConfig) { + dataSourceConfig = tryReadConfig(cwd, 'datasources') || {}; + } + + assertIsValidConfig('app', appConfig); + assertIsValidConfig('model', modelConfig); + assertIsValidConfig('data source', dataSourceConfig); + + if(appConfig.host !== undefined) { + assert(typeof appConfig.host === 'string', 'app.host must be a string'); + app.set('host', appConfig.host); + } + + if(appConfig.port !== undefined) { + var portType = typeof appConfig.port; + assert(portType === 'string' || portType === 'number', 'app.port must be a string or number'); + app.set('port', appConfig.port); + } + + // instantiate data sources + forEachKeyedObject(dataSourceConfig, function(key, obj) { + app.dataSources[key] = + app.dataSources[classify(key)] = + app.dataSources[camelize(key)] = dataSourcesFromConfig(obj); + }); + + // instantiate models + forEachKeyedObject(modelConfig, function(key, obj) { + app.model(key, obj); + }); + + // require directories + var requiredModels = requireDir(path.join(cwd, 'models')); + var requiredDataSources = requireDir(path.join(cwd, 'datasources')); +} + +function assertIsValidConfig(name, config) { + if(config) { + assert(typeof config === 'object', name + ' config must be a valid JSON object'); + } +} + +function forEachKeyedObject(obj, fn) { + if(typeof obj !== 'object') return; + + Object.keys(obj).forEach(function(key) { + fn(key, obj[key]); + }); +} + +function requireDirAs(type, dir) { + return requireDir(dir); +} + +function classify(str) { + return stringUtils.classify(str); +} + +function camelize(str) { + return stringUtils.camelize(str); +} + +function dataSourcesFromConfig(config) { + return require('./loopback').createDataSource(config); +} + +function modelFromConfig(name, config, app) { + var ModelCtor = require('./loopback').createModel(name, config.properties, config.options); + var dataSource = app.dataSources[config.dataSource]; + + assert(isDataSource(dataSource), name + ' is referencing a dataSource that does not exist: "'+ config.dataSource +'"'); + + ModelCtor.attachTo(dataSource); + return ModelCtor; +} + +function requireDir(dir, basenames) { + assert(dir, 'cannot require directory contents without directory name'); + + var requires = {}; + + if (arguments.length === 2) { + // if basenames argument is passed, explicitly include those files + basenames.forEach(function (basename) { + var filepath = Path.resolve(Path.join(dir, basename)); + requires[basename] = tryRequire(filepath); + }); + } else if (arguments.length === 1) { + // if basenames arguments isn't passed, require all javascript + // files (except for those prefixed with _) and all directories + + var files = tryReadDir(dir); + + // sort files in lowercase alpha for linux + files.sort(function (a,b) { + a = a.toLowerCase(); + b = b.toLowerCase(); + + if (a < b) { + return -1; + } else if (b < a) { + return 1; + } else { + return 0; + } + }); + + files.forEach(function (filename) { + // ignore index.js and files prefixed with underscore + if ((filename === 'index.js') || (filename[0] === '_')) { return; } + + var filepath = path.resolve(path.join(dir, filename)); + var ext = path.extname(filename); + var stats = fs.statSync(filepath); + + // only require files supported by require.extensions (.txt .md etc.) + if (stats.isFile() && !(ext in require.extensions)) { return; } + + var basename = path.basename(filename, ext); + + requires[basename] = tryRequire(filepath); + }); + + } + + return requires; +}; + +function tryRequire(modulePath) { + try { + return require.apply(this, arguments); + } catch(e) { + console.error('failed to require "%s"', modulePath); + throw e; + } +} + +function tryReadDir() { + try { + return fs.readdirSync.apply(fs, arguments); + } catch(e) { + return []; + } +} + +function isModelCtor(obj) { + return typeof obj === 'function' && obj.modelName && obj.name === 'ModelCtor'; +} + +function isDataSource(obj) { + return obj instanceof DataSource; +} + +function tryReadConfig(cwd, fileName) { + try { + return require(path.join(cwd, fileName + '.json')); + } catch(e) { + if(e.code !== "MODULE_NOT_FOUND") { + throw e; + } + } +} + diff --git a/package.json b/package.json index fdc331be..58d19211 100644 --- a/package.json +++ b/package.json @@ -1,7 +1,14 @@ { "name": "loopback", "description": "LoopBack: Open Mobile Platform for Node.js", - "keywords": [ "StrongLoop", "LoopBack", "Mobile", "Backend", "Platform", "mBaaS" ], + "keywords": [ + "StrongLoop", + "LoopBack", + "Mobile", + "Backend", + "Platform", + "mBaaS" + ], "version": "1.0.0", "scripts": { "test": "mocha -R spec", @@ -20,7 +27,8 @@ "passport-local": "~0.1.6", "nodemailer": "~0.4.4", "ejs": "~0.8.4", - "bcryptjs": "~0.7.10" + "bcryptjs": "~0.7.10", + "underscore.string": "~2.3.3" }, "devDependencies": { "blanket": "~1.1.5", diff --git a/test/app.test.js b/test/app.test.js index b499bfce..71932caf 100644 --- a/test/app.test.js +++ b/test/app.test.js @@ -2,6 +2,7 @@ describe('app', function() { describe('app.model(Model)', function() { it("Expose a `Model` to remote clients", function() { + var app = loopback(); var memory = loopback.createDataSource({connector: loopback.Memory}); var Color = memory.createModel('color', {name: String}); app.model(Color); @@ -9,6 +10,29 @@ describe('app', function() { }); }); + describe('app.model(name, properties, options)', function () { + it('Sugar for defining a fully built model', function () { + var app = loopback(); + app.boot({ + app: {port: 3000, host: '127.0.0.1'}, + dataSources: { + db: { + connector: 'memory' + } + } + }); + + app.model('foo', { + dataSource: 'db' + }); + + var Foo = app.models.foo; + var f = new Foo; + + assert(f instanceof loopback.Model); + }); + }) + describe('app.models()', function() { it("Get the app's exposed models", function() { var Color = loopback.createModel('color', {name: String}); @@ -18,4 +42,69 @@ describe('app', function() { assert.equal(models[0].modelName, 'color'); }); }); -}); \ No newline at end of file + + describe('app.boot([options])', function () { + beforeEach(function () { + var app = this.app = loopback(); + + app.boot({ + app: { + port: 3000, + host: '127.0.0.1' + }, + models: { + 'foo-bar-bat-baz': { + options: { + plural: 'foo-bar-bat-bazzies' + }, + dataSource: 'the-db' + } + }, + dataSources: { + 'the-db': { + connector: 'memory' + } + } + }); + }); + + it('Load configuration', function () { + assert.equal(this.app.get('port'), 3000); + assert.equal(this.app.get('host'), '127.0.0.1'); + }); + + it('Instantiate models', function () { + assert(app.models); + assert(app.models.FooBarBatBaz); + assert(app.models.fooBarBatBaz); + assertValidDataSource(app.models.FooBarBatBaz.dataSource); + assert.isFunc(app.models.FooBarBatBaz, 'find'); + assert.isFunc(app.models.FooBarBatBaz, 'create'); + }); + + it('Attach models to data sources', function () { + assert.equal(app.models.FooBarBatBaz.dataSource, app.dataSources.theDb); + }); + + it('Instantiate data sources', function () { + assert(app.dataSources); + assert(app.dataSources.theDb); + assertValidDataSource(app.dataSources.theDb); + assert(app.dataSources.TheDb); + }); + }); + + describe('app.boot() - config loading', function () { + it('Load config files', function () { + var app = loopback(); + + app.boot({cwd: require('path').join(__dirname, 'fixtures', 'simple-app')}); + + assert(app.models.foo); + assert(app.models.Foo); + assert(app.models.Foo.dataSource); + assert.isFunc(app.models.Foo, 'find'); + assert.isFunc(app.models.Foo, 'create'); + }); + }); +}); diff --git a/test/fixtures/simple-app/app.json b/test/fixtures/simple-app/app.json new file mode 100644 index 00000000..8358c75f --- /dev/null +++ b/test/fixtures/simple-app/app.json @@ -0,0 +1,4 @@ +{ + "port": 3000, + "host": "127.0.0.1" +} diff --git a/test/fixtures/simple-app/datasources.json b/test/fixtures/simple-app/datasources.json new file mode 100644 index 00000000..05a18b3e --- /dev/null +++ b/test/fixtures/simple-app/datasources.json @@ -0,0 +1,5 @@ +{ + "db": { + "connector": "memory" + } +} diff --git a/test/fixtures/simple-app/models.json b/test/fixtures/simple-app/models.json new file mode 100644 index 00000000..3a22f139 --- /dev/null +++ b/test/fixtures/simple-app/models.json @@ -0,0 +1,5 @@ +{ + "foo": { + "dataSource": "db" + } +} diff --git a/test/support.js b/test/support.js index cad06f7e..438e6148 100644 --- a/test/support.js +++ b/test/support.js @@ -28,4 +28,4 @@ assertValidDataSource = function (dataSource) { assert.isFunc = function (obj, name) { assert(obj, 'cannot assert function ' + name + ' on object that doesnt exist'); assert(typeof obj[name] === 'function', name + ' is not a function'); -} \ No newline at end of file +}