2014-01-06 23:52:08 +00:00
|
|
|
/*!
|
2013-05-01 19:11:43 +00:00
|
|
|
* Module dependencies.
|
|
|
|
*/
|
|
|
|
|
2013-07-30 21:26:49 +00:00
|
|
|
var DataSource = require('loopback-datasource-juggler').DataSource
|
2014-06-06 09:47:25 +00:00
|
|
|
, registry = require('./registry')
|
2013-05-23 16:53:42 +00:00
|
|
|
, assert = require('assert')
|
2013-10-29 21:12:23 +00:00
|
|
|
, fs = require('fs')
|
2014-05-25 14:27:45 +00:00
|
|
|
, extend = require('util')._extend
|
2014-02-18 19:20:23 +00:00
|
|
|
, _ = require('underscore')
|
2013-07-25 23:24:00 +00:00
|
|
|
, RemoteObjects = require('strong-remoting')
|
2013-10-29 21:12:23 +00:00
|
|
|
, stringUtils = require('underscore.string')
|
2014-09-04 01:37:39 +00:00
|
|
|
, async = require('async')
|
2013-10-29 21:12:23 +00:00
|
|
|
, path = require('path');
|
2013-05-01 19:11:43 +00:00
|
|
|
|
|
|
|
/**
|
2014-01-06 23:52:08 +00:00
|
|
|
* The `App` object represents a Loopback application.
|
|
|
|
*
|
|
|
|
* The App object extends [Express](http://expressjs.com/api.html#express) and
|
|
|
|
* supports
|
|
|
|
* [Express / Connect middleware](http://expressjs.com/api.html#middleware). See
|
|
|
|
* [Express documentation](http://expressjs.com/api.html) for details.
|
|
|
|
*
|
|
|
|
* ```js
|
|
|
|
* var loopback = require('loopback');
|
|
|
|
* var app = loopback();
|
|
|
|
*
|
|
|
|
* app.get('/', function(req, res){
|
|
|
|
* res.send('hello world');
|
|
|
|
* });
|
|
|
|
*
|
|
|
|
* app.listen(3000);
|
|
|
|
* ```
|
|
|
|
*
|
|
|
|
* @class LoopBackApplication
|
2014-01-13 16:02:32 +00:00
|
|
|
* @header var app = loopback()
|
2014-01-06 23:52:08 +00:00
|
|
|
*/
|
|
|
|
function App() {
|
2014-01-13 16:02:32 +00:00
|
|
|
// this is a dummy placeholder for jsdox
|
2014-01-06 23:52:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*!
|
2013-05-01 19:11:43 +00:00
|
|
|
* Export the app prototype.
|
|
|
|
*/
|
|
|
|
|
|
|
|
var app = exports = module.exports = {};
|
|
|
|
|
|
|
|
/**
|
2014-01-06 23:52:08 +00:00
|
|
|
* Lazily load a set of [remote objects](http://apidocs.strongloop.com/strong-remoting/#remoteobjectsoptions).
|
|
|
|
*
|
2014-04-02 22:15:21 +00:00
|
|
|
* **NOTE:** Calling `app.remotes()` more than once returns only a single set of remote objects.
|
2014-01-06 23:52:08 +00:00
|
|
|
* @returns {RemoteObjects}
|
2013-05-23 16:53:42 +00:00
|
|
|
*/
|
2013-06-17 15:01:22 +00:00
|
|
|
|
2013-05-23 16:53:42 +00:00
|
|
|
app.remotes = function () {
|
|
|
|
if(this._remotes) {
|
|
|
|
return this._remotes;
|
|
|
|
} else {
|
2014-04-14 19:25:41 +00:00
|
|
|
var options = {};
|
|
|
|
|
|
|
|
if(this.get) {
|
|
|
|
options = this.get('remoting');
|
|
|
|
}
|
|
|
|
|
2014-02-19 23:14:31 +00:00
|
|
|
return (this._remotes = RemoteObjects.create(options));
|
2013-05-23 16:53:42 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-01-06 23:52:08 +00:00
|
|
|
/*!
|
2013-05-23 16:53:42 +00:00
|
|
|
* Remove a route by reference.
|
2013-05-01 19:11:43 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
app.disuse = function (route) {
|
|
|
|
if(this.stack) {
|
|
|
|
for (var i = 0; i < this.stack.length; i++) {
|
|
|
|
if(this.stack[i].route === route) {
|
|
|
|
this.stack.splice(i, 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-05-24 14:59:23 +00:00
|
|
|
/**
|
Add createModelFromConfig and configureModel()
Add new API allowing developers to split the model definition and
configuration into two steps:
1. Build models from JSON config, export them for re-use:
```js
var Customer = loopback.createModelFromConfig({
name: 'Customer',
base: 'User',
properties: {
address: 'string'
}
});
```
2. Attach existing models to a dataSource and a loopback app,
modify certain model aspects like relations:
```js
loopback.configureModel(Customer, {
dataSource: db,
relations: { /* ... */ }
});
```
Rework `app.model` to use `loopback.configureModel` under the hood.
Here is the new usage:
```js
var Customer = require('./models').Customer;
app.model(Customer, {
dataSource: 'db',
relations: { /* ... */ }
});
```
In order to preserve backwards compatibility with loopback 1.x,
`app.model(name, config)` calls both `createModelFromConfig`
and `configureModel`.
2014-06-05 15:41:12 +00:00
|
|
|
* Attach a model to the app. The `Model` will be available on the
|
2014-01-06 23:52:08 +00:00
|
|
|
* `app.models` object.
|
2013-05-01 19:11:43 +00:00
|
|
|
*
|
2014-01-06 23:52:08 +00:00
|
|
|
* ```js
|
Add createModelFromConfig and configureModel()
Add new API allowing developers to split the model definition and
configuration into two steps:
1. Build models from JSON config, export them for re-use:
```js
var Customer = loopback.createModelFromConfig({
name: 'Customer',
base: 'User',
properties: {
address: 'string'
}
});
```
2. Attach existing models to a dataSource and a loopback app,
modify certain model aspects like relations:
```js
loopback.configureModel(Customer, {
dataSource: db,
relations: { /* ... */ }
});
```
Rework `app.model` to use `loopback.configureModel` under the hood.
Here is the new usage:
```js
var Customer = require('./models').Customer;
app.model(Customer, {
dataSource: 'db',
relations: { /* ... */ }
});
```
In order to preserve backwards compatibility with loopback 1.x,
`app.model(name, config)` calls both `createModelFromConfig`
and `configureModel`.
2014-06-05 15:41:12 +00:00
|
|
|
* // Attach an existing model
|
|
|
|
* var User = loopback.User;
|
|
|
|
* app.model(User);
|
|
|
|
*
|
|
|
|
* // Attach an existing model, alter some aspects of the model
|
|
|
|
* var User = loopback.User;
|
|
|
|
* app.model(User, { dataSource: 'db' });
|
|
|
|
*
|
|
|
|
* // LoopBack 1.x way: create and attach a new model (deprecated)
|
|
|
|
* var Widget = app.model('Widget', {
|
|
|
|
* dataSource: 'db',
|
|
|
|
* properties: {
|
|
|
|
* name: 'string'
|
|
|
|
* }
|
2014-01-06 23:52:08 +00:00
|
|
|
* });
|
|
|
|
* ```
|
|
|
|
*
|
Add createModelFromConfig and configureModel()
Add new API allowing developers to split the model definition and
configuration into two steps:
1. Build models from JSON config, export them for re-use:
```js
var Customer = loopback.createModelFromConfig({
name: 'Customer',
base: 'User',
properties: {
address: 'string'
}
});
```
2. Attach existing models to a dataSource and a loopback app,
modify certain model aspects like relations:
```js
loopback.configureModel(Customer, {
dataSource: db,
relations: { /* ... */ }
});
```
Rework `app.model` to use `loopback.configureModel` under the hood.
Here is the new usage:
```js
var Customer = require('./models').Customer;
app.model(Customer, {
dataSource: 'db',
relations: { /* ... */ }
});
```
In order to preserve backwards compatibility with loopback 1.x,
`app.model(name, config)` calls both `createModelFromConfig`
and `configureModel`.
2014-06-05 15:41:12 +00:00
|
|
|
* @param {Object|String} Model The model to attach.
|
2014-04-02 22:15:21 +00:00
|
|
|
* @options {Object} config The model's configuration.
|
Add createModelFromConfig and configureModel()
Add new API allowing developers to split the model definition and
configuration into two steps:
1. Build models from JSON config, export them for re-use:
```js
var Customer = loopback.createModelFromConfig({
name: 'Customer',
base: 'User',
properties: {
address: 'string'
}
});
```
2. Attach existing models to a dataSource and a loopback app,
modify certain model aspects like relations:
```js
loopback.configureModel(Customer, {
dataSource: db,
relations: { /* ... */ }
});
```
Rework `app.model` to use `loopback.configureModel` under the hood.
Here is the new usage:
```js
var Customer = require('./models').Customer;
app.model(Customer, {
dataSource: 'db',
relations: { /* ... */ }
});
```
In order to preserve backwards compatibility with loopback 1.x,
`app.model(name, config)` calls both `createModelFromConfig`
and `configureModel`.
2014-06-05 15:41:12 +00:00
|
|
|
* @property {String|DataSource} dataSource The `DataSource` to which to
|
|
|
|
* attach the model.
|
|
|
|
* @property {Boolean} [public] whether the model should be exposed via REST API
|
|
|
|
* @property {Object} [relations] relations to add/update
|
2014-01-06 23:52:08 +00:00
|
|
|
* @end
|
|
|
|
* @returns {ModelConstructor} the model class
|
2013-05-01 19:11:43 +00:00
|
|
|
*/
|
2013-05-24 14:59:23 +00:00
|
|
|
|
2013-10-29 21:12:23 +00:00
|
|
|
app.model = function (Model, config) {
|
2014-06-10 06:53:01 +00:00
|
|
|
var isPublic = true;
|
|
|
|
if (arguments.length > 1) {
|
|
|
|
config = config || {};
|
Add createModelFromConfig and configureModel()
Add new API allowing developers to split the model definition and
configuration into two steps:
1. Build models from JSON config, export them for re-use:
```js
var Customer = loopback.createModelFromConfig({
name: 'Customer',
base: 'User',
properties: {
address: 'string'
}
});
```
2. Attach existing models to a dataSource and a loopback app,
modify certain model aspects like relations:
```js
loopback.configureModel(Customer, {
dataSource: db,
relations: { /* ... */ }
});
```
Rework `app.model` to use `loopback.configureModel` under the hood.
Here is the new usage:
```js
var Customer = require('./models').Customer;
app.model(Customer, {
dataSource: 'db',
relations: { /* ... */ }
});
```
In order to preserve backwards compatibility,
`app.model(name, config)` calls both `createModelFromConfig`
and `configureModel`.
2014-06-05 15:41:12 +00:00
|
|
|
if (typeof Model === 'string') {
|
|
|
|
// create & attach the model - backwards compatibility
|
|
|
|
|
|
|
|
// create config for loopback.modelFromConfig
|
|
|
|
var modelConfig = extend({}, config);
|
|
|
|
modelConfig.options = extend({}, config.options);
|
|
|
|
modelConfig.name = Model;
|
|
|
|
|
|
|
|
// modeller does not understand `dataSource` option
|
|
|
|
delete modelConfig.dataSource;
|
|
|
|
|
2014-06-09 09:18:52 +00:00
|
|
|
Model = registry.createModel(modelConfig);
|
Add createModelFromConfig and configureModel()
Add new API allowing developers to split the model definition and
configuration into two steps:
1. Build models from JSON config, export them for re-use:
```js
var Customer = loopback.createModelFromConfig({
name: 'Customer',
base: 'User',
properties: {
address: 'string'
}
});
```
2. Attach existing models to a dataSource and a loopback app,
modify certain model aspects like relations:
```js
loopback.configureModel(Customer, {
dataSource: db,
relations: { /* ... */ }
});
```
Rework `app.model` to use `loopback.configureModel` under the hood.
Here is the new usage:
```js
var Customer = require('./models').Customer;
app.model(Customer, {
dataSource: 'db',
relations: { /* ... */ }
});
```
In order to preserve backwards compatibility,
`app.model(name, config)` calls both `createModelFromConfig`
and `configureModel`.
2014-06-05 15:41:12 +00:00
|
|
|
|
|
|
|
// delete config options already applied
|
|
|
|
['relations', 'base', 'acls', 'hidden'].forEach(function(prop) {
|
|
|
|
delete config[prop];
|
|
|
|
if (config.options) delete config.options[prop];
|
|
|
|
});
|
|
|
|
delete config.properties;
|
2014-05-19 22:56:26 +00:00
|
|
|
}
|
Add createModelFromConfig and configureModel()
Add new API allowing developers to split the model definition and
configuration into two steps:
1. Build models from JSON config, export them for re-use:
```js
var Customer = loopback.createModelFromConfig({
name: 'Customer',
base: 'User',
properties: {
address: 'string'
}
});
```
2. Attach existing models to a dataSource and a loopback app,
modify certain model aspects like relations:
```js
loopback.configureModel(Customer, {
dataSource: db,
relations: { /* ... */ }
});
```
Rework `app.model` to use `loopback.configureModel` under the hood.
Here is the new usage:
```js
var Customer = require('./models').Customer;
app.model(Customer, {
dataSource: 'db',
relations: { /* ... */ }
});
```
In order to preserve backwards compatibility with loopback 1.x,
`app.model(name, config)` calls both `createModelFromConfig`
and `configureModel`.
2014-06-05 15:41:12 +00:00
|
|
|
|
Add createModelFromConfig and configureModel()
Add new API allowing developers to split the model definition and
configuration into two steps:
1. Build models from JSON config, export them for re-use:
```js
var Customer = loopback.createModelFromConfig({
name: 'Customer',
base: 'User',
properties: {
address: 'string'
}
});
```
2. Attach existing models to a dataSource and a loopback app,
modify certain model aspects like relations:
```js
loopback.configureModel(Customer, {
dataSource: db,
relations: { /* ... */ }
});
```
Rework `app.model` to use `loopback.configureModel` under the hood.
Here is the new usage:
```js
var Customer = require('./models').Customer;
app.model(Customer, {
dataSource: 'db',
relations: { /* ... */ }
});
```
In order to preserve backwards compatibility,
`app.model(name, config)` calls both `createModelFromConfig`
and `configureModel`.
2014-06-05 15:41:12 +00:00
|
|
|
configureModel(Model, config, this);
|
2014-06-10 06:53:01 +00:00
|
|
|
isPublic = config.public !== false;
|
|
|
|
} else {
|
2014-06-16 07:52:11 +00:00
|
|
|
assert(Model.prototype instanceof registry.Model,
|
2014-07-16 18:04:23 +00:00
|
|
|
Model.modelName + ' must be a descendant of loopback.Model');
|
Add createModelFromConfig and configureModel()
Add new API allowing developers to split the model definition and
configuration into two steps:
1. Build models from JSON config, export them for re-use:
```js
var Customer = loopback.createModelFromConfig({
name: 'Customer',
base: 'User',
properties: {
address: 'string'
}
});
```
2. Attach existing models to a dataSource and a loopback app,
modify certain model aspects like relations:
```js
loopback.configureModel(Customer, {
dataSource: db,
relations: { /* ... */ }
});
```
Rework `app.model` to use `loopback.configureModel` under the hood.
Here is the new usage:
```js
var Customer = require('./models').Customer;
app.model(Customer, {
dataSource: 'db',
relations: { /* ... */ }
});
```
In order to preserve backwards compatibility with loopback 1.x,
`app.model(name, config)` calls both `createModelFromConfig`
and `configureModel`.
2014-06-05 15:41:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
var modelName = Model.modelName;
|
2013-10-29 21:12:23 +00:00
|
|
|
this.models[modelName] =
|
2014-06-10 06:53:01 +00:00
|
|
|
this.models[classify(modelName)] =
|
|
|
|
this.models[camelize(modelName)] = Model;
|
2013-10-29 21:12:23 +00:00
|
|
|
|
2014-06-10 06:53:01 +00:00
|
|
|
this.models().push(Model);
|
|
|
|
|
2014-06-13 08:09:25 +00:00
|
|
|
if (isPublic && Model.sharedClass) {
|
|
|
|
this.remotes().addClass(Model.sharedClass);
|
2014-07-25 00:00:27 +00:00
|
|
|
if (Model.settings.trackChanges && Model.Change) {
|
2014-07-15 07:31:16 +00:00
|
|
|
this.remotes().addClass(Model.Change.sharedClass);
|
2014-07-25 00:00:27 +00:00
|
|
|
}
|
2014-06-10 06:53:01 +00:00
|
|
|
clearHandlerCache(this);
|
2014-07-25 00:00:27 +00:00
|
|
|
this.emit('modelRemoted', Model.sharedClass);
|
2013-11-18 20:52:00 +00:00
|
|
|
}
|
2013-11-01 19:53:30 +00:00
|
|
|
|
2014-06-13 08:09:25 +00:00
|
|
|
Model.shared = isPublic;
|
2014-06-10 06:53:01 +00:00
|
|
|
Model.app = this;
|
|
|
|
Model.emit('attached', this);
|
2013-10-29 21:12:23 +00:00
|
|
|
return Model;
|
Add createModelFromConfig and configureModel()
Add new API allowing developers to split the model definition and
configuration into two steps:
1. Build models from JSON config, export them for re-use:
```js
var Customer = loopback.createModelFromConfig({
name: 'Customer',
base: 'User',
properties: {
address: 'string'
}
});
```
2. Attach existing models to a dataSource and a loopback app,
modify certain model aspects like relations:
```js
loopback.configureModel(Customer, {
dataSource: db,
relations: { /* ... */ }
});
```
Rework `app.model` to use `loopback.configureModel` under the hood.
Here is the new usage:
```js
var Customer = require('./models').Customer;
app.model(Customer, {
dataSource: 'db',
relations: { /* ... */ }
});
```
In order to preserve backwards compatibility with loopback 1.x,
`app.model(name, config)` calls both `createModelFromConfig`
and `configureModel`.
2014-06-05 15:41:12 +00:00
|
|
|
};
|
|
|
|
|
2013-05-24 14:59:23 +00:00
|
|
|
/**
|
2014-04-02 22:15:21 +00:00
|
|
|
* Get the models exported by the app. Returns only models defined using `app.model()`
|
2014-01-13 16:02:32 +00:00
|
|
|
*
|
2014-04-02 22:15:21 +00:00
|
|
|
* There are two ways to access models:
|
2014-01-13 16:02:32 +00:00
|
|
|
*
|
2014-04-02 22:15:21 +00:00
|
|
|
* 1. Call `app.models()` to get a list of all models.
|
2014-01-13 16:02:32 +00:00
|
|
|
*
|
|
|
|
* ```js
|
|
|
|
* var models = app.models();
|
|
|
|
*
|
|
|
|
* models.forEach(function (Model) {
|
|
|
|
* console.log(Model.modelName); // color
|
|
|
|
* });
|
|
|
|
* ```
|
|
|
|
*
|
2014-04-02 22:15:21 +00:00
|
|
|
* **2. Use `app.model` to access a model by name.
|
2014-01-13 16:02:32 +00:00
|
|
|
* `app.model` has properties for all defined models.
|
|
|
|
*
|
2014-04-02 22:15:21 +00:00
|
|
|
* The following example illustrates accessing the `Product` and `CustomerReceipt` models
|
|
|
|
* using the `models` object.
|
2014-01-13 16:02:32 +00:00
|
|
|
*
|
|
|
|
* ```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;
|
|
|
|
* ```
|
2014-01-06 23:52:08 +00:00
|
|
|
*
|
2014-04-02 22:15:21 +00:00
|
|
|
* @returns {Array} Array of model classes.
|
2013-05-24 14:59:23 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
app.models = function () {
|
2013-11-15 04:19:46 +00:00
|
|
|
return this._models || (this._models = []);
|
2013-05-01 19:11:43 +00:00
|
|
|
}
|
|
|
|
|
2013-10-31 17:06:43 +00:00
|
|
|
/**
|
|
|
|
* Define a DataSource.
|
2014-01-06 23:52:08 +00:00
|
|
|
*
|
|
|
|
* @param {String} name The data source name
|
2014-07-31 04:57:08 +00:00
|
|
|
* @param {Object} config The data source config
|
|
|
|
* @param {DataSource} The registered data source
|
2013-10-31 17:06:43 +00:00
|
|
|
*/
|
|
|
|
app.dataSource = function (name, config) {
|
2014-07-31 04:57:08 +00:00
|
|
|
var ds = dataSourcesFromConfig(config, this.connectors);
|
2013-10-31 17:06:43 +00:00
|
|
|
this.dataSources[name] =
|
|
|
|
this.dataSources[classify(name)] =
|
2014-07-31 04:57:08 +00:00
|
|
|
this.dataSources[camelize(name)] = ds;
|
|
|
|
return ds;
|
2013-10-31 17:06:43 +00:00
|
|
|
}
|
|
|
|
|
2014-05-28 13:02:55 +00:00
|
|
|
/**
|
|
|
|
* Register a connector.
|
|
|
|
*
|
|
|
|
* When a new data-source is being added via `app.dataSource`, the connector
|
|
|
|
* name is looked up in the registered connectors first.
|
|
|
|
*
|
|
|
|
* Connectors are required to be explicitly registered only for applications
|
|
|
|
* using browserify, because browserify does not support dynamic require,
|
|
|
|
* which is used by LoopBack to automatically load the connector module.
|
|
|
|
*
|
|
|
|
* @param {String} name Name of the connector, e.g. 'mysql'.
|
|
|
|
* @param {Object} connector Connector object as returned
|
|
|
|
* by `require('loopback-connector-{name}')`.
|
|
|
|
*/
|
|
|
|
app.connector = function(name, connector) {
|
|
|
|
this.connectors[name] =
|
|
|
|
this.connectors[classify(name)] =
|
|
|
|
this.connectors[camelize(name)] = connector;
|
|
|
|
};
|
|
|
|
|
2013-05-24 22:08:23 +00:00
|
|
|
/**
|
2013-07-17 21:30:38 +00:00
|
|
|
* Get all remote objects.
|
2014-04-02 22:15:21 +00:00
|
|
|
* @returns {Object} [Remote objects](http://apidocs.strongloop.com/strong-remoting/#remoteobjectsoptions).
|
2013-05-24 22:08:23 +00:00
|
|
|
*/
|
|
|
|
|
2013-07-17 21:30:38 +00:00
|
|
|
app.remoteObjects = function () {
|
|
|
|
var result = {};
|
2014-05-19 22:56:26 +00:00
|
|
|
|
|
|
|
this.remotes().classes().forEach(function(sharedClass) {
|
|
|
|
result[sharedClass.name] = sharedClass.ctor;
|
2013-07-17 21:30:38 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
return result;
|
2013-05-24 22:08:23 +00:00
|
|
|
}
|
2013-07-25 23:24:00 +00:00
|
|
|
|
|
|
|
/*!
|
|
|
|
* Get a handler of the specified type from the handler cache.
|
2014-08-05 10:14:39 +00:00
|
|
|
* @triggers `mounted` events on shared class constructors (models)
|
2013-07-25 23:24:00 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
app.handler = function (type) {
|
|
|
|
var handlers = this._handlers || (this._handlers = {});
|
|
|
|
if(handlers[type]) {
|
|
|
|
return handlers[type];
|
|
|
|
}
|
|
|
|
|
|
|
|
var remotes = this.remotes();
|
|
|
|
var handler = this._handlers[type] = remotes.handler(type);
|
2014-08-05 10:14:39 +00:00
|
|
|
|
|
|
|
remotes.classes().forEach(function(sharedClass) {
|
|
|
|
sharedClass.ctor.emit('mounted', app, sharedClass, remotes);
|
|
|
|
});
|
|
|
|
|
2013-07-25 23:24:00 +00:00
|
|
|
return handler;
|
|
|
|
}
|
|
|
|
|
2013-10-29 21:12:23 +00:00
|
|
|
/**
|
|
|
|
* An object to store dataSource instances.
|
|
|
|
*/
|
|
|
|
|
|
|
|
app.dataSources = app.datasources = {};
|
|
|
|
|
2013-11-15 04:19:46 +00:00
|
|
|
/**
|
|
|
|
* Enable app wide authentication.
|
|
|
|
*/
|
|
|
|
|
|
|
|
app.enableAuth = function() {
|
|
|
|
var remotes = this.remotes();
|
2014-06-06 01:53:30 +00:00
|
|
|
var app = this;
|
2013-11-15 04:19:46 +00:00
|
|
|
|
|
|
|
remotes.before('**', function(ctx, next, method) {
|
|
|
|
var req = ctx.req;
|
|
|
|
var Model = method.ctor;
|
|
|
|
var modelInstance = ctx.instance;
|
2013-12-11 05:49:18 +00:00
|
|
|
var modelId = modelInstance && modelInstance.id || req.param('id');
|
2013-11-15 04:19:46 +00:00
|
|
|
|
2014-06-06 01:53:30 +00:00
|
|
|
var modelSettings = Model.settings || {};
|
|
|
|
var errStatusCode = modelSettings.aclErrorStatus || app.get('aclErrorStatus') || 401;
|
|
|
|
if(!req.accessToken){
|
|
|
|
errStatusCode = 401;
|
|
|
|
}
|
|
|
|
|
2013-12-12 03:46:56 +00:00
|
|
|
if(Model.checkAccess) {
|
|
|
|
Model.checkAccess(
|
|
|
|
req.accessToken,
|
|
|
|
modelId,
|
2014-05-31 02:29:30 +00:00
|
|
|
method,
|
2014-08-08 05:19:27 +00:00
|
|
|
ctx,
|
2013-12-12 03:46:56 +00:00
|
|
|
function(err, allowed) {
|
|
|
|
if(err) {
|
|
|
|
console.log(err);
|
|
|
|
next(err);
|
|
|
|
} else if(allowed) {
|
|
|
|
next();
|
|
|
|
} else {
|
2014-06-06 01:53:30 +00:00
|
|
|
|
|
|
|
var messages = {
|
|
|
|
403:'Access Denied',
|
|
|
|
404: ('could not find a model with id ' + modelId),
|
|
|
|
401:'Authorization Required'
|
|
|
|
};
|
|
|
|
|
|
|
|
var e = new Error(messages[errStatusCode] || messages[403]);
|
|
|
|
e.statusCode = errStatusCode;
|
2013-12-12 03:46:56 +00:00
|
|
|
next(e);
|
|
|
|
}
|
2013-11-15 04:19:46 +00:00
|
|
|
}
|
2013-12-12 03:46:56 +00:00
|
|
|
);
|
|
|
|
} else {
|
|
|
|
next();
|
|
|
|
}
|
2013-11-15 04:19:46 +00:00
|
|
|
});
|
2014-02-05 17:46:22 +00:00
|
|
|
|
|
|
|
this.isAuthEnabled = true;
|
|
|
|
};
|
2013-11-15 04:19:46 +00:00
|
|
|
|
2013-10-29 21:12:23 +00:00
|
|
|
app.boot = function(options) {
|
2014-06-25 11:52:20 +00:00
|
|
|
throw new Error(
|
|
|
|
'`app.boot` was removed, use the new module loopback-boot instead');
|
2013-10-29 21:12:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
function classify(str) {
|
|
|
|
return stringUtils.classify(str);
|
|
|
|
}
|
|
|
|
|
|
|
|
function camelize(str) {
|
|
|
|
return stringUtils.camelize(str);
|
|
|
|
}
|
|
|
|
|
2014-05-28 13:02:55 +00:00
|
|
|
function dataSourcesFromConfig(config, connectorRegistry) {
|
2013-11-01 19:53:30 +00:00
|
|
|
var connectorPath;
|
|
|
|
|
|
|
|
assert(typeof config === 'object',
|
|
|
|
'cannont create data source without config object');
|
|
|
|
|
|
|
|
if(typeof config.connector === 'string') {
|
2014-05-28 13:02:55 +00:00
|
|
|
var name = config.connector;
|
|
|
|
if (connectorRegistry[name]) {
|
|
|
|
config.connector = connectorRegistry[name];
|
|
|
|
} else {
|
|
|
|
connectorPath = path.join(__dirname, 'connectors', name + '.js');
|
2013-11-01 19:53:30 +00:00
|
|
|
|
2014-05-28 13:02:55 +00:00
|
|
|
if (fs.existsSync(connectorPath)) {
|
|
|
|
config.connector = require(connectorPath);
|
|
|
|
}
|
2013-11-01 19:53:30 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-06-06 09:47:25 +00:00
|
|
|
return registry.createDataSource(config);
|
2013-10-29 21:12:23 +00:00
|
|
|
}
|
|
|
|
|
Add createModelFromConfig and configureModel()
Add new API allowing developers to split the model definition and
configuration into two steps:
1. Build models from JSON config, export them for re-use:
```js
var Customer = loopback.createModelFromConfig({
name: 'Customer',
base: 'User',
properties: {
address: 'string'
}
});
```
2. Attach existing models to a dataSource and a loopback app,
modify certain model aspects like relations:
```js
loopback.configureModel(Customer, {
dataSource: db,
relations: { /* ... */ }
});
```
Rework `app.model` to use `loopback.configureModel` under the hood.
Here is the new usage:
```js
var Customer = require('./models').Customer;
app.model(Customer, {
dataSource: 'db',
relations: { /* ... */ }
});
```
In order to preserve backwards compatibility with loopback 1.x,
`app.model(name, config)` calls both `createModelFromConfig`
and `configureModel`.
2014-06-05 15:41:12 +00:00
|
|
|
function configureModel(ModelCtor, config, app) {
|
2014-06-06 09:47:25 +00:00
|
|
|
assert(ModelCtor.prototype instanceof registry.Model,
|
2014-07-16 18:04:23 +00:00
|
|
|
ModelCtor.modelName + ' must be a descendant of loopback.Model');
|
2014-05-25 14:27:45 +00:00
|
|
|
|
2014-03-21 19:18:00 +00:00
|
|
|
var dataSource = config.dataSource;
|
|
|
|
|
2014-08-20 23:05:45 +00:00
|
|
|
if(dataSource) {
|
|
|
|
if(typeof dataSource === 'string') {
|
|
|
|
dataSource = app.dataSources[dataSource];
|
|
|
|
}
|
2013-10-29 21:12:23 +00:00
|
|
|
|
2014-08-20 23:05:45 +00:00
|
|
|
assert(dataSource instanceof DataSource,
|
|
|
|
ModelCtor.modelName + ' is referencing a dataSource that does not exist: "' +
|
|
|
|
config.dataSource +'"');
|
|
|
|
}
|
2014-05-25 14:27:45 +00:00
|
|
|
|
Add createModelFromConfig and configureModel()
Add new API allowing developers to split the model definition and
configuration into two steps:
1. Build models from JSON config, export them for re-use:
```js
var Customer = loopback.createModelFromConfig({
name: 'Customer',
base: 'User',
properties: {
address: 'string'
}
});
```
2. Attach existing models to a dataSource and a loopback app,
modify certain model aspects like relations:
```js
loopback.configureModel(Customer, {
dataSource: db,
relations: { /* ... */ }
});
```
Rework `app.model` to use `loopback.configureModel` under the hood.
Here is the new usage:
```js
var Customer = require('./models').Customer;
app.model(Customer, {
dataSource: 'db',
relations: { /* ... */ }
});
```
In order to preserve backwards compatibility with loopback 1.x,
`app.model(name, config)` calls both `createModelFromConfig`
and `configureModel`.
2014-06-05 15:41:12 +00:00
|
|
|
config = extend({}, config);
|
|
|
|
config.dataSource = dataSource;
|
2014-05-25 14:27:45 +00:00
|
|
|
|
2014-06-06 09:47:25 +00:00
|
|
|
registry.configureModel(ModelCtor, config);
|
2014-05-25 14:27:45 +00:00
|
|
|
}
|
|
|
|
|
2014-02-18 20:40:35 +00:00
|
|
|
function clearHandlerCache(app) {
|
|
|
|
app._handlers = undefined;
|
|
|
|
}
|
|
|
|
|
2014-01-08 14:20:17 +00:00
|
|
|
/**
|
|
|
|
* Listen for connections and update the configured port.
|
|
|
|
*
|
|
|
|
* When there are no parameters or there is only one callback parameter,
|
|
|
|
* the server will listen on `app.get('host')` and `app.get('port')`.
|
|
|
|
*
|
|
|
|
* ```js
|
|
|
|
* // listen on host/port configured in app config
|
|
|
|
* app.listen();
|
|
|
|
* ```
|
|
|
|
*
|
|
|
|
* Otherwise all arguments are forwarded to `http.Server.listen`.
|
|
|
|
*
|
|
|
|
* ```js
|
|
|
|
* // listen on the specified port and all hosts, ignore app config
|
|
|
|
* app.listen(80);
|
|
|
|
* ```
|
|
|
|
*
|
|
|
|
* The function also installs a `listening` callback that calls
|
|
|
|
* `app.set('port')` with the value returned by `server.address().port`.
|
|
|
|
* This way the port param contains always the real port number, even when
|
|
|
|
* listen was called with port number 0.
|
|
|
|
*
|
2014-04-02 22:15:21 +00:00
|
|
|
* @param {Function} cb If specified, the callback is added as a listener
|
2014-01-08 14:20:17 +00:00
|
|
|
* for the server's "listening" event.
|
|
|
|
* @returns {http.Server} A node `http.Server` with this application configured
|
|
|
|
* as the request handler.
|
|
|
|
*/
|
|
|
|
app.listen = function(cb) {
|
|
|
|
var self = this;
|
|
|
|
|
|
|
|
var server = require('http').createServer(this);
|
|
|
|
|
|
|
|
server.on('listening', function() {
|
|
|
|
self.set('port', this.address().port);
|
2014-07-01 12:27:02 +00:00
|
|
|
if (!self.get('url')) {
|
|
|
|
// A better default host would be `0.0.0.0`,
|
|
|
|
// but such URL is not supported by Windows
|
|
|
|
var host = self.get('host') || '127.0.0.1';
|
|
|
|
var url = 'http://' + host + ':' + self.get('port') + '/';
|
|
|
|
self.set('url', url);
|
|
|
|
}
|
2014-01-08 14:20:17 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
var useAppConfig =
|
|
|
|
arguments.length == 0 ||
|
|
|
|
(arguments.length == 1 && typeof arguments[0] == 'function');
|
|
|
|
|
|
|
|
if (useAppConfig) {
|
|
|
|
server.listen(this.get('port'), this.get('host'), cb);
|
|
|
|
} else {
|
|
|
|
server.listen.apply(server, arguments);
|
|
|
|
}
|
|
|
|
|
|
|
|
return server;
|
|
|
|
}
|
2014-09-04 01:37:39 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* **Do not call this method** if you are using `loopback-boot` to bootstrap your application.
|
|
|
|
* `loopback-boot` will call this method for you! Otherwise you must call `app.ready()` to run
|
|
|
|
* the `Model.ready()` hooks.
|
|
|
|
*
|
|
|
|
* Calling `ready()` will call `ready()` on all models attached to the `app`. Override
|
|
|
|
* the `ready()` method on a `Model` class to ensure you have access to a bootstrapped application.
|
|
|
|
*
|
|
|
|
* ```js
|
|
|
|
* module.exports = function(MyModel) {
|
|
|
|
* MyModel.setup = function() {
|
|
|
|
* // setup is called when a model is extended
|
|
|
|
* // you must call `base.setup()` to extend the base class properly
|
|
|
|
* this.base.setup();
|
|
|
|
*
|
|
|
|
* // add or remove remote methods and otherwise modify the `Model`
|
|
|
|
* this.remoteMethod('myMethod');
|
|
|
|
* }
|
|
|
|
*
|
|
|
|
* MyModel.beforeReady = function(app, cb) {
|
|
|
|
* // async setup, runs after all models are `setup()`
|
|
|
|
* // and before `ready()` is called
|
|
|
|
* setTimeout(cb, 100);
|
|
|
|
* }
|
|
|
|
*
|
|
|
|
* MyModel.ready = function(app) {
|
|
|
|
* // MyModel and other classes can be used
|
|
|
|
* // you should not modify any classes in this method
|
|
|
|
* console.log(this.sharedClass.methods());
|
|
|
|
* }
|
|
|
|
* }
|
|
|
|
* ```
|
|
|
|
*/
|
|
|
|
|
|
|
|
app.ready = function(cb) {
|
|
|
|
var app = this;
|
|
|
|
var models = app.models();
|
|
|
|
|
|
|
|
async.each(models, function(Model, cb) {
|
|
|
|
Model.beforeReady(app, cb);
|
|
|
|
}, function(err) {
|
|
|
|
if(err) return done(err);
|
|
|
|
|
|
|
|
models.forEach(function(Model) {
|
|
|
|
Model.ready(app);
|
|
|
|
});
|
|
|
|
|
|
|
|
done();
|
|
|
|
});
|
|
|
|
|
|
|
|
function done(err) {
|
|
|
|
if(typeof cb === 'function') return cb(err);
|
|
|
|
}
|
|
|
|
}
|