diff --git a/dist/epicenter.js b/dist/epicenter.js index 9d738415..bb8363e6 100644 --- a/dist/epicenter.js +++ b/dist/epicenter.js @@ -584,6 +584,13 @@ AuthManager.prototype = $.extend(AuthManager.prototype, { /** * Helper method to check if you're currently logged in + * + * **Example** + * + * var amILoggedIn = authMgr.isLoggedIn(); + * + * **Parameters** + * @param {none} none * @return {Boolean} true if you're logged in */ isLoggedIn: function () { @@ -2010,8 +2017,8 @@ var Strategy = classFrom(IdentityStrategy, { var filter = injectFiltersFromSession(this.options.filter, userSession); var me = this; return runService.query(filter, { - // startrecord: 0, //TODO: Uncomment when EPICENTER-2569 is fixed - // endrecord: 0, + startrecord: 0, + endrecord: 0, sort: 'created', direction: 'desc' }).then(function (runs) { @@ -2103,8 +2110,8 @@ module.exports = classFrom(Base, { var filter = $.extend(true, { trashed: false }, sessionFilter, { model: runopts.model }); var me = this; return runService.query(filter, { - // startrecord: 0, //TODO: Uncomment when EPICENTER-2569 is fixed - // endrecord: 0, + startrecord: 0, + endrecord: 0, sort: 'created', direction: 'desc' }).then(function (runs) { @@ -2665,13 +2672,13 @@ module.exports = classFrom(Base, { var runopts = runService.getCurrentConfig(); var filter = injectFiltersFromSession({ saved: false, - trashed: false, //TODO: change to '!=true' once EPICENTER-2500 is fixed, + trashed: false, model: runopts.model, }, userSession); var me = this; var outputModifiers = { - // startrecord: 0, //TODO: Uncomment when EPICENTER-2569 is fixed - // endrecord: 0, + startrecord: 0, + endrecord: 0, sort: 'created', direction: 'desc' }; @@ -5158,6 +5165,10 @@ module.exports = function (config) { /** * Removes specified runid from memory * + * **Example** + * + * rs.removeFromMemory('bb589677-d476-4971-a68e-0c58d191e450'); + * * See [details on run persistence](../../../run_persistence/#runs-in-memory) * @param {String} [runID] id of run to remove * @param {Object} [filters] (Optional) Object containing filters and operation modifiers. Use key `include` to list model variables that you want to include in the response. Other available fields include: `startrecord`, `endrecord`, `sort`, and `direction` (`asc` or `desc`). @@ -7724,4 +7735,4 @@ module.exports = (function () { }()); },{"./query-util":53}]},{},[4]) -//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["node_modules/browser-pack/_prelude.js","node_modules/Base64/base64.js","node_modules/object-assign/index.js","src/api-version.json","src/app.js","src/env-load.js","src/managers/auth-manager.js","src/managers/channel-manager.js","src/managers/epicenter-channel-manager.js","src/managers/key-names.js","src/managers/run-manager.js","src/managers/run-strategies/conditional-creation-strategy.js","src/managers/run-strategies/deprecated/new-if-initialized-strategy.js","src/managers/run-strategies/deprecated/new-if-persisted-strategy.js","src/managers/run-strategies/index.js","src/managers/run-strategies/multiplayer-strategy.js","src/managers/run-strategies/none-strategy.js","src/managers/run-strategies/reuse-across-sessions.js","src/managers/run-strategies/reuse-last-initialized.js","src/managers/run-strategies/reuse-never.js","src/managers/run-strategies/reuse-per-session.js","src/managers/saved-runs-manager.js","src/managers/scenario-manager.js","src/managers/scenario-strategies/baseline-strategy.js","src/managers/scenario-strategies/reuse-last-unsaved.js","src/managers/special-operations.js","src/managers/strategy-utils.js","src/managers/world-manager.js","src/service/admin-file-service.js","src/service/asset-api-adapter.js","src/service/auth-api-service.js","src/service/channel-service.js","src/service/configuration-service.js","src/service/data-api-service.js","src/service/group-api-service.js","src/service/introspection-api-service.js","src/service/member-api-adapter.js","src/service/presence-api-service.js","src/service/run-api-service.js","src/service/service-utils.js","src/service/state-api-adapter.js","src/service/url-config-service.js","src/service/user-api-adapter.js","src/service/variables-api-service.js","src/service/world-api-adapter.js","src/store/cookie-store.js","src/store/session-manager.js","src/store/store-factory.js","src/transport/ajax-http-transport.js","src/transport/http-transport-factory.js","src/util/inherit.js","src/util/object-util.js","src/util/option-utils.js","src/util/query-util.js","src/util/run-util.js"],"names":[],"mappings":"AAAA;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7DA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnFA;AACA;AACA;AACA;;;ACHA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;ACvEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrZA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChQA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACraA;AACA;AACA;AACA;AACA;AACA;;ACLA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1OA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnHA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3EA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/DA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1JA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1OA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9DA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACRA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9MA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1XA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxHA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjPA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChSA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5LA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5NA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5lBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvJA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3JA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5vBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChJA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACVA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3GA;AACA;AACA;AACA;AACA;AACA;AACA;;ACNA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/GA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"generated.js","sourceRoot":"","sourcesContent":["(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require==\"function\"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error(\"Cannot find module '\"+o+\"'\");throw f.code=\"MODULE_NOT_FOUND\",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require==\"function\"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})",";(function () {\n\n  var object = typeof exports != 'undefined' ? exports : self; // #8: web workers\n  var chars = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=';\n\n  function InvalidCharacterError(message) {\n    this.message = message;\n  }\n  InvalidCharacterError.prototype = new Error;\n  InvalidCharacterError.prototype.name = 'InvalidCharacterError';\n\n  // encoder\n  // [https://gist.github.com/999166] by [https://github.com/nignag]\n  object.btoa || (\n  object.btoa = function (input) {\n    var str = String(input);\n    for (\n      // initialize result and counter\n      var block, charCode, idx = 0, map = chars, output = '';\n      // if the next str index does not exist:\n      //   change the mapping table to \"=\"\n      //   check if d has no fractional digits\n      str.charAt(idx | 0) || (map = '=', idx % 1);\n      // \"8 - idx % 1 * 8\" generates the sequence 2, 4, 6, 8\n      output += map.charAt(63 & block >> 8 - idx % 1 * 8)\n    ) {\n      charCode = str.charCodeAt(idx += 3/4);\n      if (charCode > 0xFF) {\n        throw new InvalidCharacterError(\"'btoa' failed: The string to be encoded contains characters outside of the Latin1 range.\");\n      }\n      block = block << 8 | charCode;\n    }\n    return output;\n  });\n\n  // decoder\n  // [https://gist.github.com/1020396] by [https://github.com/atk]\n  object.atob || (\n  object.atob = function (input) {\n    var str = String(input).replace(/=+$/, '');\n    if (str.length % 4 == 1) {\n      throw new InvalidCharacterError(\"'atob' failed: The string to be decoded is not correctly encoded.\");\n    }\n    for (\n      // initialize result and counters\n      var bc = 0, bs, buffer, idx = 0, output = '';\n      // get next character\n      buffer = str.charAt(idx++);\n      // character found in table? initialize bit storage and add its ascii value;\n      ~buffer && (bs = bc % 4 ? bs * 64 + buffer : buffer,\n        // and if not first of each 4 characters,\n        // convert the first 8 bits to one ascii character\n        bc++ % 4) ? output += String.fromCharCode(255 & bs >> (-2 * bc & 6)) : 0\n    ) {\n      // try to find character in table (0-63, not found => -1)\n      buffer = chars.indexOf(buffer);\n    }\n    return output;\n  });\n\n}());\n","'use strict';\n/* eslint-disable no-unused-vars */\nvar hasOwnProperty = Object.prototype.hasOwnProperty;\nvar propIsEnumerable = Object.prototype.propertyIsEnumerable;\n\nfunction toObject(val) {\n\tif (val === null || val === undefined) {\n\t\tthrow new TypeError('Object.assign cannot be called with null or undefined');\n\t}\n\n\treturn Object(val);\n}\n\nfunction shouldUseNative() {\n\ttry {\n\t\tif (!Object.assign) {\n\t\t\treturn false;\n\t\t}\n\n\t\t// Detect buggy property enumeration order in older V8 versions.\n\n\t\t// https://bugs.chromium.org/p/v8/issues/detail?id=4118\n\t\tvar test1 = new String('abc');  // eslint-disable-line\n\t\ttest1[5] = 'de';\n\t\tif (Object.getOwnPropertyNames(test1)[0] === '5') {\n\t\t\treturn false;\n\t\t}\n\n\t\t// https://bugs.chromium.org/p/v8/issues/detail?id=3056\n\t\tvar test2 = {};\n\t\tfor (var i = 0; i < 10; i++) {\n\t\t\ttest2['_' + String.fromCharCode(i)] = i;\n\t\t}\n\t\tvar order2 = Object.getOwnPropertyNames(test2).map(function (n) {\n\t\t\treturn test2[n];\n\t\t});\n\t\tif (order2.join('') !== '0123456789') {\n\t\t\treturn false;\n\t\t}\n\n\t\t// https://bugs.chromium.org/p/v8/issues/detail?id=3056\n\t\tvar test3 = {};\n\t\t'abcdefghijklmnopqrst'.split('').forEach(function (letter) {\n\t\t\ttest3[letter] = letter;\n\t\t});\n\t\tif (Object.keys(Object.assign({}, test3)).join('') !==\n\t\t\t\t'abcdefghijklmnopqrst') {\n\t\t\treturn false;\n\t\t}\n\n\t\treturn true;\n\t} catch (e) {\n\t\t// We don't expect any of the above to throw, but better to be safe.\n\t\treturn false;\n\t}\n}\n\nmodule.exports = shouldUseNative() ? Object.assign : function (target, source) {\n\tvar from;\n\tvar to = toObject(target);\n\tvar symbols;\n\n\tfor (var s = 1; s < arguments.length; s++) {\n\t\tfrom = Object(arguments[s]);\n\n\t\tfor (var key in from) {\n\t\t\tif (hasOwnProperty.call(from, key)) {\n\t\t\t\tto[key] = from[key];\n\t\t\t}\n\t\t}\n\n\t\tif (Object.getOwnPropertySymbols) {\n\t\t\tsymbols = Object.getOwnPropertySymbols(from);\n\t\t\tfor (var i = 0; i < symbols.length; i++) {\n\t\t\t\tif (propIsEnumerable.call(from, symbols[i])) {\n\t\t\t\t\tto[symbols[i]] = from[symbols[i]];\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\t}\n\n\treturn to;\n};\n","module.exports={\n    \"version\": \"v2\"\n}\n","/**\n * Epicenter Javascript libraries\n * v<%= version %>\n * https://github.com/forio/epicenter-js-libs\n */\n\nvar F = {\n    _private: {}, //need this hook now because tests expect everything to be global. Delete once tests are browserified\n    util: {},\n    factory: {},\n    transport: {},\n    store: {},\n    service: {},\n    manager: {\n        strategy: {}\n    },\n\n};\n\nF.load = require('./env-load');\n\nif (!global.SKIP_ENV_LOAD) {\n    F.load();\n}\n\nF.util.query = require('./util/query-util');\nF.util.run = require('./util/run-util');\nF.util.classFrom = require('./util/inherit');\nF._private.strategyutils = require('./managers/strategy-utils');\n\nF.factory.Transport = require('./transport/http-transport-factory');\nF.transport.Ajax = require('./transport/ajax-http-transport');\n\nF.service.URL = require('./service/url-config-service');\nF.service.Config = require('./service/configuration-service');\nF.service.Run = require('./service/run-api-service');\nF.service.File = require('./service/admin-file-service');\nF.service.Variables = require('./service/variables-api-service');\nF.service.Data = require('./service/data-api-service');\nF.service.Auth = require('./service/auth-api-service');\nF.service.World = require('./service/world-api-adapter');\nF.service.State = require('./service/state-api-adapter');\nF.service.User = require('./service/user-api-adapter');\nF.service.Member = require('./service/member-api-adapter');\nF.service.Asset = require('./service/asset-api-adapter');\nF.service.Group = require('./service/group-api-service');\nF.service.Introspect = require('./service/introspection-api-service');\nF.service.Presence = require('./service/presence-api-service');\n\nF.store.Cookie = require('./store/cookie-store');\nF.factory.Store = require('./store/store-factory');\n\nF.manager.ScenarioManager = require('./managers/scenario-manager');\nF.manager.RunManager = require('./managers/run-manager');\nF.manager.AuthManager = require('./managers/auth-manager');\nF.manager.WorldManager = require('./managers/world-manager');\nF.manager.SavedRunsManager = require('./managers/saved-runs-manager');\n\nvar strategies = require('./managers/run-strategies');\nF.manager.strategy = strategies.list; //TODO: this is not really a manager so namespace this better\n\nF.manager.ChannelManager = require('./managers/epicenter-channel-manager');\nF.service.Channel = require('./service/channel-service');\n\nF.version = '<%= version %>';\nF.api = require('./api-version.json');\n\nF.constants = require('./managers/key-names');\n\nglobal.F = F;\nmodule.exports = F;\n","'use strict';\n\nvar URLConfigService = require('./service/url-config-service');\n\nvar envLoad = function (callback) {\n    var urlService = new URLConfigService();\n    var infoUrl = urlService.getAPIPath('config');\n    var envPromise = $.ajax({ url: infoUrl, async: false });\n    envPromise = envPromise.then(function (res) {\n        var overrides = res.api;\n        URLConfigService.defaults = $.extend(URLConfigService.defaults, overrides);\n    });\n    return envPromise.then(callback).fail(callback);\n};\n\nmodule.exports = envLoad;\n","/**\n* ## Authorization Manager\n*\n* The Authorization Manager provides an easy way to manage user authentication (logging in and out) and authorization (keeping track of tokens, sessions, and groups) for projects.\n*\n* The Authorization Manager is most useful for [team projects](../../../glossary/#team) with an access level of [Authenticated](../../../glossary/#access). These projects are accessed by [end users](../../../glossary/#users) who are members of one or more [groups](../../../glossary/#groups).\n*\n* #### Using the Authorization Manager\n*\n* To use the Authorization Manager, instantiate it. Then, make calls to any of the methods you need:\n*\n*       var authMgr = new F.manager.AuthManager({\n*           account: 'acme-simulations',\n*           userName: 'enduser1',\n*           password: 'passw0rd'\n*       });\n*       authMgr.login().then(function () {\n*           authMgr.getCurrentUserSessionInfo();\n*       });\n*\n*\n* The `options` object passed to the `F.manager.AuthManager()` call can include:\n*\n*   * `account`: The account id for this `userName`. In the Epicenter UI, this is the **Team ID** (for team projects) or the **User ID** (for personal projects).\n*   * `userName`: Email or username to use for logging in.\n*   * `password`: Password for specified `userName`.\n*   * `project`: The **Project ID** for the project to log this user into. Optional.\n*   * `groupId`: Id of the group to which `userName` belongs. Required for end users if the `project` is specified.\n*\n* If you prefer starting from a template, the Epicenter JS Libs [Login Component](../../#components) uses the Authorization Manager as well. This sample HTML page (and associated CSS and JS files) provides a login form for team members and end users of your project. It also includes a group selector for end users that are members of multiple groups.\n*/\n\n'use strict';\nvar AuthAdapter = require('../service/auth-api-service');\nvar MemberAdapter = require('../service/member-api-adapter');\nvar GroupService = require('../service/group-api-service');\nvar SessionManager = require('../store/session-manager');\nvar _pick = require('../util/object-util')._pick;\nvar objectAssign = require('object-assign');\n\nvar atob = window.atob || require('Base64').atob;\n\nvar defaults = {\n    requiresGroup: true\n};\n\nfunction AuthManager(options) {\n    options = $.extend(true, {}, defaults, options);\n    this.sessionManager = new SessionManager(options);\n    this.options = this.sessionManager.getMergedOptions();\n\n    this.authAdapter = new AuthAdapter(this.options);\n}\n\nvar _findUserInGroup = function (members, id) {\n    for (var j = 0; j < members.length; j++) {\n        if (members[j].userId === id) {\n            return members[j];\n        }\n    }\n    return null;\n};\n\nAuthManager.prototype = $.extend(AuthManager.prototype, {\n\n    /**\n    * Logs user in.\n    *\n    * **Example**\n    *\n    *       authMgr.login({\n    *           account: 'acme-simulations',\n    *           project: 'supply-chain-game',\n    *           userName: 'enduser1',\n    *           password: 'passw0rd'\n    *       })\n    *           .then(function(statusObj) {\n    *               // if enduser1 belongs to exactly one group\n    *               // (or if the login() call is modified to include the group id)\n    *               // continue here\n    *           })\n    *           .fail(function(statusObj) {\n    *               // if enduser1 belongs to multiple groups,\n    *               // the login() call fails\n    *               // and returns all groups of which the user is a member\n    *               for (var i=0; i < statusObj.userGroups.length; i++) {\n    *                   console.log(statusObj.userGroups[i].name, statusObj.userGroups[i].groupId);\n    *               }\n    *           });\n    *\n    * **Parameters**\n    *\n    * @param {Object} options (Optional) Overrides for configuration options. If not passed in when creating an instance of the manager (`F.manager.AuthManager()`), these options should include:\n    * @param {string} options.account The account id for this `userName`. In the Epicenter UI, this is the **Team ID** (for team projects) or the **User ID** (for personal projects).\n    * @param {string} options.userName Email or username to use for logging in.\n    * @param {string} options.password Password for specified `userName`.\n    * @param {string} options.project (Optional) The **Project ID** for the project to log this user into.\n    * @param {string} options.groupId The id of the group to which `userName` belongs. Required for [end users](../../../glossary/#users) if the `project` is specified and if the end users are members of multiple [groups](../../../glossary/#groups), otherwise optional.\n    * @return {Promise}\n    */\n    login: function (options) {\n        var me = this;\n        var $d = $.Deferred();\n        var sessionManager = this.sessionManager;\n        var adapterOptions = sessionManager.getMergedOptions({ success: $.noop, error: $.noop }, options);\n        var outSuccess = adapterOptions.success;\n        var outError = adapterOptions.error;\n        var groupId = adapterOptions.groupId;\n\n        var decodeToken = function (token) {\n            var encoded = token.split('.')[1];\n            while (encoded.length % 4 !== 0) { //eslint-disable-line\n                encoded += '=';\n            }\n            return JSON.parse(atob(encoded));\n        };\n\n        var handleGroupError = function (message, statusCode, data, type) {\n            // logout the user since it's in an invalid state with no group selected\n            me.logout().then(function () {\n                var error = $.extend(true, {}, data, { statusText: message, status: statusCode, type: type });\n                $d.reject(error);\n            });\n        };\n\n        var handleSuccess = function (response) {\n            var token = response.access_token;\n            var userInfo = decodeToken(token);\n            var oldGroups = sessionManager.getSession(adapterOptions).groups || {};\n            var userGroupOpts = $.extend(true, {}, adapterOptions, { success: $.noop });\n            var data = { auth: response, user: userInfo };\n            var project = adapterOptions.project;\n            var isTeamMember = userInfo.parent_account_id === null;\n            var requiresGroup = adapterOptions.requiresGroup && project;\n\n            var userName = (userInfo.user_name || '').split('/')[0]; //of form <user>/<team>\n            var sessionInfo = {\n                auth_token: token,\n                account: adapterOptions.account,\n                project: project,\n                userId: userInfo.user_id,\n                groups: oldGroups,\n                isTeamMember: isTeamMember,\n                userName: userName,\n            };\n            // The group is not required if the user is not logging into a project\n            if (!requiresGroup) {\n                sessionManager.saveSession(sessionInfo);\n                outSuccess.apply(this, [data]);\n                $d.resolve(data);\n                return;\n            }\n\n            var handleGroupList = function (groupList) {\n                data.userGroups = groupList;\n\n                var group = null;\n                if (groupList.length === 0) {\n                    handleGroupError('The user has no groups associated in this account', 403, data, 'NO_GROUPS');\n                    return;\n                } else if (groupList.length === 1) {\n                    // Select the only group\n                    group = groupList[0];\n                } else if (groupList.length > 1) {\n                    if (groupId) {\n                        var filteredGroups = $.grep(groupList, function (resGroup) {\n                            return resGroup.groupId === groupId;\n                        });\n                        group = filteredGroups.length === 1 ? filteredGroups[0] : null;\n                    }\n                }\n\n                if (group) {\n                    // A team member does not get the group members because is calling the Group API\n                    // but it's automatically a fac user\n                    var isFac = isTeamMember ? true : _findUserInGroup(group.members, userInfo.user_id).role === 'facilitator';\n                    var groupData = {\n                        groupId: group.groupId,\n                        groupName: group.name,\n                        isFac: isFac\n                    };\n                    var sessionInfoWithGroup = objectAssign({}, sessionInfo, groupData);\n                    sessionInfo.groups[project] = groupData;\n                    me.sessionManager.saveSession(sessionInfoWithGroup, adapterOptions);\n                    outSuccess.apply(this, [data]);\n                    $d.resolve(data);\n                } else {\n                    handleGroupError('This user is associated with more than one group. Please specify a group id to log into and try again', 403, data, 'MULTIPLE_GROUPS');\n                }\n            };\n\n            if (!isTeamMember) {\n                me.getUserGroups({ userId: userInfo.user_id, token: token }, userGroupOpts)\n                    .then(handleGroupList, $d.reject);\n            } else {\n                var opts = objectAssign({}, userGroupOpts, { token: token });\n                var groupService = new GroupService(opts);\n                groupService.getGroups({ account: adapterOptions.account, project: project })\n                    .then(function (groups) {\n                        // Group API returns id instead of groupId\n                        groups.forEach(function (group) {\n                            group.groupId = group.id;\n                        });\n\n                        if (groups.length) {\n                            handleGroupList(groups);\n                        } else {\n                            //either it's a private project or there are no groups\n                            sessionManager.saveSession(sessionInfo);\n                            outSuccess.apply(this, [data]);\n                            $d.resolve(data);\n                            return;\n                        }\n                    }, $d.reject);\n            }\n        };\n\n        adapterOptions.success = handleSuccess;\n        adapterOptions.error = function (response) {\n            if (adapterOptions.account) {\n                // Try to login as a system user\n                adapterOptions.account = null;\n                adapterOptions.error = function () {\n                    outError.apply(this, arguments);\n                    $d.reject(response);\n                };\n\n                me.authAdapter.login(adapterOptions);\n                return;\n            }\n\n            outError.apply(this, arguments);\n            $d.reject(response);\n        };\n\n        this.authAdapter.login(adapterOptions);\n        return $d.promise();\n    },\n\n    /**\n    * Logs user out by clearing all session information.\n    *\n    * **Example**\n    *\n    *       authMgr.logout();\n    *\n    * **Parameters**\n    *\n    * @param {Object} options (Optional) Overrides for configuration options.\n    * @return {Promise}\n    */\n    logout: function (options) {\n        var me = this;\n        var adapterOptions = this.sessionManager.getMergedOptions(options);\n\n        var removeCookieFn = function (response) {\n            me.sessionManager.removeSession();\n        };\n\n        return this.authAdapter.logout(adapterOptions).then(removeCookieFn);\n    },\n\n    /**\n     * Returns the existing user access token if the user is already logged in. Otherwise, logs the user in, creating a new user access token, and returns the new token. (See [more background on access tokens](../../../project_access/)).\n     *\n     * **Example**\n     *\n     *      authMgr.getToken()\n     *          .then(function (token) {\n     *              console.log('My token is ', token);\n     *          });\n     *\n     * **Parameters**\n     * @param {Object} options (Optional) Overrides for configuration options.\n     * @return {Promise}\n     */\n    getToken: function (options) {\n        var httpOptions = this.sessionManager.getMergedOptions(options);\n\n        var session = this.sessionManager.getSession(httpOptions);\n        var $d = $.Deferred();\n        if (session.auth_token) {\n            $d.resolve(session.auth_token);\n        } else {\n            this.login(httpOptions).then($d.resolve);\n        }\n        return $d.promise();\n    },\n\n    /**\n     * Returns an array of group records, one for each group of which the current user is a member. Each group record includes the group `name`, `account`, `project`, and `groupId`.\n     *\n     * If some end users in your project are members of multiple groups, this is a useful method to call on your project's login page. When the user attempts to log in, you can use this to display the groups of which the user is member, and have the user select the correct group to log in to for this session.\n     *\n     * **Example**\n     *\n     *      // get groups for current user\n     *      var sessionObj = authMgr.getCurrentUserSessionInfo();\n     *      authMgr.getUserGroups({ userId: sessionObj.userId, token: sessionObj.auth_token })\n     *          .then(function (groups) {\n     *              for (var i=0; i < groups.length; i++)\n     *                  { console.log(groups[i].name); }\n     *          });\n     *\n     *      // get groups for particular user\n     *      authMgr.getUserGroups({userId: 'b1c19dda-2d2e-4777-ad5d-3929f17e86d3', token: savedProjAccessToken });\n     *\n     * **Parameters**\n     * @param {Object} params Object with a userId and token properties.\n     * @param {String} params.userId The userId. If looking up groups for the currently logged in user, this is in the session information. Otherwise, pass a string.\n     * @param {String} params.token The authorization credentials (access token) to use for checking the groups for this user. If looking up groups for the currently logged in user, this is in the session information. A team member's token or a project access token can access all the groups for all end users in the team or project.\n     * @param {Object} options (Optional) Overrides for configuration options.\n     * @return {Promise}\n     */\n    getUserGroups: function (params, options) {\n        var adapterOptions = this.sessionManager.getMergedOptions({ success: $.noop }, options);\n        var $d = $.Deferred();\n        var outSuccess = adapterOptions.success;\n\n        adapterOptions.success = function (memberInfo) {\n            // The member API is at the account scope, we filter by project\n            if (adapterOptions.project) {\n                memberInfo = $.grep(memberInfo, function (group) {\n                    return group.project === adapterOptions.project;\n                });\n            }\n\n            outSuccess.apply(this, [memberInfo]);\n            $d.resolve(memberInfo);\n        };\n\n        var memberAdapter = new MemberAdapter({ token: params.token, server: adapterOptions.server });\n        memberAdapter.getGroupsForUser(params, adapterOptions).fail($d.reject);\n        return $d.promise();\n    },\n\n    /**\n     * Helper method to check if you're currently logged in\n     * @return {Boolean} true if you're logged in\n     */\n    isLoggedIn: function () {\n        var session = this.getCurrentUserSessionInfo();\n        return !!(session && session.userId);\n    },\n\n    /**\n     * Returns session information for the current user, including the `userId`, `account`, `project`, `groupId`, `groupName`, `isFac` (whether the end user is a facilitator of this group), and `auth_token` (user access token).\n     *\n     * *Important*: This method is synchronous. The session information is returned immediately in an object; no callbacks or promises are needed.\n     *\n     * Session information is stored in a cookie in the browser.\n     *\n     * **Example**\n     *\n     *      var sessionObj = authMgr.getCurrentUserSessionInfo();\n     *\n     * **Parameters**\n     * @param {Object} options (Optional) Overrides for configuration options.\n     * @return {Object} session information\n     */\n    getCurrentUserSessionInfo: function (options) {\n        var adapterOptions = this.sessionManager.getMergedOptions({ success: $.noop }, options);\n        return this.sessionManager.getSession(adapterOptions);\n    },\n\n    /*\n     * Adds one or more groups to the current session. \n     *\n     * This method assumes that the project and group exist and the user specified in the session is part of this project and group.\n     *\n     * Returns the new session object.\n     *\n     * **Example**\n     *\n     *      authMgr.addGroups({ project: 'hello-world', groupName: 'groupName', groupId: 'groupId' });\n     *      authMgr.addGroups([{ project: 'hello-world', groupName: 'groupName', groupId: 'groupId' }, { project: 'hello-world', groupName: '...' }]);\n     *\n     * **Parameters**\n     * @param {object|array} groups (Required) The group object must contain the `project` (**Project ID**) and `groupName` properties. If passing an array of such objects, all of the objects must contain *different* `project` (**Project ID**) values: although end users may be logged in to multiple projects at once, they may only be logged in to one group per project at a time.\n     * @param {string} group.isFac (optional) Defaults to `false`. Set to `true` if the user in the session should be a facilitator in this group.\n     * @param {string} group.groupId (optional) Defaults to undefined. Needed mostly for the Members API.\n     * @return {Object} session information\n    */\n    addGroups: function (groups) {\n        var session = this.getCurrentUserSessionInfo();\n        var isArray = Array.isArray(groups);\n        groups = isArray ? groups : [groups];\n\n        $.each(groups, function (index, group) {\n            var extendedGroup = $.extend({}, { isFac: false }, group);\n            var project = extendedGroup.project;\n            var validProps = ['groupName', 'groupId', 'isFac'];\n            if (!project || !extendedGroup.groupName) {\n                throw new Error('No project or groupName specified.');\n            }\n            // filter object\n            extendedGroup = _pick(extendedGroup, validProps);\n            session.groups[project] = extendedGroup;\n        });\n        this.sessionManager.saveSession(session);\n        return session;\n    }\n});\n\nmodule.exports = AuthManager;\n","'use strict';\n\n/**\n * ## Channel Manager\n *\n * There are two main use cases for the channel: event notifications and chat messages.\n *\n * If you are developing with Epicenter.js, you should use the [Epicenter Channel Manager](../epicenter-channel-manager/) rather than this more generic Channel Manager. (The Epicenter Channel Manager is a wrapper that instantiates a Channel Manager with Epicenter-specific defaults.) The Epicenter Channel Manager documentation also has more [background](../epicenter-channel-manager/#background) information on channels and their use. \n *\n * However, you can work directly with the Channel Manager if you like. (This might be useful if you are working through Node.js, for example, `require('manager/channel-manager')`.)\n *\n * The Channel Manager is a wrapper around the default [cometd JavaScript library](http://docs.cometd.org/2/reference/javascript.html), `$.cometd`. It provides a few nice features that `$.cometd` doesn't, including:\n *\n * * Automatic re-subscription to channels if you lose your connection\n * * Online / Offline notifications\n * * 'Events' for cometd notifications (instead of having to listen on specific meta channels)\n *\n * You'll need to include the `epicenter-multiplayer-dependencies.js` library in addition to the `epicenter.js` library in your project to use the Channel Manager. (See [Including Epicenter.js](../../#include).)\n *\n * To use the Channel Manager in client-side JavaScript, instantiate the [Epicenter Channel Manager](../epicenter-channel-manager/), get a particular channel -- that is, an instance of a [Channel Service](../channel-service/) -- then use the channel's `subscribe()` and `publish()` methods to subscribe to topics or publish data to topics.\n *\n *      var cm = new F.manager.ChannelManager();\n *      var gc = cm.getGroupChannel();\n *      // because we used an Epicenter Channel Manager to get the group channel,\n *      // subscribe() and publish() here default to the base topic for the group;\n *      gc.subscribe('', function(data) { console.log(data); });\n *      gc.publish('', { message: 'a new message to the group' });\n *\n * The parameters for instantiating a Channel Manager include:\n *\n * * `options` The options object to configure the Channel Manager. Besides the common options listed here, see http://docs.cometd.org/reference/javascript.html for other supported options.\n * * `options.url` The Cometd endpoint URL.\n * * `options.websocketEnabled` Whether websocket support is active (boolean).\n * * `options.channel` Other defaults to pass on to instances of the underlying Channel Service. See [Channel Service](../channel-service/) for details.\n *\n */\n\nvar Channel = require('../service/channel-service');\nvar SessionManager = require('../store/session-manager');\n\nvar ChannelManager = function (options) {\n    if (!$.cometd) {\n        throw new Error('Cometd library not found. Please include epicenter-multiplayer-dependencies.js');\n    }\n    if (!options || !options.url) {\n        throw new Error('Please provide an url for the cometd server');\n    }\n\n    var defaults = {\n        /**\n         * The Cometd endpoint URL.\n         * @type {string}\n         */\n        url: '',\n\n        /**\n         * The log level for the channel (logs to console).\n         * @type {string}\n         */\n        logLevel: 'info',\n\n        /**\n         * Whether websocket support is active. Defaults to `true`.\n         * @type {boolean}\n         */\n        websocketEnabled: true,\n\n        /**\n         * Whether the ACK extension is enabled. Defaults to `true`. See [https://docs.cometd.org/current/reference/#_extensions_acknowledge](https://docs.cometd.org/current/reference/#_extensions_acknowledge) for more info.\n         * @type {boolean}\n         */\n        ackEnabled: true,\n\n        /**\n         * If false each instance of Channel will have a separate cometd connection to server, which could be noisy. Set to true to re-use the same connection across instances.\n         * @type {boolean}\n         */\n        shareConnection: false,\n\n        /**\n         * Other defaults to pass on to instances of the underlying [Channel Service](../channel-service/), which are created through `getChannel()`.\n         * @type {object}\n         */\n        channel: {\n\n        },\n\n        /**\n         * Options to pass to the channel handshake.\n         *\n         * For example, the [Epicenter Channel Manager](../epicenter-channel-manager/) passes `ext` and authorization information. More information on possible options is in the details of the underlying [Push Channel API](../../../rest_apis/multiplayer/channel/).\n         *\n         * @type {object}\n         */\n        handshake: undefined\n    };\n    this.sessionManager = new SessionManager();\n    var defaultCometOptions = this.sessionManager.getMergedOptions(defaults, options);\n    this.currentSubscriptions = [];\n    this.options = defaultCometOptions;\n\n    if (defaultCometOptions.shareConnection && ChannelManager.prototype._cometd) {\n        this.cometd = ChannelManager.prototype._cometd;\n        return this;\n    }\n    var cometd = new $.CometD();\n    ChannelManager.prototype._cometd = cometd;\n\n    cometd.websocketEnabled = defaultCometOptions.websocketEnabled;\n    cometd.ackEnabled = defaultCometOptions.ackEnabled;\n\n    this.isConnected = false;\n    var connectionBroken = function (message) {\n        $(this).trigger('disconnect', message);\n    };\n    var connectionSucceeded = function (message) {\n        $(this).trigger('connect', message);\n    };\n    var me = this;\n\n    cometd.configure(defaultCometOptions);\n\n    cometd.addListener('/meta/connect', function (message) {\n        var wasConnected = this.isConnected;\n        this.isConnected = (message.successful === true);\n        if (!wasConnected && this.isConnected) { //Connecting for the first time\n            connectionSucceeded.call(this, message);\n        } else if (wasConnected && !this.isConnected) { //Only throw disconnected message fro the first disconnect, not once per try\n            connectionBroken.call(this, message);\n        }\n    }.bind(this));\n\n    cometd.addListener('/meta/disconnect', connectionBroken);\n\n    cometd.addListener('/meta/handshake', function (message) {\n        if (message.successful) {\n            //http://docs.cometd.org/reference/javascript_subscribe.html#javascript_subscribe_meta_channels\n            // ^ \"dynamic subscriptions are cleared (like any other subscription) and the application needs to figure out which dynamic subscription must be performed again\"\n            cometd.batch(function () {\n                $(me.currentSubscriptions).each(function (index, subs) {\n                    cometd.resubscribe(subs);\n                });\n            });\n        }\n    });\n\n    //Other interesting events for reference\n    cometd.addListener('/meta/subscribe', function (message) {\n        $(me).trigger('subscribe', message);\n    });\n    cometd.addListener('/meta/unsubscribe', function (message) {\n        $(me).trigger('unsubscribe', message);\n    });\n    cometd.addListener('/meta/publish', function (message) {\n        $(me).trigger('publish', message);\n    });\n    cometd.addListener('/meta/unsuccessful', function (message) {\n        $(me).trigger('error', message);\n    });\n\n    cometd.handshake(defaultCometOptions.handshake);\n\n    this.cometd = cometd;\n};\n\n\nChannelManager.prototype = $.extend(ChannelManager.prototype, {\n\n    /**\n     * Creates and returns a channel, that is, an instance of a [Channel Service](../channel-service/).\n     *\n     * **Example**\n     *\n     *      var cm = new F.manager.ChannelManager();\n     *      var channel = cm.getChannel();\n     *\n     *      channel.subscribe('topic', callback);\n     *      channel.publish('topic', { myData: 100 });\n     *\n     * **Parameters**\n     * @param {Object|String} options (Optional) If string, assumed to be the base channel url. If object, assumed to be configuration options for the constructor.\n     * @return {Channel} Channel instance\n     */\n    getChannel: function (options) {\n        //If you just want to pass in a string\n        if (options && !$.isPlainObject(options)) {\n            options = {\n                base: options\n            };\n        }\n        var defaults = {\n            transport: this.cometd\n        };\n        var channel = new Channel($.extend(true, {}, this.options.channel, defaults, options));\n\n\n        //Wrap subs and unsubs so we can use it to re-attach handlers after being disconnected\n        var subs = channel.subscribe;\n        channel.subscribe = function () {\n            var subid = subs.apply(channel, arguments);\n            this.currentSubscriptions = this.currentSubscriptions.concat(subid);\n            return subid;\n        }.bind(this);\n\n\n        var unsubs = channel.unsubscribe;\n        channel.unsubscribe = function () {\n            var removed = unsubs.apply(channel, arguments);\n            for (var i = 0; i < this.currentSubscriptions.length; i++) {\n                if (this.currentSubscriptions[i].id === removed.id) {\n                    this.currentSubscriptions.splice(i, 1);\n                }\n            }\n            return removed;\n        }.bind(this);\n\n        return channel;\n    },\n\n    /**\n     * Start listening for events on this instance. Signature is same as for jQuery Events: http://api.jquery.com/on/.\n     *\n     * Supported events are: `connect`, `disconnect`, `subscribe`, `unsubscribe`, `publish`, `error`.\n     *\n     * **Parameters**\n     *\n     * @param {string} event The event type. See more detail at jQuery Events: http://api.jquery.com/on/.\n     */\n    on: function (event) {\n        $(this).on.apply($(this), arguments);\n    },\n\n    /**\n     * Stop listening for events on this instance. Signature is same as for jQuery Events: http://api.jquery.com/off/.\n     *\n     * **Parameters**\n     *\n     * @param {string} event The event type. See more detail at jQuery Events: http://api.jquery.com/off/.\n     */\n    off: function (event) {\n        $(this).off.apply($(this), arguments);\n    },\n\n    /**\n     * Trigger events and execute handlers. Signature is same as for jQuery Events: http://api.jquery.com/trigger/.\n     *\n     * **Parameters**\n     *\n     * @param {string} event The event type. See more detail at jQuery Events: http://api.jquery.com/trigger/.\n     */\n    trigger: function (event) {\n        $(this).trigger.apply($(this), arguments);\n    }\n});\n\nmodule.exports = ChannelManager;\n","'use strict';\n\n/**\n * ## Epicenter Channel Manager\n *\n * The Epicenter platform provides a push channel, which allows you to publish and subscribe to messages within a [project](../../../glossary/#projects), [group](../../../glossary/#groups), or [multiplayer world](../../../glossary/#world).\n *\n * <a name=\"background\"></a>\n * ### Channel Background\n *\n * Channel notifications are only available for [team projects](../../../glossary/#team). There are two main use cases for the push channel: event notifications and chat messages.\n *\n * #### Event Notifications\n *\n * Within a [multiplayer simulation or world](../../../glossary/#world), it is often useful for your project's [model](../../../writing_your_model/) to alert the [user interface (browser)](../../../creating_your_interface/) that something new has happened.\n *\n * Usually, this \"something new\" is an event within the project, group, or world, such as:\n *\n * * An end user comes online (logs in) or goes offline. (This is especially interesting in a multiplayer world; only available if you have [enabled authorization](../../../updating_your_settings/#general-settings) for the channel.)\n * * An end user is assigned to a world.\n * * An end user updates a variable / makes a decision.\n * * An end user creates or updates data stored in the [Data API](../data-api-service/).\n * * An operation (method) is called. (This is especially interesting if the model is advanced, for instance, the Vensim `step` operation is called.)\n *\n * When these events occur, you often want to have the user interface for one or more end users automatically update with new information.\n *\n * #### Chat Messages\n *\n * Another reason to use the push channel is to allow players (end users) to send chat messages to other players, and to have those messages appear immediately.\n *\n * #### Getting Started\n *\n * For both the event notification and chat message use cases:\n *\n * * First, enable channel notifications for your project.\n *      * Channel notifications are only available for [team projects](../../../glossary/#team). To enable notifications for your project, [update your project settings](../../../updating_your_settings/#general-settings) to turn on the **Push Channel** setting, and optionally require authorization for the channel.\n * * Then, instantiate an Epicenter Channel Manager.\n * * Next, get the channel with the scope you want (user, world, group, data).\n * * Finally, use the channel's `subscribe()` and `publish()` methods to subscribe to topics or publish data to topics.\n *\n * Here's an example of those last three steps (instantiate, get channel, subscribe):\n *\n *     var cm = new F.manager.ChannelManager();\n *     var gc = cm.getGroupChannel();\n *     gc.subscribe('', function(data) { console.log(data); });\n *     gc.publish('', { message: 'a new message to the group' });\n *\n * For a more detailed example, see a [complete publish and subscribe example](../../../rest_apis/multiplayer/channel/#epijs-example).\n *\n * For details on what data is published automatically to which channels, see [Automatic Publishing of Events](../../../rest_apis/multiplayer/channel/#publish-message-auto).\n *\n * #### Creating an Epicenter Channel Manager\n *\n * The Epicenter Channel Manager is a wrapper around the (more generic) [Channel Manager](../channel-manager/), to instantiate it with Epicenter-specific defaults. If you are interested in including a notification or chat feature in your project, using an Epicenter Channel Manager is the easiest way to get started.\n *\n * You'll need to include the `epicenter-multiplayer-dependencies.js` library in addition to the `epicenter.js` library in your project to use the Epicenter Channel Manager. See [Including Epicenter.js](../../#include).\n *\n * The parameters for instantiating an Epicenter Channel Manager include:\n *\n * * `options` Object with details about the Epicenter project for this Epicenter Channel Manager instance.\n * * `options.account` The Epicenter account id (**Team ID** for team projects, **User ID** for personal projects).\n * * `options.project` Epicenter project id.\n * * `options.userName` Epicenter userName used for authentication.\n * * `options.userId` Epicenter user id used for authentication. Optional; `options.userName` is preferred.\n * * `options.token` Epicenter token used for authentication. (You can retrieve this using `authManager.getToken()` from the [Authorization Manager](../auth-manager/).)\n * * `options.allowAllChannels` If not included or if set to `false`, all channel paths are validated; if your project requires [Push Channel Authorization](../../../updating_your_settings/), you should use this option. If you want to allow other channel paths, set to `true`; this is not common.\n */\n\nvar ChannelManager = require('./channel-manager');\nvar ConfigService = require('../service/configuration-service');\nvar classFrom = require('../util/inherit');\nvar SessionManager = require('../store/session-manager');\n\nvar validTypes = {\n    project: true,\n    group: true,\n    world: true,\n    user: true,\n    data: true,\n    general: true,\n    chat: true\n};\nvar getFromSessionOrError = function (value, sessionKeyName, settings) {\n    if (!value) {\n        if (settings && settings[sessionKeyName]) {\n            value = settings[sessionKeyName];\n        } else {\n            throw new Error(sessionKeyName + ' not found. Please log-in again, or specify ' + sessionKeyName + ' explicitly');\n        }\n    }\n    return value;\n};\n\nvar isPresenceData = function (payload) {\n    return payload.data && payload.data.type === 'user' && payload.data.user;\n};\n\nvar __super = ChannelManager.prototype;\nvar EpicenterChannelManager = classFrom(ChannelManager, {\n    constructor: function (options) {\n        this.sessionManager = new SessionManager(options);\n        var defaultCometOptions = this.sessionManager.getMergedOptions(options);\n\n        var urlConfig = new ConfigService(defaultCometOptions).get('server');\n        if (!defaultCometOptions.url) {\n            defaultCometOptions.url = urlConfig.getAPIPath('channel');\n        }\n\n        if (defaultCometOptions.handshake === undefined) {\n            var userName = defaultCometOptions.userName;\n            var userId = defaultCometOptions.userId;\n            var token = defaultCometOptions.token;\n            if ((userName || userId) && token) {\n                var userProp = userName ? 'userName' : 'userId';\n                var ext = {\n                    authorization: 'Bearer ' + token\n                };\n                ext[userProp] = userName ? userName : userId;\n\n                defaultCometOptions.handshake = {\n                    ext: ext\n                };\n            }\n        }\n\n        this.options = defaultCometOptions;\n        return __super.constructor.call(this, defaultCometOptions);\n    },\n\n    /**\n     * Creates and returns a channel, that is, an instance of a [Channel Service](../channel-service/).\n     *\n     * This method enforces Epicenter-specific channel naming: all channels requested must be in the form `/{type}/{account id}/{project id}/{...}`, where `type` is one of `run`, `data`, `user`, `world`, or `chat`.\n     *\n     * **Example**\n     *\n     *      var cm = new F.manager.EpicenterChannelManager();\n     *      var channel = cm.getChannel('/group/acme/supply-chain-game/');\n     *\n     *      channel.subscribe('topic', callback);\n     *      channel.publish('topic', { myData: 100 });\n     *\n     * **Parameters**\n     * @param {Object|String} options (Optional) If string, assumed to be the base channel url. If object, assumed to be configuration options for the constructor.\n     * @return {Channel} Channel instance\n     */\n    getChannel: function (options) {\n        if (options && typeof options !== 'object') {\n            options = {\n                base: options\n            };\n        }\n        var channelOpts = $.extend({}, this.options, options);\n        var base = channelOpts.base;\n        if (!base) {\n            throw new Error('No base topic was provided');\n        }\n\n        if (!channelOpts.allowAllChannels) {\n            var baseParts = base.split('/');\n            var channelType = baseParts[1];\n            if (baseParts.length < 4) { //eslint-disable-line\n                throw new Error('Invalid channel base name, it must be in the form /{type}/{account id}/{project id}/{...}');\n            }\n            if (!validTypes[channelType]) {\n                throw new Error('Invalid channel type');\n            }\n        }\n        return __super.getChannel.apply(this, arguments);\n    },\n\n    /**\n     * Create and return a publish/subscribe channel (from the underlying [Channel Manager](../channel-manager/)) for the given [group](../../../glossary/#groups). The group must exist in the account (team) and project provided.\n     *\n     * There are no notifications from Epicenter on this channel; all messages are user-originated.\n     *\n     * **Example**\n     *\n     *     var cm = new F.manager.ChannelManager();\n     *     var gc = cm.getGroupChannel();\n     *     gc.subscribe('broadcasts', callback);\n     *\n     * **Return Value**\n     *\n     * * *Channel* Returns the channel (an instance of the [Channel Service](../channel-service/)).\n     *\n     * **Parameters**\n     *\n     * @param  {String} groupName (Optional) Group to broadcast to. If not provided, picks up group from current session if end user is logged in.\n     * @return {Channel} Channel instance\n     */\n    getGroupChannel: function (groupName) {\n        var session = this.sessionManager.getMergedOptions(this.options);\n        groupName = getFromSessionOrError(groupName, 'groupName', session);\n        var account = getFromSessionOrError('', 'account', session);\n        var project = getFromSessionOrError('', 'project', session);\n\n        var baseTopic = ['/group', account, project, groupName].join('/');\n        var channel = __super.getChannel.call(this, { base: baseTopic });\n        var oldsubs = channel.subscribe;\n        channel.subscribe = function (topic, callback, context, options) {\n            var callbackWithoutPresenceData = function (payload) {\n                if (!isPresenceData(payload)) {\n                    callback.call(context, payload);\n                }\n            };\n            return oldsubs.call(channel, topic, callbackWithoutPresenceData, context, options);\n        };\n        return channel;\n    },\n\n    /**\n     * Create and return a publish/subscribe channel (from the underlying [Channel Manager](../channel-manager/)) for the given [world](../../../glossary/#world).\n     *\n     * This is typically used together with the [World Manager](../world-manager).\n     *\n     * **Example**\n     *\n     *     var cm = new F.manager.ChannelManager();\n     *     var worldManager = new F.manager.WorldManager({\n     *         account: 'acme-simulations',\n     *         project: 'supply-chain-game',\n     *         group: 'team1',\n     *         run: { model: 'model.eqn' }\n     *     });\n     *     worldManager.getCurrentWorld().then(function (worldObject, worldAdapter) {\n     *         var worldChannel = cm.getWorldChannel(worldObject);\n     *         worldChannel.subscribe('', function (data) {\n     *             console.log(data);\n     *         });\n     *      });\n     *\n     * **Return Value**\n     *\n     * * *Channel* Returns the channel (an instance of the [Channel Service](../channel-service/)).\n     *\n     * **Parameters**\n     *\n     * @param  {String|Object} world The world object or id.\n     * @param  {String} groupName (Optional) Group the world exists in. If not provided, picks up group from current session if end user is logged in.\n     * @return {Channel} Channel instance\n     */\n    getWorldChannel: function (world, groupName) {\n        var worldid = ($.isPlainObject(world) && world.id) ? world.id : world;\n        if (!worldid) {\n            throw new Error('Please specify a world id');\n        }\n        var session = this.sessionManager.getMergedOptions(this.options);\n\n        groupName = getFromSessionOrError(groupName, 'groupName', session);\n        var account = getFromSessionOrError('', 'account', session);\n        var project = getFromSessionOrError('', 'project', session);\n\n        var baseTopic = ['/world', account, project, groupName, worldid].join('/');\n        return __super.getChannel.call(this, { base: baseTopic });\n    },\n\n    /**\n     * Create and return a publish/subscribe channel (from the underlying [Channel Manager](../channel-manager/)) for the current [end user](../../../glossary/#users) in that user's current [world](../../../glossary/#world).\n     *\n     * This is typically used together with the [World Manager](../world-manager). Note that this channel only gets notifications for worlds currently in memory. (See more background on [persistence](../../../run_persistence).)\n     *\n     * **Example**\n     *\n     *     var cm = new F.manager.ChannelManager();\n     *     var worldManager = new F.manager.WorldManager({\n     *         account: 'acme-simulations',\n     *         project: 'supply-chain-game',\n     *         group: 'team1',\n     *         run: { model: 'model.eqn' }\n     *     });\n     *     worldManager.getCurrentWorld().then(function (worldObject, worldAdapter) {\n     *         var userChannel = cm.getUserChannel(worldObject);\n     *         userChannel.subscribe('', function (data) {\n     *             console.log(data);\n     *         });\n     *      });\n     *\n     *\n     * **Return Value**\n     *\n     * * *Channel* Returns the channel (an instance of the [Channel Service](../channel-service/)).\n     *\n     * **Parameters**\n     *\n     * @param  {String|Object} world World object or id.\n     * @param  {String|Object} user (Optional) User object or id. If not provided, picks up user id from current session if end user is logged in.\n     * @param  {String} groupName (Optional) Group the world exists in. If not provided, picks up group from current session if end user is logged in.\n     * @return {Channel} Channel instance\n     */\n    getUserChannel: function (world, user, groupName) {\n        var worldid = ($.isPlainObject(world) && world.id) ? world.id : world;\n        if (!worldid) {\n            throw new Error('Please specify a world id');\n        }\n        var session = this.sessionManager.getMergedOptions(this.options);\n\n        var userid = ($.isPlainObject(user) && user.id) ? user.id : user;\n        userid = getFromSessionOrError(userid, 'userId', session);\n        groupName = getFromSessionOrError(groupName, 'groupName', session);\n\n        var account = getFromSessionOrError('', 'account', session);\n        var project = getFromSessionOrError('', 'project', session);\n\n        var baseTopic = ['/user', account, project, groupName, worldid, userid].join('/');\n        return __super.getChannel.call(this, { base: baseTopic });\n    },\n\n    /**\n     * Create and return a publish/subscribe channel (from the underlying [Channel Manager](../channel-manager/)) that automatically tracks the presence of an [end user](../../../glossary/#users), that is, whether the end user is currently online in this group. Notifications are automatically sent when the end user comes online, and when the end user goes offline (not present for more than 2 minutes). Useful in multiplayer games for letting each end user know whether other users in their group are also online.\n     *\n     * Note that the presence channel is tracking all end users in a group. In particular, if the project additionally splits each group into [worlds](../world-manager/), this channel continues to show notifications for all end users in the group (not restricted by worlds).\n     *\n     * **Example**\n     *\n     *     var cm = new F.manager.ChannelManager();\n     *     var pc = cm.getPresenceChannel();\n     *     pc.subscribe('', function (data) {\n     *          // 'data' is the entire message object to the channel;\n     *          // parse for information of interest\n     *          if (data.data.subType === 'disconnect') {\n     *               console.log('user ', data.data.user.userName, 'disconnected at ', data.data.date);\n     *          }\n     *          if (data.data.subType === 'connect') {\n     *               console.log('user ', data.data.user.userName, 'connected at ', data.data.date);\n     *          }\n     *     });\n     *\n     *\n     * **Return Value**\n     *\n     * * *Channel* Returns the channel (an instance of the [Channel Service](../channel-service/)).\n     *\n     * **Parameters**\n     *\n     * @param  {String} groupName (Optional) Group the end user is in. If not provided, picks up group from current session if end user is logged in.\n     * @return {Channel} Channel instance\n     */\n    getPresenceChannel: function (groupName) {\n        var session = this.sessionManager.getMergedOptions(this.options);\n        groupName = getFromSessionOrError(groupName, 'groupName', session);\n        var account = getFromSessionOrError('', 'account', session);\n        var project = getFromSessionOrError('', 'project', session);\n\n        var baseTopic = ['/group', account, project, groupName].join('/');\n        var channel = __super.getChannel.call(this, { base: baseTopic });\n        var oldsubs = channel.subscribe;\n        channel.subscribe = function (topic, callback, context, options) {\n            var callbackWithOnlyPresenceData = function (payload) {\n                if (isPresenceData(payload)) {\n                    callback.call(context, payload);\n                }\n            };\n            return oldsubs.call(channel, topic, callbackWithOnlyPresenceData, context, options);\n        };\n        return channel;\n    },\n\n    /**\n     * Create and return a publish/subscribe channel (from the underlying [Channel Manager](../channel-manager/)) for the given collection. (The collection name is specified in the `root` argument when the [Data Service](../data-api-service/) is instantiated.) Must be one of the collections in this account (team) and project.\n     *\n     * There are automatic notifications from Epicenter on this channel when data is created, updated, or deleted in this collection. See more on [automatic messages to the data channel](../../../rest_apis/multiplayer/channel/#data-messages).\n     *\n     * **Example**\n     *\n     *     var cm = new F.manager.ChannelManager();\n     *     var dc = cm.getDataChannel('survey-responses');\n     *     dc.subscribe('', function(data, meta) {\n     *          console.log(data);\n     *\n     *          // meta.date is time of change,\n     *          // meta.subType is the kind of change: new, update, or delete\n     *          // meta.path is the full path to the changed data\n     *          console.log(meta);\n     *     });\n     *\n     * **Return Value**\n     *\n     * * *Channel* Returns the channel (an instance of the [Channel Service](../channel-service/)).\n     *\n     * **Parameters**\n     *\n     * @param  {String} collection Name of collection whose automatic notifications you want to receive.\n     * @return {Channel} Channel instance\n     */\n    getDataChannel: function (collection) {\n        if (!collection) {\n            throw new Error('Please specify a collection to listen on.');\n        }\n\n        var session = this.sessionManager.getMergedOptions(this.options);\n        var account = getFromSessionOrError('', 'account', session);\n        var project = getFromSessionOrError('', 'project', session);\n        var baseTopic = ['/data', account, project, collection].join('/');\n        var channel = __super.getChannel.call(this, { base: baseTopic });\n\n        //TODO: Fix after Epicenter bug is resolved\n        var oldsubs = channel.subscribe;\n        channel.subscribe = function (topic, callback, context, options) {\n            var callbackWithCleanData = function (payload) {\n                var meta = {\n                    path: payload.channel,\n                    subType: payload.data.subType,\n                    date: payload.data.date,\n                    dataPath: payload.data.data.path,\n                };\n                var actualData = payload.data.data;\n                if (actualData.data !== undefined) { //Delete notifications are one data-level behind of course\n                    actualData = actualData.data;\n                }\n\n                callback.call(context, actualData, meta);\n            };\n            return oldsubs.call(channel, topic, callbackWithCleanData, context, options);\n        };\n\n        return channel;\n    }\n});\n\nmodule.exports = EpicenterChannelManager;\n","'use strict';\n\nmodule.exports = {\n    EPI_SESSION_KEY: 'epicenterjs.session',\n    STRATEGY_SESSION_KEY: 'epicenter-scenario'\n};","/**\n* ## Run Manager\n*\n* The Run Manager gives you access to runs for your project. This allows you to read and update variables, call operations, etc. Additionally, the Run Manager gives you control over run creation depending on run states. Specifically, you can select [run creation strategies (rules)](../strategies/) for which runs end users of your project work with when they log in to your project.\n*\n* There are many ways to create new runs, including the Epicenter.js [Run Service](../run-api-service/) and the RESFTful [Run API](../../../rest_apis/aggregate_run_api). However, for some projects it makes more sense to pick up where the user left off, using an existing run. And in some projects, whether to create a new run or use an existing one is conditional, for example based on characteristics of the existing run or your own knowledge about the model. The Run Manager provides this level of control: your call to `getRun()`, rather than always returning a new run, returns a run based on the strategy you've specified.\n*\n*\n* ### Using the Run Manager to create and access runs\n*\n* To use the Run Manager, instantiate it by passing in:\n*\n*   * `run`: (required) Run object. Must contain:\n*       * `account`: Epicenter account id (**Team ID** for team projects, **User ID** for personal projects).\n*       * `project`: Epicenter project id.\n*       * `model`: The name of your primary model file. (See more on [Writing your Model](../../../writing_your_model/).)\n*       * `scope`: (optional) Scope object for the run, for example `scope.group` with value of the name of the group.\n*       * `server`: (optional) An object with one field, `host`. The value of `host` is the string `api.forio.com`, the URI of the Forio server. This is automatically set, but you can pass it explicitly if desired. It is most commonly used for clarity when you are [hosting an Epicenter project on your own server](../../../how_to/self_hosting/).\n*       * `files`: (optional) If and only if you are using a Vensim model and you have additional data to pass in to your model, you can optionally pass a `files` object with the names of the files, for example: `\"files\": {\"data\": \"myExtraData.xls\"}`. (See more on [Using External Data in Vensim](../../../model_code/vensim/vensim_example_xls/).)\n*\n*   * `strategy`: (optional) Run creation strategy for when to create a new run and when to reuse an end user's existing run. This is *optional*; by default, the Run Manager selects `reuse-per-session`, or `reuse-last-initialized` if you also pass in an initial operation. See [below](#using-the-run-manager-to-access-and-register-strategies) for more information on strategies.\n*\n*   * `strategyOptions`: (optional) Additional options passed directly to the [run creation strategy](../strategies/).\n*\n*   * `sessionKey`: (optional) Name of browser cookie in which to store run information, including run id. Many conditional strategies, including the provided strategies, rely on this browser cookie to store the run id and help make the decision of whether to create a new run or use an existing one. The name of this cookie defaults to `epicenter-scenario` and can be set with the `sessionKey` parameter. This can also be a function which returns a string, if you'd like to control this at runtime.\n*\n*\n* After instantiating a Run Manager, make a call to `getRun()` whenever you need to access a run for this end user. The `RunManager.run` contains the instantiated [Run Service](../run-api-service/). The Run Service allows you to access variables, call operations, etc.\n*\n* **Example**\n*\n*       var rm = new F.manager.RunManager({\n*           run: {\n*               account: 'acme-simulations',\n*               project: 'supply-chain-game',\n*               model: 'supply-chain-model.jl',\n*               server: { host: 'api.forio.com' }\n*           },\n*           strategy: 'reuse-never',\n*           sessionKey: 'epicenter-session'\n*       });\n*       rm.getRun()\n*           .then(function(run) {\n*               // the return value of getRun() is a run object\n*               var thisRunId = run.id;\n*               // the RunManager.run also contains the instantiated Run Service,\n*               // so any Run Service method is valid here\n*               rm.run.do('runModel');\n*       })\n*\n*\n* ### Using the Run Manager to access and register strategies\n*\n* The `strategy` for a Run Manager describes when to create a new run and when to reuse an end user's existing run. The Run Manager is responsible for passing a strategy everything it might need to determine the 'correct' run, that is, how to find the best existing run and how to decide when to create a new run.\n*\n* There are several common strategies provided as part of Epicenter.js, which you can list by accessing `F.manager.RunManager.strategies`. You can also create your own strategies, and register them to use with Run Managers. See [Run Manager Strategies](../strategies/) for details.\n* \n*/\n\n'use strict';\nvar strategies = require('./run-strategies');\nvar specialOperations = require('./special-operations');\n\nvar RunService = require('../service/run-api-service');\nvar SessionManager = require('../store/session-manager');\n\nvar util = require('../util/object-util');\nvar keyNames = require('./key-names');\n\nfunction patchRunService(service, manager) {\n    if (service.patched) {\n        return service;\n    }\n\n    var orig = service.do;\n    service.do = function (operation, params, options) {\n        var reservedOps = Object.keys(specialOperations);\n        if (reservedOps.indexOf(operation) === -1) {\n            return orig.apply(service, arguments);\n        } else {\n            return specialOperations[operation].call(service, params, options, manager);\n        }\n    };\n\n    service.patched = true;\n\n    return service;\n}\n\nfunction sessionKeyFromOptions(options, runService) {\n    var config = runService.getCurrentConfig();\n    var sessionKey = $.isFunction(options.sessionKey) ? options.sessionKey(config) : options.sessionKey;\n    return sessionKey;\n}\n\nfunction setRunInSession(sessionKey, run, sessionManager) {\n    if (sessionKey) {\n        delete run.variables;\n        sessionManager.getStore().set(sessionKey, JSON.stringify(run));\n    }\n}\n\nvar defaults = {\n    sessionKey: function (config) { \n        var baseKey = keyNames.STRATEGY_SESSION_KEY;\n        var key = ['account', 'project', 'model'].reduce(function (accum, key) {\n            return config[key] ? accum + '-' + config[key] : accum; \n        }, baseKey);\n        return key;\n    }\n};\n\nfunction RunManager(options) {\n    this.options = $.extend(true, {}, defaults, options);\n\n    if (this.options.run instanceof RunService) {\n        this.run = this.options.run;\n    } else if (!util.isEmpty(this.options.run)) {\n        this.run = new RunService(this.options.run);\n    } else {\n        throw new Error('No run options passed to RunManager');\n    }\n    patchRunService(this.run, this);\n\n    this.strategy = strategies.getBestStrategy(this.options);\n    this.sessionManager = new SessionManager(this.options);\n}\n\nRunManager.prototype = {\n    /**\n     * Returns the run object for the 'correct' run. The correct run is defined by the strategy. \n     *\n     * For example, if the strategy is `reuse-never`, the call\n     * to `getRun()` always returns a newly created run; if the strategy is `reuse-per-session`,\n     * `getRun()` returns the run currently referenced in the browser cookie, and if there is none, creates a new run. \n     * See [Run Manager Strategies](../strategies/) for more on strategies.\n     *\n     *  **Example**\n     *\n     *      rm.getRun().then(function (run) {\n     *          // use the run object\n     *          var thisRunId = run.id;\n     *\n     *          // use the Run Service object\n     *          rm.run.do('runModel');\n     *      });\n     *\n     *      rm.getRun(['sample_int']).then(function (run) {\n     *         // an object whose fields are the name : value pairs of the variables passed to getRun()\n     *         console.log(run.variables);\n     *         // the value of sample_int\n     *         console.log(run.variables.sample_int); \n     *      });\n     *\n     * @param {Array} variables (Optional) The run object is populated with the provided model variables, if provided. Note: `getRun()` does not throw an error if you try to get a variable which doesn't exist. Instead, the variables list is empty, and any errors are logged to the console.\n     * @param {Object} options (Optional) Configuration options; passed on to [RunService#create](../run-api-service/#create) if the strategy does create a new run.\n     * @return {$promise} Promise to complete the call.\n     */\n    getRun: function (variables, options) {\n        var me = this;\n        var sessionStore = this.sessionManager.getStore();\n\n        var sessionContents = sessionStore.get(sessionKeyFromOptions(this.options, me.run));\n        var runSession = JSON.parse(sessionContents || '{}');\n        \n        if (runSession.runId) {\n            //EpiJS < 2.2 used runId as key, so maintain comptaibility. Remove at some future date (Summer `17?)\n            runSession.id = runSession.runId;\n        }\n\n        var authSession = this.sessionManager.getSession();\n        if (this.strategy.requiresAuth && util.isEmpty(authSession)) {\n            console.error('No user-session available', this.options.strategy, 'requires authentication.');\n            return $.Deferred().reject('No user-session available').promise();\n        }\n        return this.strategy\n                .getRun(this.run, authSession, runSession, options).then(function (run) {\n                    if (run && run.id) {\n                        me.run.updateConfig({ filter: run.id });\n                        var sessionKey = sessionKeyFromOptions(me.options, me.run);\n                        setRunInSession(sessionKey, run, me.sessionManager);\n\n                        if (variables && variables.length) {\n                            return me.run.variables().query(variables).then(function (results) {\n                                run.variables = results;\n                                return run;\n                            }).catch(function (err) {\n                                run.variables = {};\n                                console.error(err);\n                                return run;\n                            });\n                        }\n                    }\n                    return run;\n                });\n    },\n\n    /**\n     * Returns the run object for a 'reset' run. The definition of a reset is defined by the strategy, but typically means forcing the creation of a new run. For example, `reset()` for the default strategies `reuse-per-session` and `reuse-last-initialized` both create new runs.\n     *\n     *  **Example**\n     *\n     *      rm.reset().then(function (run) {\n     *          // use the (new) run object\n     *          var thisRunId = run.id;\n     *\n     *          // use the Run Service object\n     *          rm.run.do('runModel');\n     *      });\n     *\n     * **Parameters**\n     * @param {Object} options (Optional) Configuration options; passed on to [RunService#create](../run-api-service/#create).\n     * @return {Promise}\n     */\n    reset: function (options) {\n        var me = this;\n        var authSession = this.sessionManager.getSession();\n        if (this.strategy.requiresAuth && util.isEmpty(authSession)) {\n            console.error('No user-session available', this.options.strategy, 'requires authentication.');\n            return $.Deferred().reject('No user-session available').promise();\n        }\n        return this.strategy.reset(this.run, authSession, options).then(function (run) {\n            if (run && run.id) {\n                me.run.updateConfig({ filter: run.id });\n                var sessionKey = sessionKeyFromOptions(me.options, me.run);\n                setRunInSession(sessionKey, run.id, me.sessionManager);\n            }\n            return run;\n        });\n    }\n};\n\nRunManager.strategies = strategies;\nmodule.exports = RunManager;\n","'use strict';\n\nvar Base = require('./none-strategy');\nvar classFrom = require('../../util/inherit');\n\n/**\n* ## Conditional Creation Strategy\n*\n* This strategy will try to get the run stored in the cookie and\n* evaluate if it needs to create a new run by calling the `condition` function.\n*/\n\nvar Strategy = classFrom(Base, {\n    constructor: function Strategy(condition) {\n        if (condition == null) { //eslint-disable-line\n            throw new Error('Conditional strategy needs a condition to create a run');\n        }\n        this.condition = typeof condition !== 'function' ? function () { return condition; } : condition;\n    },\n\n    /**\n     * Gets a new 'correct' run, or updates the existing one (the definition of 'correct' depends on strategy implementation).\n     * @param  {RunService} runService A Run Service instance for the current run, as determined by the Run Manager.\n     * @param  {Object} userSession Information about the current user session. See [AuthManager#getCurrentUserSessionInfo](../auth-manager/#getcurrentusersessioninfo) for format.\n     * @param  {Object} options (Optional) See [RunService#create](../run-api-service/#create) for supported options.\n     * @return {Promise}             \n     */\n    reset: function (runService, userSession, options) {\n        var group = userSession && userSession.groupName;\n        var opt = $.extend({\n            scope: { group: group }\n        }, runService.getCurrentConfig());\n\n        return runService\n                .create(opt, options)\n                .then(function (run) {\n                    run.freshlyCreated = true;\n                    return run;\n                });\n    },\n\n    /**\n     * Gets the 'correct' run (the definition of 'correct' depends on strategy implementation).\n     * @param  {RunService} runService A Run Service instance for the current run, as determined by the Run Manager.\n     * @param  {Object} userSession Information about the current user session. See [AuthManager#getCurrentUserSessionInfo](../auth-manager/#getcurrentusersessioninfo) for format.\n     * @param  {Object} runSession The Run Manager stores the 'last accessed' run in a cookie and passes it back here.\n     * @param  {Object} options (Optional) See [RunService#create](../run-api-service/#create) for supported options.\n     * @return {Promise}             \n     */\n    getRun: function (runService, userSession, runSession, options) {\n        var me = this;\n        if (runSession && runSession.id) {\n            return this.loadAndCheck(runService, userSession, runSession, options).catch(function () {\n                return me.reset(runService, userSession, options); //if it got the wrong cookie for e.g.\n            });\n        } else {\n            return this.reset(runService, userSession, options);\n        }\n    },\n\n    loadAndCheck: function (runService, userSession, runSession, options) {\n        var shouldCreate = false;\n        var me = this;\n\n        return runService\n            .load(runSession.id, null, {\n                success: function (run, msg, headers) {\n                    shouldCreate = me.condition(run, headers, userSession, runSession);\n                }\n            })\n            .then(function (run) {\n                if (shouldCreate) {\n                    return me.reset(runService, userSession, options);\n                }\n                return run;\n            });\n    }\n});\n\nmodule.exports = Strategy;\n","/**\n * The `new-if-initialized` strategy creates a new run if the current one is in memory or has its `initialized` field set to `true`. The `initialized` field in the run record is automatically set to `true` at run creation, but can be changed.\n * \n * This strategy is useful if your project is structured such that immediately after a run is created, the model is executed completely (for example, a Vensim model is stepped to the end). It is similar to the `new-if-missing` strategy, except that it checks a field of the run record.\n * \n * Specifically, the strategy is:\n *\n * * Check the `sessionKey` cookie. \n *  * This cookie is set by the [Run Manager](../run-manager/) and configurable through its options.\n *  * If the cookie exists, check whether the run is in memory or only persisted in the database. Additionally, check whether the run's `initialized` field is `true`. \n *      * If the run is in memory, create a new run.\n *      * If the run's `initialized` field is `true`, create a new run.\n *      * Otherwise, use the existing run.\n *  * If the cookie does not exist, create a new run for this end user.\n *  \n *  @deprecated Consider using `reuse-last-initialized` instead\n */\n\n'use strict';\nvar classFrom = require('../../../util/inherit');\nvar ConditionalStrategy = require('../conditional-creation-strategy');\n\nvar __super = ConditionalStrategy.prototype;\n\nvar Strategy = classFrom(ConditionalStrategy, {\n    constructor: function (options) {\n        __super.constructor.call(this, this.createIf, options);\n        console.warn('This strategy is deprecated; all runs now default to being initialized by default making this redundant. Consider using `reuse-last-initialized` instead.');\n    },\n\n    createIf: function (run, headers) {\n        return headers.getResponseHeader('pragma') === 'persistent' || run.initialized;\n    }\n});\n\nmodule.exports = Strategy;\n","/**\n * The `new-if-persisted` strategy creates a new run when the current one becomes persisted (end user is idle for a set period), but otherwise uses the current one. \n * \n * Using this strategy means that when end users navigate between pages in your project, or refresh their browsers, they will still be working with the same run. \n * \n * However, if they are idle for longer than your project's **Model Session Timeout** (configured in your project's [Settings](../../../updating_your_settings/)), then their run is persisted; the next time they interact with the project, they will get a new run. (See more background on [Run Persistence](../../../run_persistence/).)\n * \n * This strategy is useful for multi-page projects where end users play through a simulation in one sitting, stepping through the model sequentially (for example, a Vensim model that uses the `step` operation) or calling specific functions until the model is \"complete.\" However, you will need to guarantee that your end users will remain engaged with the project from beginning to end &mdash; or at least, that if they are idle for longer than the **Model Session Timeout**, it is okay for them to start the project from scratch (with an uninitialized model). \n * \n * Specifically, the strategy is:\n *\n * * Check the `sessionKey` cookie.\n *   * This cookie is set by the [Run Manager](../run-manager/) and configurable through its options.\n *   * If the cookie exists, check whether the run is in memory or only persisted in the database. \n *      * If the run is in memory, use the run.\n *      * If the run is only persisted (and not still in memory), create a new run for this end user.\n *      * If the cookie does not exist, create a new run for this end user.\n *\n * @deprecated The run-service now sets a header to automatically bring back runs into memory\n */\n\n'use strict';\nvar classFrom = require('../../../util/inherit');\nvar ConditionalStrategy = require('../conditional-creation-strategy');\n\nvar __super = ConditionalStrategy.prototype;\n\nvar Strategy = classFrom(ConditionalStrategy, {\n    constructor: function (options) {\n        __super.constructor.call(this, this.createIf, options);\n        console.warn('This strategy is deprecated; the run-service now sets a header to automatically bring back runs into memory');\n    },\n\n    createIf: function (run, headers) {\n        return headers.getResponseHeader('pragma') === 'persistent';\n    }\n});\n\nmodule.exports = Strategy;\n","/**\n * ### Working with Run Strategies\n *\n * You can access a list of available strategies using `F.manager.RunManager.strategies.list`. You can also ask for a particular strategy by name.\n *\n * If you decide to [create your own run strategy](#create-your-own), you can register your strategy. Registering your strategy means that:\n *\n * * You can pass the strategy by name to a Run Manager (as opposed to passing the strategy function): `new F.manager.RunManager({ strategy: 'mynewname'})`.\n * * You can pass configuration options to your strategy.\n * * You can specify whether or not your strategy requires authorization (a valid user session) to work.\n */\n\n\nvar list = {\n    'conditional-creation': require('./conditional-creation-strategy'),\n    'new-if-initialized': require('./deprecated/new-if-initialized-strategy'), //deprecated\n    'new-if-persisted': require('./deprecated/new-if-persisted-strategy'), //deprecated\n\n    none: require('./none-strategy'),\n\n    multiplayer: require('./multiplayer-strategy'),\n    'reuse-never': require('./reuse-never'),\n    'reuse-per-session': require('./reuse-per-session'),\n    'reuse-across-sessions': require('./reuse-across-sessions'),\n    'reuse-last-initialized': require('./reuse-last-initialized'),\n};\n\n//Add back older aliases\nlist['always-new'] = list['reuse-never'];\nlist['new-if-missing'] = list['reuse-per-session'];\nlist['persistent-single-player'] = list['reuse-across-sessions'];\n\n\nmodule.exports = {\n    /**\n     * List of available strategies. Within this object, each key is the strategy name and the associated value is the strategy constructor.\n     * @type {Object} \n     */\n    list: list,\n\n    /**\n     * Gets strategy by name.\n     *\n     * **Example**\n     *\n     *      var reuseStrat = F.manager.RunManager.strategies.byName('reuse-across-sessions');\n     *      // shows strategy function\n     *      console.log('reuseStrat = ', reuseStrat);\n     *      // create a new run manager using this strategy\n     *      var rm = new F.manager.RunManager({strategy: reuseStrat, run: { model: 'model.vmf'} });\n     *\n     * **Parameters**\n     * @param  {String} strategyName Name of strategy to get.\n     * @return {Function} Strategy function.\n     */\n    byName: function (strategyName) {\n        return list[strategyName];\n    },\n\n    getBestStrategy: function (options) {\n        var strategy = options.strategy;\n        if (!strategy) {\n            if (options.strategyOptions && options.strategyOptions.initOperation) {\n                strategy = 'reuse-last-initialized';\n            } else {\n                strategy = 'reuse-per-session';\n            }\n        }\n\n        if (strategy.getRun) {\n            return strategy;\n        }\n        var StrategyCtor = typeof strategy === 'function' ? strategy : this.byName(strategy);\n        if (!StrategyCtor) {\n            throw new Error('Specified run creation strategy was invalid:', strategy);\n        }\n\n        var strategyInstance = new StrategyCtor(options);\n        if (!strategyInstance.getRun || !strategyInstance.reset) {\n            throw new Error('All strategies should implement a `getRun` and `reset` interface', options.strategy);\n        }\n        strategyInstance.requiresAuth = StrategyCtor.requiresAuth;\n\n        return strategyInstance;\n    },\n\n    /**\n     * Adds a new strategy.\n     *\n     * **Example**\n     *\n     *      // this \"favorite run\" strategy always returns the same run, no matter what\n     *      // (not a useful strategy, except as an example)\n     *      F.manager.RunManager.strategies.register(\n     *          'favRun', \n     *          function() { \n     *              return { getRun: function() { return '0000015a4cd1700209cd0a7d207f44bac289'; },\n     *                      reset: function() { return '0000015a4cd1700209cd0a7d207f44bac289'; } \n     *              } \n     *          }, \n     *          { requiresAuth: true }\n     *      );\n     *      \n     *      var rm = new F.manager.RunManager({strategy: 'favRun', run: { model: 'model.vmf'} });\n     *\n     * **Parameters**\n     * @param  {String} name Name for strategy. This string can then be passed to a Run Manager as `new F.manager.RunManager({ strategy: 'mynewname'})`.\n     * @param  {Function} strategy The strategy constructor. Will be called with `new` on Run Manager initialization.\n     * @param  {Object} options  Options for strategy.\n     * @param  {Boolean} options.requiresAuth Specify if the strategy requires a valid user session to work.\n     */\n    register: function (name, strategy, options) {\n        strategy.options = options;\n        list[name] = strategy;\n    }\n};","/**\n * The `multiplayer` strategy is for use with [multiplayer worlds](../../../glossary/#world). It checks the current world for this end user, and always returns the current run for that world. This is equivalent to calling `getCurrentWorldForUser()` and then `getCurrentRunId()` from the [World API Adapater](../world-api-adapter/). If you use the [World Manager](../world-manager/), you are automatically using this strategy.\n * \n * Using this strategy means that end users in projects with multiplayer worlds always see the most current world and run. This ensures that they are in sync with the other end users sharing their world and run. In turn, this allows for competitive or collaborative multiplayer projects.\n */\n'use strict';\n\nvar classFrom = require('../../util/inherit');\n\nvar IdentityStrategy = require('./none-strategy');\nvar WorldApiAdapter = require('../../service/world-api-adapter');\n\nvar defaults = {};\n\nvar Strategy = classFrom(IdentityStrategy, {\n    constructor: function (options) {\n        this.options = $.extend(true, {}, defaults, options);\n        this.worldApi = new WorldApiAdapter(this.options.run);\n    },\n\n    reset: function (runService, session, options) {\n        var curUserId = session.userId;\n        var curGroupName = session.groupName;\n\n        return this.worldApi\n            .getCurrentWorldForUser(curUserId, curGroupName)\n            .then(function (world) {\n                return this.worldApi.newRunForWorld(world.id, options).then(function (runid) {\n                    return {\n                        id: runid\n                    };\n                });\n            }.bind(this));\n    },\n\n    getRun: function (runService, session) {\n        var curUserId = session.userId;\n        var curGroupName = session.groupName;\n        var worldApi = this.worldApi;\n        var model = this.options.model;\n        var me = this;\n        var dtd = $.Deferred();\n\n        if (!curUserId) {\n            return dtd.reject({ statusCode: 400, error: 'We need an authenticated user to join a multiplayer world. (ERR: no userId in session)' }, session).promise();\n        }\n\n        var loadRunFromWorld = function (world) {\n            if (!world) {\n                return dtd.reject({ statusCode: 404, error: 'The user is not in any world.' }, { options: me.options, session: session });\n            }\n            return worldApi.getCurrentRunId({ model: model, filter: world.id })\n                .then(function (id) {\n                    return runService.load(id);\n                })\n                .then(dtd.resolve)\n                .fail(dtd.reject);\n        };\n\n        var serverError = function (error) {\n            // is this possible?\n            return dtd.reject(error, session, me.options);\n        };\n\n        this.worldApi\n            .getCurrentWorldForUser(curUserId, curGroupName)\n            .then(loadRunFromWorld)\n            .fail(serverError);\n\n        return dtd.promise();\n    },\n\n});\n\nmodule.exports = Strategy;\n","/**\n * The `none` strategy never returns a run or tries to create a new run. It simply returns the contents of the current [Run Service instance](../run-api-service/).\n * \n * This strategy is useful if you want to manually decide how to create your own runs and don't want any automatic assistance.\n */\n\n'use strict';\n\nvar classFrom = require('../../util/inherit');\nvar Base = {};\n\n// Interface that all strategies need to implement\nmodule.exports = classFrom(Base, {\n    constructor: function (options) {\n\n    },\n\n    reset: function () {\n        // return a newly created run\n        return $.Deferred().resolve().promise();\n    },\n\n    getRun: function (runService) {\n        // return a usable run\n        return $.Deferred().resolve(runService).promise();\n    }\n});\n","/**\n * The `reuse-across-sessions` strategy returns the latest (most recent) run for this user, whether it is in memory or not. If there are no runs for this user, it creates a new one.\n *\n * This strategy is useful if end users are using your project for an extended period of time, possibly over several sessions. This is most common in cases where a user of your project executes the model step by step (as opposed to a project where the model is executed completely, for example, a Vensim model that is immediately stepped to the end).\n *\n * Specifically, the strategy is:\n * \n * * Check if there are any runs for this end user.\n *     * If there are no runs (either in memory or in the database), create a new one.\n *     * If there are runs, take the latest (most recent) one.\n *\n * @name persistent-single-player\n */\n\n'use strict';\n\nvar classFrom = require('../../util/inherit');\nvar IdentityStrategy = require('./none-strategy');\nvar injectFiltersFromSession = require('../strategy-utils').injectFiltersFromSession;\nvar injectScopeFromSession = require('../strategy-utils').injectScopeFromSession;\n\nvar defaults = {\n    /**\n     * (Optional) Additional criteria to use while selecting the last run\n     * @type {Object}\n     */\n    filter: {},\n};\n\nvar Strategy = classFrom(IdentityStrategy, {\n    constructor: function Strategy(options) {\n        var strategyOptions = options ? options.strategyOptions : {};\n        this.options = $.extend(true, {}, defaults, strategyOptions);\n    },\n\n    reset: function (runService, userSession, options) {\n        var opt = injectScopeFromSession(runService.getCurrentConfig(), userSession);\n        return runService\n            .create(opt, options)\n            .then(function (run) {\n                run.freshlyCreated = true;\n                return run;\n            });\n    },\n\n    getRun: function (runService, userSession, runSession, options) {\n        var filter = injectFiltersFromSession(this.options.filter, userSession);\n        var me = this;\n        return runService.query(filter, { \n            // startrecord: 0, //TODO: Uncomment when EPICENTER-2569 is fixed\n            // endrecord: 0,\n            sort: 'created', \n            direction: 'desc'\n        }).then(function (runs) {\n            if (!runs.length) {\n                return me.reset(runService, userSession, options);\n            }\n            return runs[0];\n        });\n    }\n});\n\nmodule.exports = Strategy;\n","/**\n * The `reuse-last-initialized` strategy looks for the most recent run that matches particular criteria; if it cannot find one, it creates a new run and immediately executes a set of \"initialization\" operations. \n *\n * This strategy is useful if you have a time-based model and always want the run you're operating on to start at a particular step. For example:\n *\n *      var rm = new F.manager.RunManager({\n *          strategy: 'reuse-last-initialized',\n *          strategyOptions: {\n *              initOperation: [{ step: 10 }]\n *          }\n *      });\n * \n * This strategy is also useful if you have a custom initialization function in your model, and want to make sure it's always executed for new runs.\n *\n * Specifically, the strategy is:\n *\n * * Look for the most recent run that matches the (optional) `flag` criteria\n * * If there are no runs that match the `flag` criteria, create a new run. Immediately \"initialize\" this new run by:\n *     *  Calling the model operation(s) specified in the `initOperation` array.\n *     *  Optionally, setting a `flag` in the run.\n *\n */\n\n'use strict';\nvar classFrom = require('../../util/inherit');\nvar injectFiltersFromSession = require('../strategy-utils').injectFiltersFromSession;\nvar injectScopeFromSession = require('../strategy-utils').injectScopeFromSession;\n\nvar Base = {};\n\nvar defaults = {\n    /**\n     * Operations to execute in the model for initialization to be considered complete.\n     * @type {Array} Can be in any of the formats [Run Service's `serial()`](../run-api-service/#serial) supports.\n     */\n    initOperation: [],\n\n    /**\n     * (Optional) Flag to set in run after initialization operations are run. You typically would not override this unless you needed to set additional properties as well.\n     * @type {Object}\n     */\n    flag: null,\n};\nmodule.exports = classFrom(Base, {\n    constructor: function (options) {\n        var strategyOptions = options ? options.strategyOptions : {};\n        this.options = $.extend(true, {}, defaults, strategyOptions);\n        if (!this.options.initOperation || !this.options.initOperation.length) {\n            throw new Error('Specifying an init function is required for this strategy');\n        }\n        if (!this.options.flag) {\n            this.options.flag = {\n                isInitComplete: true\n            };\n        }\n    },\n\n    reset: function (runService, userSession, options) {\n        var opt = injectScopeFromSession(runService.getCurrentConfig(), userSession);\n        var me = this;\n        return runService.create(opt, options).then(function (createResponse) {\n            return runService.serial([].concat(me.options.initOperation)).then(function () {\n                return createResponse;\n            });\n        }).then(function (createResponse) {\n            return runService.save(me.options.flag).then(function (patchResponse) {\n                return $.extend(true, {}, createResponse, patchResponse);\n            });\n        });\n    },\n\n    getRun: function (runService, userSession, runSession, options) {\n        var sessionFilter = injectFiltersFromSession(this.options.flag, userSession);\n        var runopts = runService.getCurrentConfig();\n        var filter = $.extend(true, { trashed: false }, sessionFilter, { model: runopts.model });\n        var me = this;\n        return runService.query(filter, { \n            // startrecord: 0,  //TODO: Uncomment when EPICENTER-2569 is fixed\n            // endrecord: 0,\n            sort: 'created', \n            direction: 'desc'\n        }).then(function (runs) {\n            if (!runs.length) {\n                return me.reset(runService, userSession, options);\n            }\n            return runs[0];\n        });\n    }\n});","/**\n * The `reuse-never` strategy always creates a new run for this end user irrespective of current state. This is equivalent to calling `F.service.Run.create()` from the [Run Service](../run-api-service/) every time. \n * \n * This strategy means that every time your end users refresh their browsers, they get a new run. \n * \n * This strategy can be useful for basic, single-page projects. This strategy is also useful for prototyping or project development: it creates a new run each time you refresh the page, and you can easily check the outputs of the model. However, typically you will use one of the other strategies for a production project.\n *\n * @name always-new\n */\n\n'use strict';\n\nvar classFrom = require('../../util/inherit');\nvar ConditionalStrategy = require('./conditional-creation-strategy');\n\nvar __super = ConditionalStrategy.prototype;\n\nvar Strategy = classFrom(ConditionalStrategy, {\n    constructor: function (options) {\n        __super.constructor.call(this, this.createIf, options);\n    },\n\n    createIf: function (run, headers) {\n        // always create a new run!\n        return true;\n    }\n});\n\nmodule.exports = Strategy;\n","/**\n * The `reuse-per-session` strategy creates a new run when the current one is not in the browser cookie.\n * \n * Using this strategy means that when end users navigate between pages in your project, or refresh their browsers, they will still be working with the same run. However, if end users log out and return to the project at a later date, a new run is created.\n *\n * This strategy is useful if your project is structured such that immediately after a run is created, the model is executed completely (for example, a Vensim model that is stepped to the end as soon as it is created). In contrast, if end users play with your project for an extended period of time, executing the model step by step, the `reuse-across-sessions` strategy is probably a better choice (it allows end users to pick up where they left off, rather than starting from scratch each browser session).\n * \n * Specifically, the strategy is:\n *\n * * Check the `sessionKey` cookie.\n *     * This cookie is set by the [Run Manager](../run-manager/) and configurable through its options. \n *     * If the cookie exists, use the run id stored there. \n *     * If the cookie does not exist, create a new run for this end user.\n *\n * @name new-if-missing\n */\n\n'use strict';\n\nvar classFrom = require('../../util/inherit');\nvar ConditionalStrategy = require('./conditional-creation-strategy');\n\nvar __super = ConditionalStrategy.prototype;\n\n/*\n*  create a new run only if nothing is stored in the cookie\n*  this is useful for baseRuns.\n*/\nvar Strategy = classFrom(ConditionalStrategy, {\n    constructor: function (options) {\n        __super.constructor.call(this, this.createIf, options);\n    },\n\n    createIf: function (run, headers) {\n        // if we are here, it means that the run exists... so we don't need a new one\n        return false;\n    }\n});\n\nmodule.exports = Strategy;\n","/**\n * ## Saved Runs Manager\n *\n * The Saved Runs Manager is a specific type of [Run Manager](../../run-manager/) which provides access to a list of runs (rather than just one run). It also provides utility functions for dealing with multiple runs (e.g. saving, deleting, listing).\n *\n * An instance of a Saved Runs Manager is included automatically in every instance of a [Scenario Manager](../), and is accessible from the Scenario Manager at `.savedRuns`. See [more information](../#properties) on using `.savedRuns` within the Scenario Manager.\n *\n */\n'use strict';\n\nvar RunService = require('../service/run-api-service');\nvar SessionManager = require('../store/session-manager');\n\nvar injectFiltersFromSession = require('./strategy-utils').injectFiltersFromSession;\n\nvar SavedRunsManager = function (config) {\n    var defaults = {\n        /**\n         * If set, will only pull runs from current group. Defaults to `true`.\n         * @type {Boolean}\n         */\n        scopeByGroup: true,\n\n        /**\n         * If set, will only pull runs from current user. Defaults to `true`.\n         *\n         * For multiplayer run comparison projects, set this to false so that all end users in a group can view the shared set of saved runs.\n         * @type {Boolean}\n         */\n        scopeByUser: true,\n    };\n\n    this.sessionManager = new SessionManager();\n\n    var options = $.extend(true, {}, defaults, config);\n    if (options.run) {\n        if (options.run instanceof RunService) {\n            this.runService = options.run;\n        } else {\n            this.runService = new RunService(options.run);\n        }\n        this.options = options;\n    } else {\n        throw new Error('No run options passed to SavedRunsManager');\n    }\n};\n\nSavedRunsManager.prototype = {\n    /**\n     * Marks a run as saved. \n     *\n     * Note that while any run can be saved, only runs which also match the configuration options `scopeByGroup` and `scopeByUser` are returned by the `getRuns()` method.\n     *\n     * **Example**\n     *\n     *      var sm = new F.manager.ScenarioManager();\n     *      sm.savedRuns.save('0000015a4cd1700209cd0a7d207f44bac289');\n     *\n     * @param  {String|RunService} run Run to save. Pass in either the run id, as a string, or the [Run Service](../../run-api-service/).\n     * @param  {Object} otherFields (Optional) Any other meta-data to save with the run.\n     * @return {Promise}\n     */\n    save: function (run, otherFields) {\n        var param = $.extend(true, {}, otherFields, { saved: true, trashed: false });\n        return this.mark(run, param);\n    },\n    /**\n     * Marks a run as removed; the inverse of marking as saved.\n     *\n     * **Example**\n     *\n     *      var sm = new F.manager.ScenarioManager();\n     *      sm.savedRuns.remove('0000015a4cd1700209cd0a7d207f44bac289');\n     *\n     * @param  {String|RunService} run Run to remove. Pass in either the run id, as a string, or the [Run Service](../../run-api-service/).\n     * @param  {Object} otherFields (Optional) any other meta-data to save with the run.\n     * @return {Promise}\n     */\n    remove: function (run, otherFields) {\n        var param = $.extend(true, {}, otherFields, { saved: false, trashed: true });\n        return this.mark(run, param);\n    },\n\n\n    /**\n     * Sets additional fields on a run. This is a convenience method for [RunService#save](../../run-api-service/#save).\n     *\n     * **Example**\n     *\n     *      var sm = new F.manager.ScenarioManager();\n     *      sm.savedRuns.mark('0000015a4cd1700209cd0a7d207f44bac289', \n     *          { 'myRunName': 'sample policy decisions' });\n     *\n     * @param  {String|RunService} run  Run to operate on. Pass in either the run id, as a string, or the [Run Service](../../run-api-service/).\n     * @param  {Object} toMark Fields to set, as name : value pairs.\n     * @return {Promise}\n     */\n    mark: function (run, toMark) {\n        var rs;\n        var existingOptions = this.runService.getCurrentConfig();\n        if (run instanceof RunService) {\n            rs = run;\n        } else if (run && (typeof run === 'string')) {\n            rs = new RunService($.extend(true, {}, existingOptions, { id: run, autoRestore: false }));\n        } else if ($.isArray(run)) {\n            var me = this;\n            var proms = run.map(function (r) {\n                return me.mark(r, toMark);\n            });\n            return $.when.apply(null, proms);\n        } else {\n            throw new Error('Invalid run object provided');\n        }\n        return rs.save(toMark);\n    },\n\n    /**\n     * Returns a list of saved runs.\n     *\n     * **Example**\n     *\n     *      var sm = new F.manager.ScenarioManager();\n     *      sm.savedRuns.getRuns().then(function (runs) {\n     *          for (var i=0; i<runs.length; i++) {\n     *              console.log('run id of saved run: ', runs[i].id);\n     *          }\n     *      });\n     *\n     * @param  {Array} variables (Optional) If provided, in the returned list of runs, each run will have a `.variables` property with these set.\n     * @param  {Object} filter    (Optional) Any filters to apply while fetching the run. See [RunService#filter](../../run-api-service/#filter) for details.\n     * @param  {Object} modifiers (Optional) Use for paging/sorting etc. See [RunService#filter](../../run-api-service/#filter) for details.\n     * @return {Promise}\n     */\n    getRuns: function (variables, filter, modifiers) {\n        var session = this.sessionManager.getSession(this.runService.getCurrentConfig());\n\n        var runopts = this.runService.getCurrentConfig();\n        var scopedFilter = injectFiltersFromSession($.extend(true, {}, {\n            saved: true, \n            trashed: false,\n            model: runopts.model,\n        }, filter), session, this.options);\n\n        var opModifiers = $.extend(true, {}, {\n            sort: 'created',\n            direction: 'asc',\n        }, modifiers);\n        if (variables) {\n            opModifiers.include = [].concat(variables);\n        }\n        return this.runService.query(scopedFilter, opModifiers);\n    }\n};\nmodule.exports = SavedRunsManager;\n","/**\n* ## Scenario Manager\n*\n* In some projects, often called \"turn-by-turn\" projects, end users advance through the project's model step-by-step, working either individually or together to make decisions at each step. \n*\n* In other projects, often called \"run comparison\" or \"scenario comparison\" projects, end users set some initial decisions, then simulate the model to its end. Typically end users will do this several times, creating several runs, and compare the results. \n*\n* The Scenario Manager makes it easy to create these \"run comparison\" projects. Each Scenario Manager allows you to compare the results of several runs. This is mostly useful for time-based models; by default, you can use the Scenario Manager with [Vensim](../../../model_code/vensim/), [Powersim](../../../model_code/powersim/), and [SimLang](../../../model_code/forio_simlang). (You can use the Scenario Manager with other languages as well, by using the Scenario Manager's [configuration options](#configuration-options) to change the `advanceOperation`.)\n*\n* The Scenario Manager can be thought of as a collection of [Run Managers](../run-manager/) with pre-configured [strategies](../strategies/). Just as the Run Manager provides use case -based abstractions and utilities for managing the [Run Service](../run-api-service/), the Scenario Manager does the same for the Run Manager.\n*\n* There are typically three components to building a run comparison:\n*\n* * A `current` run in which to make decisions; this is defined as a run that hasn't been advanced yet, and so can be used to set initial decisions. The current run maintains state across different sessions.\n* * A list of `saved` runs, that is, all runs that you want to use for comparisons.\n* * A `baseline` run to compare against; this is defined as a run \"advanced to the end\" of your model using just the model defaults. Comparing against a baseline run is optional; you can [configure](#configuration-options) the Scenario Manager to not include one.\n*\n* To satisfy these needs a Scenario Manager instance has three Run Managers: [baseline](./baseline/), [current](./current/), and [savedRuns](./saved/).\n*\n* ### Using the Scenario Manager to create a run comparison project\n*\n* To use the Scenario Manager, instantiate it, then access its Run Managers as needed to create your project's user interface:\n*\n* **Example**\n*\n*       var sm = new F.manager.ScenarioManager({\n*           run: {\n*               model: 'mymodel.vmf'\n*           }\n*       });\n*\n*       // The current is an instance of a Run Manager,\n*       // with a strategy which picks up the most recent unsaved run.\n*       // It is typically used to store the decisions being made by the end user. \n*       var currentRM = sm.current;\n*\n*       // The Run Manager operation, which retrieves the current run.\n*       sm.current.getRun();\n*       // The Run Manager operation, which resets the decisions made on the current run.\n*       sm.current.reset();\n*       // A special method on the current run,\n*       // which clones the current run, then advances and saves this clone\n*       // (it becomes part of the saved runs list).\n*       // The current run is unchanged and can continue to be used\n*       // to store decisions being made by the end user.\n*       sm.current.saveAndAdvance();\n*\n*       // The savedRuns is an instance of a Saved Runs Manager \n*       // (itself a variant of a Run Manager).\n*       // It is typically displayed in the project's UI as part of a run comparison table or chart.\n*       var savedRM = sm.savedRuns;\n*       // Mark a run as saved, adding it to the set of saved runs.\n*       sm.savedRuns.save(run);\n*       // Mark a run as removed, removing it from the set of saved runs.\n*       sm.savedRuns.remove(run);\n*       // List the saved runs, optionally including some specific model variables for each.\n*       sm.savedRuns.getRuns();\n*\n*       // The baseline is an instance of a Run Manager,\n*       // with a strategy which locates the most recent baseline run\n*       // (that is, flagged as `saved` and not `trashed`), or creates a new one.\n*       // It is typically displayed in the project's UI as part of a run comparison table or chart.\n*       var baselineRM = sm.baseline;\n*\n*       // The Run Manager operation, which retrieves the baseline run.\n*       sm.baseline.getRun();\n*       // The Run Manager operation, which resets the baseline run.\n*       // Useful if the model has changed since the baseline run was created.\n*       sm.baseline.reset(); \n*/\n\n'use strict';\n\n// See integration-test-scenario-manager for usage examples\nvar RunManager = require('./run-manager');\nvar SavedRunsManager = require('./saved-runs-manager');\nvar strategyUtils = require('./strategy-utils');\nvar rutil = require('../util/run-util');\n\nvar NoneStrategy = require('./run-strategies/none-strategy');\n\nvar StateService = require('../service/state-api-adapter');\nvar RunService = require('../service/run-api-service');\n\nvar BaselineStrategy = require('./scenario-strategies/baseline-strategy');\nvar LastUnsavedStrategy = require('./scenario-strategies/reuse-last-unsaved');\n\nvar defaults = {\n    /**\n     * Operations to perform on each run to indicate that the run is complete. Operations are executed [serially](../run-api-service/#serial). Defaults to calling the model operation `stepTo('end')`, which advances Vensim, Powersim, and SimLang models to the end. \n     * @type {Array}\n     */\n    advanceOperation: [{ name: 'stepTo', params: ['end'] }],\n\n    /**\n     * Additional options to pass through to run creation (for e.g., `files`, etc.). Defaults to empty object.\n     * @type {Object}\n     */\n    run: {},\n\n    /**\n     * Whether or not to include a baseline run in this Scenario Manager. Defaults to `true`.\n     * @type {Boolean}\n     */\n    includeBaseLine: true,\n\n    /**\n     * Additional configuration for the `baseline` run. \n     *\n     * * `baseline.runName`: Name of the baseline run. Defaults to 'Baseline'. \n     * * `baseline.run`: Additional options to pass through to run creation, specifically for the baseline run. These will override any options provided under `run`. Defaults to empty object. \n     * @type {Object}\n     */\n    baseline: {\n        runName: 'Baseline',\n        run: {}\n    },\n\n    /**\n     * Additional configuration for the `current` run. \n     *\n     * * `current.run`: Additional options to pass through to run creation, specifically for the current run. These will override any options provided under `run`. Defaults to empty object.\n     * @type {Object}\n     */\n    current: {\n        run: {}\n    },\n\n    /**\n     * Options to pass through to the `savedRuns` list. See the [Saved Runs Manager](./saved/) for complete description of available options. Defaults to empty object.\n     * @type {Object}\n     */\n    savedRuns: {}\n};\n\nfunction cookieNameFromOptions(prefix, config) {\n    var key = ['account', 'project', 'model'].reduce(function (accum, key) {\n        return config[key] ? accum + '-' + config[key] : accum; \n    }, prefix);\n    return key;\n}\n\nfunction ScenarioManager(config) {\n    var opts = $.extend(true, {}, defaults, config);\n    if (config && config.advanceOperation) {\n        opts.advanceOperation = config.advanceOperation; //jquery.extend does a poor job trying to merge arrays\n    }\n\n    var BaselineStrategyToUse = opts.includeBaseLine ? BaselineStrategy : NoneStrategy;\n    /**\n     * A [Run Manager](../run-manager/) instance containing a 'baseline' run to compare against; this is defined as a run \"advanced to the end\" of your model using just the model defaults. By default the \"advance\" operation is assumed to be `stepTo: end`, which works for time-based models in [Vensim](../../../model_code/vensim/), [Powersim](../../../model_code/powersim/), and [SimLang](../../../model_code/forio_simlang). If you're using a different language, or need to change this, just pass in a different `advanceOperation` option while creating the Scenario Manager. The baseline run is typically displayed in the project's UI as part of a run comparison table or chart.\n     * @return {RunManager}\n     */\n    this.baseline = new RunManager({\n        strategy: BaselineStrategyToUse,\n        sessionKey: cookieNameFromOptions.bind(null, 'sm-baseline-run'),\n        run: strategyUtils.mergeRunOptions(opts.run, opts.baseline.run),\n        strategyOptions: {\n            baselineName: opts.baseline.runName,\n            initOperation: opts.advanceOperation\n        }\n    });\n\n    /**\n     * A [SavedRunsManager](../saved-runs-manager/) instance containing a list of saved runs, that is, all runs that you want to use for comparisons. The saved runs are typically displayed in the project's UI as part of a run comparison table or chart.\n     * @return {SavedRunsManager}\n     */\n    this.savedRuns = new SavedRunsManager($.extend(true, {}, {\n        run: opts.run,\n    }, opts.savedRuns));\n\n    var origGetRuns = this.savedRuns.getRuns;\n    var me = this;\n    this.savedRuns.getRuns = function () {\n        var args = Array.apply(null, arguments);\n        return me.baseline.getRun().then(function () {\n            return origGetRuns.apply(me.savedRuns, args);\n        });\n    };\n\n    /**\n     * A [Run Manager](../run-manager/) instance containing a 'current' run; this is defined as a run that hasn't been advanced yet, and so can be used to set initial decisions. The current run is typically used to store the decisions being made by the end user.\n     * @return {RunManager}\n     */\n    this.current = new RunManager({\n        strategy: LastUnsavedStrategy,\n        sessionKey: cookieNameFromOptions.bind(null, 'sm-current-run'),\n        run: strategyUtils.mergeRunOptions(opts.run, opts.current.run)\n    });\n\n    /**\n     * Clones the current run, advances this clone by calling the `advanceOperation`, and saves the cloned run (it becomes part of the `savedRuns` list). Additionally, adds any provided metadata to the cloned run; typically used for naming the run. The current run is unchanged and can continue to be used to store decisions being made by the end user.\n     *\n     * Available only for the Scenario Manager's `current` property (Run Manager). \n     *\n     * **Example**\n     *\n     *      var sm = new F.manager.ScenarioManager();\n     *      sm.current.saveAndAdvance({'myRunName': 'sample policy decisions'});\n     *\n     * **Parameters**\n     * @param  {Object} metadata   Metadata to save, for example, the run name.\n     * @return {Promise}\n     */\n    this.current.saveAndAdvance = function (metadata) {\n        function clone(run) {\n            var sa = new StateService();\n            var advanceOpns = rutil.normalizeOperations(opts.advanceOperation); \n            //run i'm cloning shouldn't have the advance operations there by default, but just in case\n            return sa.clone({ runId: run.id, exclude: advanceOpns.ops }).then(function (response) {\n                var rs = new RunService(me.current.run.getCurrentConfig());\n                return rs.load(response.run);\n            });\n        }\n        function markSaved(run) {\n            return me.savedRuns.save(run.id, metadata).then(function (savedResponse) {\n                return $.extend(true, {}, run, savedResponse);\n            });\n        }\n        function advance(run) {\n            var rs = new RunService(run);\n            return rs.serial(opts.advanceOperation).then(function () {\n                return run;\n            });\n        }\n        return me.current\n                .getRun()\n                .then(clone)\n                .then(advance)\n                .then(markSaved);\n    };\n}\n\nmodule.exports = ScenarioManager;\n","/**\n * ## Baseline\n *\n * An instance of a [Run Manager](../../run-manager/) with a baseline strategy is included automatically in every instance of a [Scenario Manager](../), and is accessible from the Scenario Manager at `.baseline`.\n *\n * A baseline is defined as a run \"advanced to the end\" using just the model defaults. The baseline run is typically displayed in the project's UI as part of a run comparison table or chart.\n *\n * The `baseline` strategy looks for the most recent run named as 'Baseline' (or named as specified in the `baseline.runName` [configuration option of the Scenario Manager](../#configuration-options)) that is flagged as `saved` and not `trashed`. If the strategy cannot find such a run, it creates a new run and immediately executes a set of initialization operations. \n *\n * Comparing against a baseline run is optional in a Scenario Manager; you can [configure](../#configuration-options) your Scenario Manager to not include one. See [more information](../#properties) on using `.baseline` within the Scenario Manager.\n *\n * See also: [additional information on run strategies](../../strategies/).\n *\n */\n\n'use strict';\n\nvar ReuseinitStrategy = require('../run-strategies/reuse-last-initialized');\n\nmodule.exports = function (options) {\n    var defaults = {\n        baselineName: 'Baseline',\n        initOperation: [{ stepTo: 'end' }]\n    };\n    var strategyOptions = options ? options.strategyOptions : {};\n    var opts = $.extend({}, defaults, strategyOptions);\n    return new ReuseinitStrategy({\n        strategyOptions: {\n            initOperation: opts.initOperation,\n            flag: {\n                saved: true,\n                trashed: false,\n                name: opts.baselineName\n            }\n        }\n    });\n};\n","/**\n * ## Current (reuse-last-unsaved)\n *\n * An instance of a [Run Manager](../../run-manager/) with this strategy is included automatically in every instance of a [Scenario Manager](../), and is accessible from the Scenario Manager at `.current`.\n *\n * The `reuse-last-unsaved` strategy returns the most recent run that is not flagged as `trashed` and also not flagged as `saved`.\n * \n * Using this strategy means that end users continue working with the most recent run that has not been explicitly flagged by the project. However, if there are no runs for this end user, a new run is created.\n * \n * Specifically, the strategy is:\n *\n * * Check the `saved` and `trashed` fields of the run to determine if the run has been explicitly saved or explicitly flagged as no longer useful.\n *     * Return the most recent run that is not `trashed` and also not `saved`.\n *     * If there are no runs, create a new run for this end user. \n *\n * See [more information](../#properties) on using `.current` within the Scenario Manager.\n *\n * See also: [additional information on run strategies](../../strategies/).\n */\n\n'use strict';\nvar classFrom = require('../../util/inherit');\nvar injectFiltersFromSession = require('../strategy-utils').injectFiltersFromSession;\nvar injectScopeFromSession = require('../strategy-utils').injectScopeFromSession;\n\nvar Base = {};\n\n//TODO: Make a more generic version of this called 'reuse-by-matching-filter';\nmodule.exports = classFrom(Base, {\n    constructor: function (options) {\n        var strategyOptions = options ? options.strategyOptions : {};\n        this.options = strategyOptions;\n    },\n\n    reset: function (runService, userSession, options) {\n        var opt = injectScopeFromSession(runService.getCurrentConfig(), userSession);\n        return runService.create(opt, options).then(function (createResponse) {\n            return $.extend(true, {}, createResponse, { freshlyCreated: true });\n        });\n    },\n\n    getRun: function (runService, userSession, opts) {\n        var runopts = runService.getCurrentConfig();\n        var filter = injectFiltersFromSession({ \n            saved: false,\n            trashed: false, //TODO: change to '!=true' once EPICENTER-2500 is fixed,\n            model: runopts.model,\n        }, userSession);\n        var me = this;\n        var outputModifiers = { \n            // startrecord: 0,  //TODO: Uncomment when EPICENTER-2569 is fixed\n            // endrecord: 0,\n            sort: 'created', \n            direction: 'desc'\n        };\n        return runService.query(filter, outputModifiers).then(function (runs) {\n            if (!runs.length) {\n                return me.reset(runService, userSession);\n            }\n            return runs[0];\n        });\n    }\n}, { requiresAuth: false });","'use strict';\n\n\nmodule.exports = {\n    reset: function (params, options, manager) {\n        return manager.reset(options);\n    }\n};\n","'use strict';\n\nvar RunService = require('../service/run-api-service');\n\nmodule.exports = {\n    mergeRunOptions: function (run, options) {\n        if (run instanceof RunService) {\n            run.updateConfig(options);\n            return run;\n        } \n        return $.extend(true, {}, run, options);\n    },\n\n    injectFiltersFromSession: function (currentFilter, session, options) {\n        var defaults = {\n            scopeByGroup: true,\n            scopeByUser: true,\n        };\n        var opts = $.extend(true, {}, defaults, options);\n\n        var filter = $.extend(true, {}, currentFilter);\n        if (opts.scopeByGroup && session && session.groupName) {\n            filter['scope.group'] = session.groupName;\n        }\n        if (opts.scopeByUser && session && session.userId) {\n            filter['user.id'] = session.userId;\n        }\n        return filter;\n    },\n\n    injectScopeFromSession: function (currentParams, session) {\n        var group = session && session.groupName;\n        var params = $.extend(true, {}, currentParams);\n        if (group) {\n            $.extend(params, {\n                scope: { group: group }\n            });\n        }\n        return params;\n    }\n};","/**\n* ## World Manager\n*\n* As discussed under the [World API Adapter](../world-api-adapter/), a [run](../../../glossary/#run) is a collection of end user interactions with a project and its model. For building multiplayer simulations you typically want multiple end users to share the same set of interactions, and work within a common state. Epicenter allows you to create \"worlds\" to handle such cases.\n*\n* The World Manager provides an easy way to track and access the current world and run for particular end users. It is typically used in pages that end users will interact with. (The related [World API Adapter](../world-api-adapter/) handles creating multiplayer worlds, and adding and removing end users and runs from a world. Because of this, typically the World Adapter is used for facilitator pages in your project.)\n*\n* ### Using the World Manager\n*\n* To use the World Manager, instantiate it. Then, make calls to any of the methods you need.\n*\n* When you instantiate a World Manager, the world's account id, project id, and group are automatically taken from the session (thanks to the [Authentication Service](../auth-api-service)).\n*\n* Note that the World Manager does *not* create worlds automatically. (This is different than the [Run Manager](../run-manager).) However, you can pass in specific options to any runs created by the manager, using a `run` object.\n*\n* The parameters for creating a World Manager are:\n*\n*   * `account`: The **Team ID** in the Epicenter user interface for this project.\n*   * `project`: The **Project ID** for this project.\n*   * `group`: The **Group Name** for this world.\n*   * `run`: Options to use when creating new runs with the manager, e.g. `run: { files: ['data.xls'] }`.\n*   * `run.model`: The name of the primary model file for this project. Required if you have not already passed it in as part of the `options` parameter for an enclosing call.\n*\n* For example:\n*\n*       var wMgr = new F.manager.WorldManager({\n*          account: 'acme-simulations',\n*          project: 'supply-chain-game',\n*          run: { model: 'supply-chain.py' },\n*          group: 'team1'\n*       });\n*\n*       wMgr.getCurrentRun();\n*/\n\n'use strict';\n\nvar WorldApi = require('../service/world-api-adapter');\nvar RunManager = require('./run-manager');\nvar AuthManager = require('./auth-manager');\nvar worldApi;\n\nfunction buildStrategy(worldId, dtd) {\n\n    return function Ctor(options) {\n        this.options = options;\n\n        $.extend(this, {\n            reset: function () {\n                throw new Error('not implementd. Need api changes');\n            },\n\n            getRun: function (runService) {\n                var me = this;\n                //get or create!\n                // Model is required in the options\n                var model = this.options.run.model || this.options.model;\n                return worldApi.getCurrentRunId({ model: model, filter: worldId })\n                    .then(function (runId) {\n                        return runService.load(runId);\n                    })\n                    .then(function (run) {\n                        dtd.resolveWith(me, [run]);\n                    })\n                    .fail(dtd.reject);\n            }\n        }\n        );\n    };\n}\n\n\nmodule.exports = function (options) {\n    this.options = options || { run: {}, world: {} };\n\n    $.extend(true, this.options, this.options.run);\n    $.extend(true, this.options, this.options.world);\n\n    worldApi = new WorldApi(this.options);\n    this._auth = new AuthManager();\n    var me = this;\n\n    var api = {\n\n        /**\n        * Returns the current world (object) and an instance of the [World API Adapter](../world-api-adapter/).\n        *\n        * **Example**\n        *\n        *       wMgr.getCurrentWorld()\n        *           .then(function(world, worldAdapter) {\n        *               console.log(world.id);\n        *               worldAdapter.getCurrentRunId();\n        *           });\n        *\n        * **Parameters**\n        * @param {string} userId (Optional) The id of the user whose world is being accessed. Defaults to the user in the current session.\n        * @param {string} groupName (Optional) The name of the group whose world is being accessed. Defaults to the group for the user in the current session.\n        * @return {Promise}\n        */\n        getCurrentWorld: function (userId, groupName) {\n            var session = this._auth.getCurrentUserSessionInfo();\n            if (!userId) {\n                userId = session.userId;\n            }\n            if (!groupName) {\n                groupName = session.groupName;\n            }\n            return worldApi.getCurrentWorldForUser(userId, groupName);\n        },\n\n        /**\n        * Returns the current run (object) and an instance of the [Run API Service](../run-api-service/).\n        *\n        * **Example**\n        *\n        *       wMgr.getCurrentRun('myModel.py')\n        *           .then(function(run, runService) {\n        *               console.log(run.id);\n        *               runService.do('startGame');\n        *           });\n        *\n        * **Parameters**\n        * @param {string} model (Optional) The name of the model file. Required if not already passed in as `run.model` when the World Manager is created.\n        * @return {Promise}\n        */\n        getCurrentRun: function (model) {\n            var dtd = $.Deferred();\n            var session = this._auth.getCurrentUserSessionInfo();\n            var curUserId = session.userId;\n            var curGroupName = session.groupName;\n\n            function getAndRestoreLatestRun(world) {\n                if (!world) {\n                    return dtd.reject({ error: 'The user is not part of any world!' });\n                }\n\n                var currentWorldId = world.id;\n                var runOpts = $.extend(true, me.options, { model: model });\n                var strategy = buildStrategy(currentWorldId, dtd);\n                var opt = $.extend(true, {}, {\n                    strategy: strategy,\n                    run: runOpts\n                });\n                var rm = new RunManager(opt);\n                return rm.getRun()\n                    .then(function (run) {\n                        dtd.resolve(run, rm.runService, rm);\n                    });\n            }\n\n            this.getCurrentWorld(curUserId, curGroupName)\n                .then(getAndRestoreLatestRun);\n\n            return dtd.promise();\n        }\n    };\n\n    $.extend(this, api);\n};\n","/**\n * ## File API Service\n *\n * The File API Service allows you to upload and download files directly onto Epicenter, analogous to using the File Manager UI in Epicenter directly or SFTPing files in. It is based on the Epicenter File API.\n *\n *       var fa = new F.service.File({\n *          account: 'acme-simulations',\n *          project: 'supply-chain-game',\n *       });\n *       fa.create('test.txt', 'these are my filecontents');\n *\n *       // alternatively, create a new file using a file uploaded through a file input\n *       // <input id=\"fileupload\" type=\"file\">\n *       //\n *       $('#fileupload').on('change', function (e) {\n *          var file = e.target.files[0];\n *          var data = new FormData();\n *          data.append('file', file, file.name);\n *          fa.create(file.name, data);\n *       });\n */\n\n'use strict';\n\nvar ConfigService = require('./configuration-service');\nvar TransportFactory = require('../transport/http-transport-factory');\nvar SessionManager = require('../store/session-manager');\n\nmodule.exports = function (config) {\n    var defaults = {\n        /**\n         * For projects that require authentication, pass in the user access token (defaults to empty string). If the user is already logged in to Epicenter, the user access token is already set in a cookie and automatically loaded from there. (See [more background on access tokens](../../../project_access/)).\n         * @see [Authentication API Service](../auth-api-service/) for getting tokens.\n         * @type {String}\n         */\n        token: undefined,\n\n        /**\n         * The account id. In the Epicenter UI, this is the **Team ID** (for team projects) or **User ID** (for personal projects). Defaults to undefined.\n         * @type {String}\n         */\n        account: undefined,\n\n        /**\n         * The project id. Defaults to undefined.\n         * @type {String}\n         */\n        project: undefined,\n\n        /**\n         * The folder type.  One of `model` | `static` | `node`.\n         * @type {String}\n         */\n        folderType: 'static',\n\n\n        /**\n         * Options to pass on to the underlying transport layer. All jquery.ajax options at http://api.jquery.com/jQuery.ajax/ are available. Defaults to empty object.\n         * @type {Object}\n         */\n        transport: {}\n    };\n\n    this.sessionManager = new SessionManager();\n    var serviceOptions = this.sessionManager.getMergedOptions(defaults, config);\n    var urlConfig = new ConfigService(serviceOptions).get('server');\n    if (serviceOptions.account) {\n        urlConfig.accountPath = serviceOptions.account;\n    }\n    if (serviceOptions.project) {\n        urlConfig.projectPath = serviceOptions.project;\n    }\n\n    var httpOptions = $.extend(true, {}, serviceOptions.transport, {\n        url: urlConfig.getAPIPath('file')\n    });\n\n    if (serviceOptions.token) {\n        httpOptions.headers = {\n            Authorization: 'Bearer ' + serviceOptions.token\n        };\n    }\n    var http = new TransportFactory(httpOptions);\n\n    function uploadBody(fileName, contents) {\n        var boundary = '---------------------------7da24f2e50046';\n\n        return {\n            body: '--' + boundary + '\\r\\n' +\n                    'Content-Disposition: form-data; name=\"file\";' +\n                    'filename=\"' + fileName + '\"\\r\\n' +\n                    'Content-type: text/html\\r\\n\\r\\n' +\n                    contents + '\\r\\n' +\n                    '--' + boundary + '--',\n            boundary: boundary\n        };\n    }\n\n    function uploadFileOptions(filePath, contents, options) {\n        filePath = filePath.split('/');\n        var fileName = filePath.pop();\n        filePath = filePath.join('/');\n        var path = serviceOptions.folderType + '/' + filePath;\n\n        var extraParams = {};\n        if (contents instanceof FormData) {\n            extraParams = {\n                data: contents,\n                processData: false,\n                contentType: false,\n            };\n        } else {\n            var upload = uploadBody(fileName, contents);\n            extraParams = {\n                data: upload.body,\n                contentType: 'multipart/form-data; boundary=' + upload.boundary\n            };\n        }\n\n        return $.extend(true, {}, serviceOptions, options, {\n            url: urlConfig.getAPIPath('file') + path,\n        }, extraParams);\n    }\n\n    var publicAsyncAPI = {\n        /**\n         * Get a directory listing, or contents of a file.\n         * @param {String} filePath  Path to the file\n         * @param {Object} options (Optional) Overrides for configuration options.\n         * @return {Promise}\n         */\n        getContents: function (filePath, options) {\n            var path = serviceOptions.folderType + '/' + filePath;\n            var httpOptions = $.extend(true, {}, serviceOptions, options, {\n                url: urlConfig.getAPIPath('file') + path\n            });\n            return http.get('', httpOptions);\n        },\n\n        /**\n         * Replaces the file at the given file path.\n         * @param  {String} filePath Path to the file\n         * @param  {String | FormData } contents Contents to write to file\n         * @param  {Object} options  (Optional) Overrides for configuration options\n         * @return {Promise}\n         */\n        replace: function (filePath, contents, options) {\n            var httpOptions = uploadFileOptions(filePath, contents, options);\n            return http.put(httpOptions.data, httpOptions);\n        },\n\n        /**\n         * Creates a file in the given file path.\n         * @param  {String} filePath Path to the file\n         * @param  {String | FormData } contents Contents to write to file\n         * @param  {Boolean} replaceExisting Replace file if it already exists; defaults to false\n         * @param  {Object} options (Optional) Overrides for configuration options\n         * @return {Promise}\n         */\n        create: function (filePath, contents, replaceExisting, options) {\n            var httpOptions = uploadFileOptions(filePath, contents, options);\n            var prom = http.post(httpOptions.data, httpOptions);\n            var me = this;\n            if (replaceExisting === true) {\n                prom = prom.then(null, function (xhr) {\n                    var conflictStatus = 409;\n                    if (xhr.status === conflictStatus) {\n                        return me.replace(filePath, contents, options);\n                    }\n                });\n            }\n            return prom;\n        },\n\n        /**\n         * Removes the file.\n         * @param  {String} filePath Path to the file\n         * @param  {Object} options  (Optional) Overrides for configuration options\n         * @return {Promise}\n         */\n        remove: function (filePath, options) {\n            var path = serviceOptions.folderType + '/' + filePath;\n            var httpOptions = $.extend(true, {}, serviceOptions, options, {\n                url: urlConfig.getAPIPath('file') + path\n            });\n            return http.delete(null, httpOptions);\n        },\n\n        /**\n         * Renames the file.\n         * @param  {String} filePath Path to the file\n         * @param  {String} newName  New name of file\n         * @param  {Object} options  (Optional) Overrides for configuration options\n         * @return {Promise}\n         */\n        rename: function (filePath, newName, options) {\n            var path = serviceOptions.folderType + '/' + filePath;\n            var httpOptions = $.extend(true, {}, serviceOptions, options, {\n                url: urlConfig.getAPIPath('file') + path\n            });\n            return http.patch({ name: newName }, httpOptions);\n        }\n    };\n\n    $.extend(this, publicAsyncAPI);\n};\n","/**\n * ## Asset API Adapter\n *\n * The Asset API Adapter allows you to store assets -- resources or files of any kind -- used by a project with a scope that is specific to project, group, or end user.\n *\n * Assets are used with [team projects](../../../project_admin/#team). One common use case is having end users in a [group](../../../glossary/#groups) or in a [multiplayer world](../../../glossary/#world) upload data -- videos created during game play, profile pictures for customizing their experience, etc. -- as part of playing through the project.\n *\n * Resources created using the Asset Adapter are scoped:\n *\n *  * Project assets are writable only by [team members](../../../glossary/#team), that is, Epicenter authors.\n *  * Group assets are writable by anyone with access to the project that is part of that particular [group](../../../glossary/#groups). This includes all [team members](../../../glossary/#team) (Epicenter authors) and any [end users](../../../glossary/#users) who are members of the group -- both facilitators and standard end users.\n *  * User assets are writable by the specific end user, and by the facilitator of the group.\n *  * All assets are readable by anyone with the exact URI.\n *\n * To use the Asset Adapter, instantiate it and then access the methods provided. Instantiating requires the account id (**Team ID** in the Epicenter user interface) and project id (**Project ID**). The group name is required for assets with a group scope, and the group name and userId are required for assets with a user scope. If not included, they are taken from the logged in user's session information if needed.\n *\n * When creating an asset, you can pass in text (encoded data) to the `create()` call. Alternatively, you can make the `create()` call as part of an HTML form and pass in a file uploaded via the form.\n *\n *       // instantiate the Asset Adapter\n *       var aa = new F.service.Asset({\n *          account: 'acme-simulations',\n *          project: 'supply-chain-game',\n *          group: 'team1',\n *          userId: '12345'\n *       });\n *\n *       // create a new asset using encoded text\n *       aa.create('test.txt', {\n *           encoding: 'BASE_64',\n *           data: 'VGhpcyBpcyBhIHRlc3QgZmlsZS4=',\n *           contentType: 'text/plain'\n *       }, { scope: 'user' });\n *\n *       // alternatively, create a new asset using a file uploaded through a form\n *       // this sample code goes with an html form that looks like this:\n *       //\n *       // <form id=\"upload-file\">\n *       //   <input id=\"file\" type=\"file\">\n *       //   <input id=\"filename\" type=\"text\" value=\"myFile.txt\">\n *       //   <button type=\"submit\">Upload myFile</button>\n *       // </form>\n *       //\n *       $('#upload-file').on('submit', function (e) {\n *          e.preventDefault();\n *          var filename = $('#filename').val();\n *          var data = new FormData();\n *          var inputControl = $('#file')[0];\n *          data.append('file', inputControl.files[0], filename);\n *\n *          aa.create(filename, data, { scope: 'user' });\n *       });\n *\n */\n\n'use strict';\n\nvar ConfigService = require('./configuration-service');\nvar TransportFactory = require('../transport/http-transport-factory');\nvar _pick = require('../util/object-util')._pick;\nvar SessionManager = require('../store/session-manager');\n\nvar apiEndpoint = 'asset';\n\nmodule.exports = function (config) {\n    var defaults = {\n        /**\n         * For projects that require authentication, pass in the user access token (defaults to empty string). If the user is already logged in to Epicenter, the user access token is already set in a cookie and automatically loaded from there. (See [more background on access tokens](../../../project_access/)).\n         * @see [Authentication API Service](../auth-api-service/) for getting tokens.\n         * @type {String}\n         */\n        token: undefined,\n        /**\n         * The account id. In the Epicenter UI, this is the **Team ID** (for team projects). If left undefined, taken from the URL.\n         * @type {String}\n         */\n        account: undefined,\n        /**\n         * The project id. If left undefined, taken from the URL.\n         * @type {String}\n         */\n        project: undefined,\n        /**\n         * The group name. Defaults to session's `groupName`.\n         * @type {String}\n         */\n        group: undefined,\n        /**\n         * The user id. Defaults to session's `userId`.\n         * @type {String}\n         */\n        userId: undefined,\n        /**\n         * The scope for the asset. Valid values are: `user`, `group`, and `project`. See above for the required permissions to write to each scope. Defaults to `user`, meaning the current end user or a facilitator in the end user's group can edit the asset.\n         * @type {String}\n         */\n        scope: 'user',\n        /**\n         * Determines if a request to list the assets in a scope includes the complete URL for each asset (`true`), or only the file names of the assets (`false`). Defaults to `true`.\n         * @type {boolean}\n         */\n        fullUrl: true,\n        /**\n         * The transport object contains the options passed to the XHR request.\n         * @type {object}\n         */\n        transport: {\n            processData: false\n        }\n    };\n    this.sessionManager = new SessionManager();\n    var serviceOptions = this.sessionManager.getMergedOptions(defaults, config);\n    var urlConfig = new ConfigService(serviceOptions).get('server');\n\n    if (!serviceOptions.account) {\n        serviceOptions.account = urlConfig.accountPath;\n    }\n\n    if (!serviceOptions.project) {\n        serviceOptions.project = urlConfig.projectPath;\n    }\n\n    var transportOptions = $.extend(true, {}, serviceOptions.transport, {\n        url: urlConfig.getAPIPath(apiEndpoint)\n    });\n\n    if (serviceOptions.token) {\n        transportOptions.headers = {\n            Authorization: 'Bearer ' + serviceOptions.token\n        };\n    }\n\n    var http = new TransportFactory(transportOptions);\n\n    var assetApiParams = ['encoding', 'data', 'contentType'];\n    var scopeConfig = {\n        user: ['scope', 'account', 'project', 'group', 'userId'],\n        group: ['scope', 'account', 'project', 'group'],\n        project: ['scope', 'account', 'project'],\n    };\n\n    var validateFilename = function (filename) {\n        if (!filename) {\n            throw new Error('filename is needed.');\n        }\n    };\n\n    var validateUrlParams = function (options) {\n        var partKeys = scopeConfig[options.scope];\n        if (!partKeys) {\n            throw new Error('scope parameter is needed.');\n        }\n\n        $.each(partKeys, function () {\n            if (!options[this]) {\n                throw new Error(this + ' parameter is needed.');\n            }\n        });\n    };\n\n    var buildUrl = function (filename, options) {\n        validateUrlParams(options);\n        var partKeys = scopeConfig[options.scope];\n        var parts = $.map(partKeys, function (key) {\n            return options[key];\n        });\n        if (filename) {\n            // This prevents adding a trailing / in the URL as the Asset API\n            // does not work correctly with it\n            filename = '/' + filename;\n        }\n        return urlConfig.getAPIPath(apiEndpoint) + parts.join('/') + filename;\n    };\n\n    // Private function, all requests follow a more or less same approach to\n    // use the Asset API and the difference is the HTTP verb\n    //\n    // @param {string} method` (Required) HTTP verb\n    // @param {string} filename` (Required) Name of the file to delete/replace/create\n    // @param {object} params` (Optional) Body parameters to send to the Asset API\n    // @param {object} options` (Optional) Options object to override global options.\n    var upload = function (method, filename, params, options) {\n        validateFilename(filename);\n        // make sure the parameter is clean\n        method = method.toLowerCase();\n        var contentType = params instanceof FormData === true ? false : 'application/json';\n        if (contentType === 'application/json') {\n            // whitelist the fields that we actually can send to the api\n            params = _pick(params, assetApiParams);\n        } else { // else we're sending form data which goes directly in request body\n            // For multipart/form-data uploads the filename is not set in the URL,\n            // it's getting picked by the FormData field filename.\n            filename = method === 'post' || method === 'put' ? '' : filename;\n        }\n        var urlOptions = $.extend({}, serviceOptions, options);\n        var url = buildUrl(filename, urlOptions);\n        var createOptions = $.extend(true, {}, urlOptions, { url: url, contentType: contentType });\n\n        return http[method](params, createOptions);\n    };\n\n    var publicAPI = {\n        /**\n        * Creates a file in the Asset API. The server returns an error (status code `409`, conflict) if the file already exists, so\n        * check first with a `list()` or a `get()`.\n        *\n        *  **Example**\n        *\n        *       var aa = new F.service.Asset({\n        *          account: 'acme-simulations',\n        *          project: 'supply-chain-game',\n        *          group: 'team1',\n        *          userId: ''\n        *       });\n        *\n        *       // create a new asset using encoded text\n        *       aa.create('test.txt', {\n        *           encoding: 'BASE_64',\n        *           data: 'VGhpcyBpcyBhIHRlc3QgZmlsZS4=',\n        *           contentType: 'text/plain'\n        *       }, { scope: 'user' });\n        *\n        *       // alternatively, create a new asset using a file uploaded through a form\n        *       // this sample code goes with an html form that looks like this:\n        *       //\n        *       // <form id=\"upload-file\">\n        *       //   <input id=\"file\" type=\"file\">\n        *       //   <input id=\"filename\" type=\"text\" value=\"myFile.txt\">\n        *       //   <button type=\"submit\">Upload myFile</button>\n        *       // </form>\n        *       //\n        *       $('#upload-file').on('submit', function (e) {\n        *          e.preventDefault();\n        *          var filename = $('#filename').val();\n        *          var data = new FormData();\n        *          var inputControl = $('#file')[0];\n        *          data.append('file', inputControl.files[0], filename);\n        *\n        *          aa.create(filename, data, { scope: 'user' });\n        *       });\n        *\n        *\n        *  **Parameters**\n        * @param {string} filename (Required) Name of the file to create.\n        * @param {object} params (Optional) Body parameters to send to the Asset API. Required if the `options.transport.contentType` is `application/json`, otherwise ignored.\n        * @param {string} params.encoding Either `HEX` or `BASE_64`. Required if `options.transport.contentType` is `application/json`.\n        * @param {string} params.data The encoded data for the file. Required if `options.transport.contentType` is `application/json`.\n        * @param {string} params.contentType The mime type of the file. Optional.\n        * @param {object} options (Optional) Options object to override global options.\n        * @return {Promise}\n        */\n        create: function (filename, params, options) {\n            return upload('post', filename, params, options);\n        },\n\n        /**\n        * Gets a file from the Asset API, fetching the asset content. (To get a list\n        * of the assets in a scope, use `list()`.)\n        *\n        *  **Parameters**\n        * @param {string} filename (Required) Name of the file to retrieve.\n        * @param {object} options (Optional) Options object to override global options.\n        * @return {Promise}\n        */\n        get: function (filename, options) {\n            var getServiceOptions = _pick(serviceOptions, ['scope', 'account', 'project', 'group', 'userId']);\n            var urlOptions = $.extend({}, getServiceOptions, options);\n            var url = buildUrl(filename, urlOptions);\n            var getOptions = $.extend(true, {}, urlOptions, { url: url });\n\n            return http.get({}, getOptions);\n        },\n\n        /**\n        * Gets the list of the assets in a scope.\n        *\n        * **Example**\n        *\n        *       aa.list({ fullUrl: true }).then(function(fileList){\n        *           console.log('array of files = ', fileList);\n        *       });\n        *\n        *  **Parameters**\n        * @param {object} options (Optional) Options object to override global options.\n        * @param {string} options.scope (Optional) The scope (`user`, `group`, `project`).\n        * @param {boolean} options.fullUrl (Optional) Determines if the list of assets in a scope includes the complete URL for each asset (`true`), or only the file names of the assets (`false`).\n        * @return {Promise}\n        */\n        list: function (options) {\n            var dtd = $.Deferred();\n            var me = this;\n            var urlOptions = $.extend({}, serviceOptions, options);\n            var url = buildUrl('', urlOptions);\n            var getOptions = $.extend(true, {}, urlOptions, { url: url });\n            var fullUrl = getOptions.fullUrl;\n\n            if (!fullUrl) {\n                return http.get({}, getOptions);\n            }\n\n            http.get({}, getOptions)\n                .then(function (files) {\n                    var fullPathFiles = $.map(files, function (file) {\n                        return buildUrl(file, urlOptions);\n                    });\n                    dtd.resolveWith(me, [fullPathFiles]);\n                })\n                .fail(dtd.reject);\n\n            return dtd.promise();\n        },\n\n        /**\n        * Replaces an existing file in the Asset API.\n        *\n        * **Example**\n        *\n        *       // replace an asset using encoded text\n        *       aa.replace('test.txt', {\n        *           encoding: 'BASE_64',\n        *           data: 'VGhpcyBpcyBhIHNlY29uZCB0ZXN0IGZpbGUu',\n        *           contentType: 'text/plain'\n        *       }, { scope: 'user' });\n        *\n        *       // alternatively, replace an asset using a file uploaded through a form\n        *       // this sample code goes with an html form that looks like this:\n        *       //\n        *       // <form id=\"replace-file\">\n        *       //   <input id=\"file\" type=\"file\">\n        *       //   <input id=\"replace-filename\" type=\"text\" value=\"myFile.txt\">\n        *       //   <button type=\"submit\">Replace myFile</button>\n        *       // </form>\n        *       //\n        *       $('#replace-file').on('submit', function (e) {\n        *          e.preventDefault();\n        *          var filename = $('#replace-filename').val();\n        *          var data = new FormData();\n        *          var inputControl = $('#file')[0];\n        *          data.append('file', inputControl.files[0], filename);\n        *\n        *          aa.replace(filename, data, { scope: 'user' });\n        *       });\n        *\n        *  **Parameters**\n        * @param {string} filename (Required) Name of the file being replaced.\n        * @param {object} params (Optional) Body parameters to send to the Asset API. Required if the `options.transport.contentType` is `application/json`, otherwise ignored.\n        * @param {string} params.encoding Either `HEX` or `BASE_64`. Required if `options.transport.contentType` is `application/json`.\n        * @param {string} params.data The encoded data for the file. Required if `options.transport.contentType` is `application/json`.\n        * @param {string} params.contentType The mime type of the file. Optional.\n        * @param {object} options (Optional) Options object to override global options.\n        * @return {Promise}\n        */\n        replace: function (filename, params, options) {\n            return upload('put', filename, params, options);\n        },\n\n        /**\n        * Deletes a file from the Asset API.\n        *\n        * **Example**\n        *\n        *       aa.delete(sampleFileName);\n        *\n        *  **Parameters**\n        * @param {string} filename (Required) Name of the file to delete.\n        * @param {object} options (Optional) Options object to override global options.\n        * @return {Promise}\n        */\n        delete: function (filename, options) {\n            return upload('delete', filename, {}, options);\n        },\n\n        assetUrl: function (filename, options) {\n            var urlOptions = $.extend({}, serviceOptions, options);\n            return buildUrl(filename, urlOptions);\n        }\n    };\n    $.extend(this, publicAPI);\n};\n","/**\n *\n * ## Authentication API Service\n *\n * The Authentication API Service provides a method for logging in, which creates and returns a user access token.\n *\n * User access tokens are required for each call to Epicenter. (See [Project Access](../../../project_access/) for more information.)\n *\n * If you need additional functionality -- such as tracking session information, easily retrieving the user token, or getting the groups to which an end user belongs -- consider using the [Authorization Manager](../auth-manager/) instead.\n *\n *      var auth = new F.service.Auth();\n *      auth.login({ userName: 'jsmith@acmesimulations.com',\n *                  password: 'passw0rd' });\n */\n\n'use strict';\n\nvar ConfigService = require('./configuration-service');\nvar TransportFactory = require('../transport/http-transport-factory');\n\nmodule.exports = function (config) {\n    var defaults = {\n        /**\n         * Email or username to use for logging in. Defaults to empty string.\n         * @type {String}\n         */\n        userName: '',\n\n        /**\n         * Password for specified `userName`. Defaults to empty string.\n         * @type {String}\n         */\n        password: '',\n\n        /**\n         * The account id for this `userName`. In the Epicenter UI, this is the **Team ID** (for team projects) or the **User ID** (for personal projects). Required if the `userName` is for an [end user](../../../glossary/#users). Defaults to empty string.\n         * @type {String}\n         */\n        account: '',\n\n        /**\n         * Options to pass on to the underlying transport layer. All jquery.ajax options at http://api.jquery.com/jQuery.ajax/ are available. Defaults to empty object.\n         * @type {Object}\n         */\n        transport: {}\n    };\n    var serviceOptions = $.extend({}, defaults, config);\n    var urlConfig = new ConfigService(serviceOptions).get('server');\n\n    var transportOptions = $.extend(true, {}, serviceOptions.transport, {\n        url: urlConfig.getAPIPath('authentication')\n    });\n    var http = new TransportFactory(transportOptions);\n\n    var publicAPI = {\n\n        /**\n         * Logs user in, returning the user access token.\n         *\n         * If no `userName` or `password` were provided in the initial configuration options, they are required in the `options` here. If no `account` was provided in the initial configuration options and the `userName` is for an [end user](../../../glossary/#users), the `account` is required as well.\n         *\n         * **Example**\n         *\n         *      auth.login({\n         *          userName: 'jsmith',\n         *          password: 'passw0rd',\n         *          account: 'acme-simulations' })\n         *      .then(function (token) {\n         *          console.log(\"user access token is: \", token.access_token);\n         *      });\n         *\n         * **Parameters**\n         * @param {Object} options (Optional) Overrides for configuration options.\n         * @return {Promise}\n         */\n        login: function (options) {\n            var httpOptions = $.extend(true, { success: $.noop }, serviceOptions, options);\n            if (!httpOptions.userName || !httpOptions.password) {\n                var resp = { status: 401, statusMessage: 'No username or password specified.' };\n                if (options.error) {\n                    options.error.call(this, resp);\n                }\n\n                return $.Deferred().reject(resp).promise();\n            }\n\n            var postParams = {\n                userName: httpOptions.userName,\n                password: httpOptions.password,\n            };\n            if (httpOptions.account) {\n                //pass in null for account under options if you don't want it to be sent\n                postParams.account = httpOptions.account;\n            }\n\n            return http.post(postParams, httpOptions);\n        },\n\n        // (replace with /* */ comment block, to make visible in docs, once this is more than a noop)\n        //\n        // Logs user out from specified accounts.\n        //\n        // Epicenter logout is not implemented yet, so for now this is a dummy promise that gets automatically resolved.\n        //\n        // **Example**\n        //\n        //      auth.logout();\n        //\n        // **Parameters**\n        // @param {Object} `options` (Optional) Overrides for configuration options.\n        //\n        logout: function (options) {\n            var dtd = $.Deferred();\n            dtd.resolve();\n            return dtd.promise();\n        }\n    };\n\n    $.extend(this, publicAPI);\n};\n","/**\n * ## Channel Service\n *\n * The Epicenter platform provides a push channel, which allows you to publish and subscribe to messages within a [project](../../../glossary/#projects), [group](../../../glossary/#groups), or [multiplayer world](../../../glossary/#world). There are two main use cases for the channel: event notifications and chat messages.\n *\n * If you are developing with Epicenter.js, you should use the [Epicenter Channel Manager](../epicenter-channel-manager/) directly. The Epicenter Channel Manager documentation also has more [background](../epicenter-channel-manager/#background) information on channels and their use.\n *\n * The Channel Service is a building block for this functionality. It creates a publish-subscribe object, allowing you to publish messages, subscribe to messages, or unsubscribe from messages for a given 'topic' on a `$.cometd` transport instance.\n *\n * You'll need to include the `epicenter-multiplayer-dependencies.js` library in addition to the `epicenter.js` library in your project to use the Channel Service. See [Including Epicenter.js](../../#include).\n *\n * To use the Channel Service, instantiate it, then make calls to any of the methods you need.\n *\n *        var cs = new F.service.Channel();\n *        cs.publish('/acme-simulations/supply-chain-game/fall-seminar/run/variables', { price: 50 });\n *\n * If you are working through the [Epicenter Channel Manager](../epicenter-channel-manager/), when you ask to \"get\" a particular channel, you are really asking for an instance of the Channel Service with a topic already set, for example to the appropriate group or world:\n *\n *      var cm = new F.manager.ChannelManager();\n *      var gc = cm.getGroupChannel();\n *      // because we used an Epicenter Channel Manager to get the group channel,\n *      // subscribe() and publish() here default to the base topic for the group\n *      gc.subscribe('', function(data) { console.log(data); });\n *      gc.publish('', { message: 'a new message to the group' });\n *\n * The parameters for instantiating a Channel Service include:\n *\n * * `options` The options object to configure the Channel Service.\n * * `options.base` The base topic. This is added as a prefix to all further topics you publish or subscribe to while working with this Channel Service.\n * * `options.topicResolver` A function that processes all 'topics' passed into the `publish` and `subscribe` methods. This is useful if you want to implement your own serialize functions for converting custom objects to topic names. Returns a String. By default, it just echoes the topic.\n * * `options.transport` The instance of `$.cometd` to hook onto. See http://docs.cometd.org/reference/javascript.html for additional background on cometd.\n */\n\n'use strict';\nvar Channel = function (options) {\n    var defaults = {\n\n        /**\n         * The base topic. This is added as a prefix to all further topics you publish or subscribe to while working with this Channel Service.\n         * @type {string}\n         */\n        base: '',\n\n        /**\n         * A function that processes all 'topics' passed into the `publish` and `subscribe` methods. This is useful if you want to implement your own serialize functions for converting custom objects to topic names. By default, it just echoes the topic.\n         *\n         * **Parameters**\n         *\n         * * `topic` Topic to parse.\n         *\n         * **Return Value**\n         *\n         * * *String*: This function should return a string topic.\n         *\n         * @type {function}\n         * @param {String} topic topic to resolve\n         * @return {String}\n         */\n        topicResolver: function (topic) {\n            return topic;\n        },\n\n        /**\n         * The instance of `$.cometd` to hook onto.\n         * @type {object}\n         */\n        transport: null\n    };\n    this.channelOptions = $.extend(true, {}, defaults, options);\n};\n\nvar makeName = function (channelName, topic) {\n    //Replace trailing/double slashes\n    var newName = (channelName ? (channelName + '/' + topic) : topic).replace(/\\/\\//g, '/').replace(/\\/$/, '');\n    return newName;\n};\n\n\nChannel.prototype = $.extend(Channel.prototype, {\n\n    // future functionality:\n    //      // Set the context for the callback\n    //      cs.subscribe('run', function () { this.innerHTML = 'Triggered'}, document.body);\n     //\n     //      // Control the order of operations by setting the `priority`\n     //      cs.subscribe('run', cb, this, {priority: 9});\n     //\n     //      // Only execute the callback, `cb`, if the value of the `price` variable is 50\n     //      cs.subscribe('run/variables/price', cb, this, {priority: 30, value: 50});\n     //\n     //      // Only execute the callback, `cb`, if the value of the `price` variable is greater than 50\n     //      subscribe('run/variables/price', cb, this, {priority: 30, value: '>50'});\n     //\n     //      // Only execute the callback, `cb`, if the value of the `price` variable is even\n     //      subscribe('run/variables/price', cb, this, {priority: 30, value: function (val) {return val % 2 === 0}});\n\n\n    /**\n     * Subscribe to changes on a topic.\n     *\n     * The topic should include the full path of the account id (**Team ID** for team projects), project id, and group name. (In most cases, it is simpler to use the [Epicenter Channel Manager](../epicenter-channel-manager/) instead, in which case this is configured for you.)\n     *\n     *  **Examples**\n     *\n     *      var cb = function(val) { console.log(val.data); };\n     *\n     *      // Subscribe to changes on a top-level 'run' topic\n     *      cs.subscribe('/acme-simulations/supply-chain-game/fall-seminar/run', cb);\n     *\n     *      // Subscribe to changes on children of the 'run' topic. Note this will also be triggered for changes to run.x.y.z.\n     *      cs.subscribe('/acme-simulations/supply-chain-game/fall-seminar/run/*', cb);\n     *\n     *      // Subscribe to changes on both the top-level 'run' topic and its children\n     *      cs.subscribe(['/acme-simulations/supply-chain-game/fall-seminar/run',\n     *          '/acme-simulations/supply-chain-game/fall-seminar/run/*'], cb);\n     *\n     *      // Subscribe to changes on a particular variable\n     *      subscribe('/acme-simulations/supply-chain-game/fall-seminar/run/variables/price', cb);\n     *\n     *\n     * **Return Value**\n     *\n     * * *String* Returns a token you can later use to unsubscribe.\n     *\n     * **Parameters**\n     * @param  {String|Array}   topic    List of topics to listen for changes on.\n     * @param  {Function} callback Callback function to execute. Callback is called with signature `(evt, payload, metadata)`.\n     * @param  {Object}   context  Context in which the `callback` is executed.\n     * @param  {Object}   options  (Optional) Overrides for configuration options.\n     * @param  {Number}   options.priority  Used to control order of operations. Defaults to 0. Can be any +ve or -ve number.\n     * @param  {String|Number|Function}   options.value The `callback` is only triggered if this condition matches. See examples for details.\n     * @return {string} Subscription ID\n     */\n    subscribe: function (topic, callback, context, options) {\n\n        var topics = [].concat(topic);\n        var me = this;\n        var subscriptionIds = [];\n        var opts = me.channelOptions;\n\n        opts.transport.batch(function () {\n            $.each(topics, function (index, topic) {\n                topic = makeName(opts.base, opts.topicResolver(topic));\n                subscriptionIds.push(opts.transport.subscribe(topic, callback));\n            });\n        });\n        return (subscriptionIds[1] ? subscriptionIds : subscriptionIds[0]);\n    },\n\n    /**\n     * Publish data to a topic.\n     *\n     * **Examples**\n     *\n     *      // Send data to all subscribers of the 'run' topic\n     *      cs.publish('/acme-simulations/supply-chain-game/fall-seminar/run', { completed: false });\n     *\n     *      // Send data to all subscribers of the 'run/variables' topic\n     *      cs.publish('/acme-simulations/supply-chain-game/fall-seminar/run/variables', { price: 50 });\n     *\n     * **Parameters**\n     *\n     * @param  {String} topic Topic to publish to.\n     * @param  {*} data  Data to publish to topic.\n     * @return {Array | Object} Responses to published data\n     *\n     */\n    publish: function (topic, data) {\n        var topics = [].concat(topic);\n        var me = this;\n        var returnObjs = [];\n        var opts = me.channelOptions;\n\n\n        opts.transport.batch(function () {\n            $.each(topics, function (index, topic) {\n                topic = makeName(opts.base, opts.topicResolver(topic));\n                if (topic.charAt(topic.length - 1) === '*') {\n                    topic = topic.replace(/\\*+$/, '');\n                    console.warn('You can cannot publish to channels with wildcards. Publishing to ', topic, 'instead');\n                }\n                returnObjs.push(opts.transport.publish(topic, data));\n            });\n        });\n        return (returnObjs[1] ? returnObjs : returnObjs[0]);\n    },\n\n    /**\n     * Unsubscribe from changes to a topic.\n     *\n     * **Example**\n     *\n     *      cs.unsubscribe('sampleToken');\n     *\n     * **Parameters**\n     * @param  {String} token The token for topic is returned when you initially subscribe. Pass it here to unsubscribe from that topic.\n     * @return {Object} reference to current instance\n     */\n    unsubscribe: function (token) {\n        this.channelOptions.transport.unsubscribe(token);\n        return this;\n    },\n\n    /**\n     * Start listening for events on this instance. Signature is same as for jQuery Events: http://api.jquery.com/on/.\n     *\n     * Supported events are: `connect`, `disconnect`, `subscribe`, `unsubscribe`, `publish`, `error`.\n     *\n     * **Parameters**\n     *\n     * @param {string} event The event type. See more detail at jQuery Events: http://api.jquery.com/on/.\n     */\n    on: function (event) {\n        $(this).on.apply($(this), arguments);\n    },\n\n    /**\n     * Stop listening for events on this instance. Signature is same as for jQuery Events: http://api.jquery.com/off/.\n     *\n     * **Parameters**\n     *\n     * @param {string} event The event type. See more detail at jQuery Events: http://api.jquery.com/off/.\n     */\n    off: function (event) {\n        $(this).off.apply($(this), arguments);\n    },\n\n    /**\n     * Trigger events and execute handlers. Signature is same as for jQuery Events: http://api.jquery.com/trigger/.\n     *\n     * **Parameters**\n     *\n     * @param {string} event The event type. See more detail at jQuery Events: http://api.jquery.com/trigger/.\n     */\n    trigger: function (event) {\n        $(this).trigger.apply($(this), arguments);\n    }\n\n});\n\nmodule.exports = Channel;\n","/**\n * @class ConfigurationService\n *\n * All services take in a configuration settings object to configure themselves. A JS hash {} is a valid configuration object, but optionally you can use the configuration service to toggle configs based on the environment\n *\n * @example\n *     var cs = require('configuration-service')({\n *          dev: { //environment\n                port: 3000,\n                host: 'localhost',\n            },\n            prod: {\n                port: 8080,\n                host: 'api.forio.com',\n                logLevel: 'none'\n            },\n            logLevel: 'DEBUG' //global\n *     });\n *\n *      cs.get('logLevel'); //returns 'DEBUG'\n *\n *      cs.setEnv('dev');\n *      cs.get('logLevel'); //returns 'DEBUG'\n *\n *      cs.setEnv('prod');\n *      cs.get('logLevel'); //returns 'none'\n *\n */\n\n'use strict';\nvar urlService = require('./url-config-service');\n\nmodule.exports = function (config) {\n    //TODO: Environments\n    var defaults = {\n        logLevel: 'NONE'\n    };\n    var serviceOptions = $.extend({}, defaults, config);\n    serviceOptions.server = urlService(serviceOptions.server);\n\n    return {\n\n        data: serviceOptions,\n\n        /**\n         * Set the environment key to get configuration options from\n         * @param { string} env\n         */\n        setEnv: function (env) {\n\n        },\n\n        /**\n         * Get configuration.\n         * @param  { string} property optional\n         * @return {*}          Value of property if specified, the entire config object otherwise\n         */\n        get: function (property) {\n            return serviceOptions[property];\n        },\n\n        /**\n         * Set configuration.\n         * @param  { string|Object} key if a key is provided, set a key to that value. Otherwise merge object with current config\n         * @param  {*} value  value for provided key\n         */\n        set: function (key, value) {\n            serviceOptions[key] = value;\n        }\n    };\n};\n\n","/**\n * ## Data API Service\n *\n * The Data API Service allows you to create, access, and manipulate data related to any of your projects. Data are organized in collections. Each collection contains a document; each element of this top-level document is a JSON object. (See additional information on the underlying [Data API](../../../rest_apis/data_api/).)\n *\n * All API calls take in an \"options\" object as the last parameter. The options can be used to extend/override the Data API Service defaults. In particular, there are three required parameters when you instantiate the Data Service:\n *\n * * `account`: Epicenter account id (**Team ID** for team projects, **User ID** for personal projects).\n * * `project`: Epicenter project id.\n * * `root`: The the name of the collection. If you have multiple collections within each of your projects, you can also pass the collection name as an option for each call.\n *\n *       var ds = new F.service.Data({\n *          account: 'acme-simulations',\n *          project: 'supply-chain-game',\n *          root: 'survey-responses',\n *          server: { host: 'api.forio.com' }\n *       });\n *       ds.saveAs('user1',\n *          { 'question1': 2, 'question2': 10,\n *          'question3': false, 'question4': 'sometimes' } );\n *       ds.saveAs('user2',\n *          { 'question1': 3, 'question2': 8,\n *          'question3': true, 'question4': 'always' } );\n *       ds.query('',{ 'question2': { '$gt': 9} });\n *\n * Note that in addition to the `account`, `project`, and `root`, the Data Service parameters optionally include a `server` object, whose `host` field contains the URI of the Forio server. This is automatically set, but you can pass it explicitly if desired. It is most commonly used for clarity when you are [hosting an Epicenter project on your own server](../../../how_to/self_hosting/).\n */\n\n'use strict';\n\nvar ConfigService = require('./configuration-service');\nvar qutil = require('../util/query-util');\nvar TransportFactory = require('../transport/http-transport-factory');\nvar SessionManager = require('../store/session-manager');\n\nmodule.exports = function (config) {\n    var defaults = {\n        /**\n         * Name of collection. Required. Defaults to `/`, that is, the root level of your project at `forio.com/app/your-account-id/your-project-id/`, but must be set to a collection name.\n         * @type {String}\n         */\n        root: '/',\n\n        /**\n         * The account id. In the Epicenter UI, this is the **Team ID** (for team projects) or **User ID** (for personal projects). Defaults to empty string. If left undefined, taken from the URL.\n         * @type {String}\n         */\n        account: undefined,\n\n        /**\n         * The project id. Defaults to empty string. If left undefined, taken from the URL.\n         * @type {String}\n         */\n        project: undefined,\n\n        /**\n         * For operations that require authentication, pass in the user access token (defaults to empty string). If the user is already logged in to Epicenter, the user access token is already set in a cookie and automatically loaded from there. (See [more background on access tokens](../../../project_access/)).\n         * @see [Authentication API Service](../auth-api-service/) for getting tokens.\n         * @type {String}\n         */\n        token: undefined,\n\n        //Options to pass on to the underlying transport layer\n        transport: {}\n    };\n    this.sessionManager = new SessionManager();\n    var serviceOptions = this.sessionManager.getMergedOptions(defaults, config);\n\n    var urlConfig = new ConfigService(serviceOptions).get('server');\n    if (serviceOptions.account) {\n        urlConfig.accountPath = serviceOptions.account;\n    }\n    if (serviceOptions.project) {\n        urlConfig.projectPath = serviceOptions.project;\n    }\n\n    var getURL = function (key, root) {\n        if (!root) {\n            root = serviceOptions.root;\n        }\n        var url = urlConfig.getAPIPath('data') + qutil.addTrailingSlash(root);\n        if (key) {\n            url += qutil.addTrailingSlash(key);\n        }\n        return url;\n    };\n\n    var httpOptions = $.extend(true, {}, serviceOptions.transport, {\n        url: getURL\n    });\n    if (serviceOptions.token) {\n        httpOptions.headers = {\n            Authorization: 'Bearer ' + serviceOptions.token\n        };\n    }\n    var http = new TransportFactory(httpOptions);\n\n    var publicAPI = {\n\n        /**\n         * Search for data within a collection.\n         *\n         * Searching using comparison or logical operators (as opposed to exact matches) requires MongoDB syntax. See the underlying [Data API](../../../rest_apis/data_api/#searching) for additional details.\n         *\n         * **Examples**\n         *\n         *      // request all data associated with document 'user1'\n         *      ds.query('user1');\n         *\n         *      // exact matching:\n         *      // request all documents in collection where 'question2' is 9\n         *      ds.query('', { 'question2': 9});\n         *\n         *      // comparison operators:\n         *      // request all documents in collection\n         *      // where 'question2' is greater than 9\n         *      ds.query('', { 'question2': { '$gt': 9} });\n         *\n         *      // logical operators:\n         *      // request all documents in collection\n         *      // where 'question2' is less than 10, and 'question3' is false\n         *      ds.query('', { '$and': [ { 'question2': { '$lt':10} }, { 'question3': false }] });\n         *\n         *      // regular expresssions: use any Perl-compatible regular expressions\n         *      // request all documents in collection\n         *      // where 'question5' contains the string '.*day'\n         *      ds.query('', { 'question5': { '$regex': '.*day' } });\n         *\n         * **Parameters**\n         * @param {String} key The name of the document to search. Pass the empty string ('') to search the entire collection.\n         * @param {Object} query The query object. For exact matching, this object contains the field name and field value to match. For matching based on comparison, this object contains the field name and the comparison expression. For matching based on logical operators, this object contains an expression using MongoDB syntax. See the underlying [Data API](../../../rest_apis/data_api/#searching) for additional examples.\n         * @param {Object} outputModifier (Optional) Available fields include: `startrecord`, `endrecord`, `sort`, and `direction` (`asc` or `desc`).\n         * @param {Object} options (Optional) Overrides for configuration options.\n         * @return {Promise} \n         */\n        query: function (key, query, outputModifier, options) {\n            var params = $.extend(true, { q: query }, outputModifier);\n            var httpOptions = $.extend(true, {}, serviceOptions, options);\n            httpOptions.url = getURL(key, httpOptions.root);\n            return http.get(params, httpOptions);\n        },\n\n        /**\n         * Save data in an anonymous document within the collection.\n         *\n         * The `root` of the collection must be specified. By default the `root` is taken from the Data Service configuration options; you can also pass the `root` to the `save` call explicitly by overriding the options (third parameter).\n         *\n         * (Additional background: Documents are top-level elements within a collection. Collections must be unique within this account (team or personal account) and project and are set with the `root` field in the `option` parameter. See the underlying [Data API](../../../rest_apis/data_api/) for more information. The `save` method is making a `POST` request.)\n         *\n         * **Example**\n         *\n         *      // Create a new document, with one element, at the default root level\n         *      ds.save('question1', 'yes');\n         *\n         *      // Create a new document, with two elements, at the default root level\n         *      ds.save({ question1:'yes', question2: 32 });\n         *\n         *      // Create a new document, with two elements, at `/students/`\n         *      ds.save({ name:'John', className: 'CS101' }, { root: 'students' });\n         *\n         * **Parameters**\n         *\n         * @param {String|Object} key If `key` is a string, it is the id of the element to save (create) in this document. If `key` is an object, the object is the data to save (create) in this document. In both cases, the id for the document is generated automatically.\n         * @param {Object} value (Optional) The data to save. If `key` is a string, this is the value to save. If `key` is an object, the value(s) to save are already part of `key` and this argument is not required.\n         * @param {Object} options (Optional) Overrides for configuration options. If you want to override the default `root` of the collection, do so here.\n         * @return {Promise} \n         */\n        save: function (key, value, options) {\n            var attrs;\n            if (typeof key === 'object') {\n                attrs = key;\n                options = value;\n            } else {\n                (attrs = {})[key] = value;\n            }\n            var httpOptions = $.extend(true, {}, serviceOptions, options);\n            httpOptions.url = getURL('', httpOptions.root);\n\n            return http.post(attrs, httpOptions);\n        },\n\n        /**\n         * Save (create or replace) data in a named document or element within the collection. \n         * \n         * The `root` of the collection must be specified. By default the `root` is taken from the Data Service configuration options; you can also pass the `root` to the `saveAs` call explicitly by overriding the options (third parameter).\n         *\n         * Optionally, the named document or element can include path information, so that you are saving just part of the document.\n         *\n         * (Additional background: Documents are top-level elements within a collection. Collections must be unique within this account (team or personal account) and project and are set with the `root` field in the `option` parameter. See the underlying [Data API](../../../rest_apis/data_api/) for more information. The `saveAs` method is making a `PUT` request.)\n         *\n         * **Example**\n         *\n         *      // Create (or replace) the `user1` document at the default root level.\n         *      // Note that this replaces any existing content in the `user1` document.\n         *      ds.saveAs('user1',\n         *          { 'question1': 2, 'question2': 10,\n         *           'question3': false, 'question4': 'sometimes' } );\n         *\n         *      // Create (or replace) the `student1` document at the `students` root, \n         *      // that is, the data at `/students/student1/`.\n         *      // Note that this replaces any existing content in the `/students/student1/` document.\n         *      // However, this will keep existing content in other paths of this collection.\n         *      // For example, the data at `/students/student2/` is unchanged by this call.\n         *      ds.saveAs('student1',\n         *          { firstName: 'john', lastName: 'smith' },\n         *          { root: 'students' });\n         *\n         *      // Create (or replace) the `mgmt100/groupB` document at the `myclasses` root,\n         *      // that is, the data at `/myclasses/mgmt100/groupB/`.\n         *      // Note that this replaces any existing content in the `/myclasses/mgmt100/groupB/` document.\n         *      // However, this will keep existing content in other paths of this collection.\n         *      // For example, the data at `/myclasses/mgmt100/groupA/` is unchanged by this call.\n         *      ds.saveAs('mgmt100/groupB',\n         *          { scenarioYear: '2015' },\n         *          { root: 'myclasses' });\n         *\n         * **Parameters**\n         *\n         * @param {String} key Id of the document.\n         * @param {Object} value (Optional) The data to save, in key:value pairs.\n         * @param {Object} options (Optional) Overrides for configuration options. If you want to override the default `root` of the collection, do so here.\n         * @return {Promise} \n         */\n        saveAs: function (key, value, options) {\n            var httpOptions = $.extend(true, {}, serviceOptions, options);\n            httpOptions.url = getURL(key, httpOptions.root);\n\n            return http.put(value, httpOptions);\n        },\n\n        /**\n         * Get data for a specific document or field.\n         *\n         * **Example**\n         *\n         *      ds.load('user1');\n         *      ds.load('user1/question3');\n         *\n         * **Parameters**\n         * @param  {String|Object} key The id of the data to return. Can be the id of a document, or a path to data within that document.\n         * @param {Object} outputModifier (Optional) Available fields include: `startrecord`, `endrecord`, `sort`, and `direction` (`asc` or `desc`).\n         * @param {Object} options Overrides for configuration options.\n         * @return {Promise} \n         */\n        load: function (key, outputModifier, options) {\n            var httpOptions = $.extend(true, {}, serviceOptions, options);\n            httpOptions.url = getURL(key, httpOptions.root);\n            return http.get(outputModifier, httpOptions);\n        },\n\n        /**\n         * Removes data from collection. Only documents (top-level elements in each collection) can be deleted.\n         *\n         * **Example**\n         *\n         *     ds.remove('user1');\n         *\n         *\n         * **Parameters**\n         *\n         * @param {String|Array} keys The id of the document to remove from this collection, or an array of such ids.\n         * @param {Object} options (Optional) Overrides for configuration options.\n         * @return {Promise} \n         */\n        remove: function (keys, options) {\n            var httpOptions = $.extend(true, {}, serviceOptions, options);\n            var params;\n            if ($.isArray(keys)) {\n                params = { id: keys };\n            } else {\n                params = '';\n                httpOptions.url = getURL(keys, httpOptions.root);\n            }\n            return http.delete(params, httpOptions);\n        }\n\n        // Epicenter doesn't allow nuking collections\n        //     /**\n        //      * Removes collection being referenced\n        //      * @return null\n        //      */\n        //     destroy: function (options) {\n        //         return this.remove('', options);\n        //     }\n    };\n\n    $.extend(this, publicAPI);\n};\n","/**\n *\n * ## Group API Adapter\n *\n * The Group API Adapter provides methods to look up, create, change or remove information about groups in a project. It is based on query capabilities of the underlying RESTful [Group API](../../../rest_apis/user_management/group/).\n *\n * This is only needed for Authenticated projects, that is, team projects with [end users and groups](../../../groups_and_end_users/).\n *\n *      var ma = new F.service.Group({ token: 'user-or-project-access-token' });\n *      ma.getGroupsForProject({ account: 'acme', project: 'sample' });\n */\n\n'use strict';\n\nvar serviceUtils = require('./service-utils');\nvar TransportFactory = require('../transport/http-transport-factory');\nvar objectAssign = require('object-assign');\n\nvar apiEndpoint = 'group/local';\n\nvar GroupService = function (config) {\n    var defaults = {\n        /**\n         * Epicenter account name. Defaults to undefined.\n         * @type {string}\n         */\n        account: undefined,\n\n        /**\n         * Epicenter project name. Defaults to undefined.\n         * @type {string}\n         */\n        project: undefined,\n\n        /**\n         * Options to pass on to the underlying transport layer. All jquery.ajax options at http://api.jquery.com/jQuery.ajax/ are available. Defaults to empty object.\n         * @type {object}\n         */\n        transport: {}\n    };\n    var serviceOptions = serviceUtils.getDefaultOptions(defaults, config, { apiEndpoint: apiEndpoint });\n    var transportOptions = serviceOptions.transport;\n    delete serviceOptions.transport;\n    var http = new TransportFactory(transportOptions, serviceOptions);\n    var publicAPI = {\n        /**\n        * Gets information for a group or multiple groups.\n        * @param {Object} params object with query parameters\n        * @patam {string} params.q partial match for name, organization or event.\n        * @patam {string} params.account Epicenter's Team ID\n        * @patam {string} params.project Epicenter's Project ID\n        * @patam {string} params.name Epicenter's Group Name\n        * @param {Object} options (Optional) Overrides for configuration options.\n        * @return {Promise}\n        */\n        getGroups: function (params, options) {\n            //groupID is part of the URL\n            //q, account and project are part of the query string\n            var finalOpts = objectAssign({}, serviceOptions, options);\n            var finalParams;\n            if (typeof params === 'string') {\n                finalOpts.url = serviceUtils.getApiUrl(apiEndpoint + '/' + params, finalOpts);\n            } else {\n                finalParams = params;\n            }\n            return http.get(finalParams, finalOpts);\n        }\n    };\n    objectAssign(this, publicAPI);\n};\n\nmodule.exports = GroupService;\n","/**\n *\n * ## Introspection API Service\n *\n * The Introspection API Service allows you to view a list of the variables and operations in a model. Typically used in conjunction with the [Run API Service](../run-api-service/).\n *\n * The Introspection API Service is not available for Forio SimLang.\n *\n *       var intro = new F.service.Introspect({\n *               account: 'acme-simulations',\n *               project: 'supply-chain-game'\n *       });\n *       intro.byModel('supply-chain.py').then(function(data){ ... });\n *       intro.byRunID('2b4d8f71-5c34-435a-8c16-9de674ab72e6').then(function(data){ ... });\n *\n */\n\n'use strict';\n\nvar ConfigService = require('./configuration-service');\nvar TransportFactory = require('../transport/http-transport-factory');\nvar SessionManager = require('../store/session-manager');\n\nvar apiEndpoint = 'model/introspect';\n\nmodule.exports = function (config) {\n    var defaults = {\n        /**\n         * For projects that require authentication, pass in the user access token (defaults to empty string). If the user is already logged in to Epicenter, the user access token is already set in a cookie and automatically loaded from there. (See [more background on access tokens](../../../project_access/)).\n         * @see [Authentication API Service](../auth-api-service/) for getting tokens.\n         * @type {String}\n         */\n        token: undefined,\n\n        /**\n         * The account id. In the Epicenter UI, this is the **Team ID** (for team projects) or **User ID** (for personal projects). Defaults to empty string. If left undefined, taken from the URL.\n         * @type {String}\n         */\n        account: undefined,\n\n        /**\n         * The project id. Defaults to empty string. If left undefined, taken from the URL.\n         * @type {String}\n         */\n        project: undefined,\n\n    };\n\n    var sessionManager = new SessionManager();\n    var serviceOptions = sessionManager.getMergedOptions(defaults, config);\n\n    var urlConfig = new ConfigService(serviceOptions).get('server');\n    if (serviceOptions.account) {\n        urlConfig.accountPath = serviceOptions.account;\n    }\n    if (serviceOptions.project) {\n        urlConfig.projectPath = serviceOptions.project;\n    }\n\n    var transportOptions = $.extend(true, {}, serviceOptions.transport, {\n        url: urlConfig.getAPIPath(apiEndpoint)\n    });\n    if (serviceOptions.token) {\n        transportOptions.headers = {\n            Authorization: 'Bearer ' + serviceOptions.token\n        };\n    }\n    var http = new TransportFactory(transportOptions);\n\n    var publicAPI = {\n        /**\n         * Get the available variables and operations for a given model file.\n         *\n         * Note: This does not work for any model which requires additional parameters, such as `files`.\n         *\n         * **Example**\n         *\n         *      intro.byModel('abc.vmf')\n         *          .then(function(data) {\n         *              // data contains an object with available functions (used with operations API) and available variables (used with variables API)\n         *              console.log(data.functions);\n         *              console.log(data.variables);\n         *          });\n         *\n         * **Parameters**\n         * @param  {String} modelFile Name of the model file to introspect.\n         * @param  {Object} options (Optional) Overrides for configuration options.\n         * @return {Promise} \n         */\n        byModel: function (modelFile, options) {\n            var opts = $.extend(true, {}, serviceOptions, options);\n            if (!opts.account || !opts.project) {\n                throw new Error('Account and project are required when using introspect#byModel');\n            }\n            if (!modelFile) {\n                throw new Error('modelFile is required when using introspect#byModel');\n            }\n            var url = { url: urlConfig.getAPIPath(apiEndpoint) + [opts.account, opts.project, modelFile].join('/') };\n            var httpOptions = $.extend(true, {}, serviceOptions, options, url);\n            return http.get('', httpOptions);\n        },\n\n        /**\n         * Get the available variables and operations for a given model file.\n         *\n         * Note: This does not work for any model which requires additional parameters such as `files`.\n         *\n         * **Example**\n         *\n         *      intro.byRunID('2b4d8f71-5c34-435a-8c16-9de674ab72e6')\n         *          .then(function(data) {\n         *              // data contains an object with available functions (used with operations API) and available variables (used with variables API)\n         *              console.log(data.functions);\n         *              console.log(data.variables);\n         *          });\n         *\n         * **Parameters**\n         * @param  {String} runID Id of the run to introspect.\n         * @param  {Object} options (Optional) Overrides for configuration options.\n         * @return {Promise} \n         */\n        byRunID: function (runID, options) {\n            if (!runID) {\n                throw new Error('runID is required when using introspect#byModel');\n            }\n            var url = { url: urlConfig.getAPIPath(apiEndpoint) + runID };\n            var httpOptions = $.extend(true, {}, serviceOptions, options, url);\n            return http.get('', httpOptions);\n        }\n    };\n    $.extend(this, publicAPI);\n};\n","/**\n *\n * ## Member API Adapter\n *\n * The Member API Adapter provides methods to look up information about end users for your project and how they are divided across groups. It is based on query capabilities of the underlying RESTful [Member API](../../../rest_apis/user_management/member/).\n *\n * This is only needed for Authenticated projects, that is, team projects with [end users and groups](../../../groups_and_end_users/). For example, if some of your end users are facilitators, or if your end users should be treated differently based on which group they are in, use the Member API to find that information.\n *\n *      var ma = new F.service.Member({ token: 'user-or-project-access-token' });\n *      ma.getGroupsForUser({ userId: 'b6b313a3-ab84-479c-baea-206f6bff337' });\n *      ma.getGroupDetails({ groupId: '00b53308-9833-47f2-b21e-1278c07d53b8' });\n */\n\n'use strict';\n\nvar ConfigService = require('./configuration-service');\nvar TransportFactory = require('../transport/http-transport-factory');\nvar SessionManager = require('../store/session-manager');\nvar _pick = require('../util/object-util')._pick;\nvar apiEndpoint = 'member/local';\n\nmodule.exports = function (config) {\n    var defaults = {\n        /**\n         * Epicenter user id. Defaults to a blank string.\n         * @type {string}\n         */\n        userId: undefined,\n\n        /**\n         * Epicenter group id. Defaults to a blank string. Note that this is the group *id*, not the group *name*.\n         * @type {string}\n         */\n        groupId: undefined,\n\n        /**\n         * Options to pass on to the underlying transport layer. All jquery.ajax options at http://api.jquery.com/jQuery.ajax/ are available. Defaults to empty object.\n         * @type {object}\n         */\n        transport: {}\n    };\n    this.sessionManager = new SessionManager();\n    var serviceOptions = this.sessionManager.getMergedOptions(defaults, config);\n    var urlConfig = new ConfigService(serviceOptions).get('server');\n\n    var transportOptions = $.extend(true, {}, serviceOptions.transport, {\n        url: urlConfig.getAPIPath(apiEndpoint)\n    });\n\n    if (serviceOptions.token) {\n        transportOptions.headers = {\n            Authorization: 'Bearer ' + serviceOptions.token\n        };\n    }\n    var http = new TransportFactory(transportOptions, serviceOptions);\n\n    var getFinalParams = function (params) {\n        if (typeof params === 'object') {\n            return $.extend(true, serviceOptions, params);\n        }\n        return serviceOptions;\n    };\n\n    var patchUserActiveField = function (params, active, options) {\n        var httpOptions = $.extend(true, serviceOptions, options, {\n            url: urlConfig.getAPIPath(apiEndpoint) + params.groupId + '/' + params.userId\n        });\n\n        return http.patch({ active: active }, httpOptions);\n    };\n\n    var publicAPI = {\n\n        /**\n        * Retrieve details about all of the group memberships for one end user. The membership details are returned in an array, with one element (group record) for each group to which the end user belongs.\n        *\n        * In the membership array, each group record includes the group id, project id, account (team) id, and an array of members. However, only the user whose userId is included in the call is listed in the members array (regardless of whether there are other members in this group).\n        *\n        * **Example**\n        *\n        *       var ma = new F.service.Member({ token: 'user-or-project-access-token' });\n        *       ma.getGroupsForUser('42836d4b-5b61-4fe4-80eb-3136e956ee5c')\n        *           .then(function(memberships){\n        *               for (var i=0; i<memberships.length; i++) {\n        *                   console.log(memberships[i].groupId);\n        *               }\n        *           });\n        *\n        *       ma.getGroupsForUser({ userId: '42836d4b-5b61-4fe4-80eb-3136e956ee5c' });\n        *\n        * **Parameters**\n        * @param {string|object} params The user id for the end user. Alternatively, an object with field `userId` and value the user id.\n        * @param {object} options (Optional) Overrides for configuration options.\n        */\n\n        getGroupsForUser: function (params, options) {\n            options = options || {};\n            var httpOptions = $.extend(true, serviceOptions, options);\n            var isString = typeof params === 'string';\n            var objParams = getFinalParams(params);\n            if (!isString && !objParams.userId) {\n                throw new Error('No userId specified.');\n            }\n\n            var getParms = isString ? { userId: params } : _pick(objParams, 'userId');\n            return http.get(getParms, httpOptions);\n        },\n\n        /**\n        * Retrieve details about one group, including an array of all its members.\n        *\n        * **Example**\n        *\n        *       var ma = new F.service.Member({ token: 'user-or-project-access-token' });\n        *       ma.getGroupDetails('80257a25-aa10-4959-968b-fd053901f72f')\n        *           .then(function(group){\n        *               for (var i=0; i<group.members.length; i++) {\n        *                   console.log(group.members[i].userName);\n        *               }\n        *           });\n        *\n        *       ma.getGroupDetails({ groupId: '80257a25-aa10-4959-968b-fd053901f72f' });\n        *\n        * **Parameters**\n        * @param {string|object} params The group id. Alternatively, an object with field `groupId` and value the group id.\n        * @param {object} options (Optional) Overrides for configuration options.\n        * @return {Promise}\n        */\n        getGroupDetails: function (params, options) {\n            options = options || {};\n            var isString = typeof params === 'string';\n            var objParams = getFinalParams(params);\n            if (!isString && !objParams.groupId) {\n                throw new Error('No groupId specified.');\n            }\n\n            var groupId = isString ? params : objParams.groupId;\n            var httpOptions = $.extend(true, serviceOptions,\n                options,\n                { url: urlConfig.getAPIPath(apiEndpoint) + groupId }\n            );\n\n            return http.get({}, httpOptions);\n        },\n\n        /**\n        * Set a particular end user as `active`. Active end users can be assigned to [worlds](../world-manager/) in multiplayer games during automatic assignment.\n        *\n        * **Example**\n        *\n        *       var ma = new F.service.Member({ token: 'user-or-project-access-token' });\n        *       ma.makeUserActive({ userId: '42836d4b-5b61-4fe4-80eb-3136e956ee5c',\n        *                           groupId: '80257a25-aa10-4959-968b-fd053901f72f' });\n        *\n        * **Parameters**\n        * @param {object} params The end user and group information.\n        * @param {string} params.userId The id of the end user to make active.\n        * @param {string} params.groupId The id of the group to which this end user belongs, and in which the end user should become active.\n        * @param {object} options (Optional) Overrides for configuration options.\n        * @return {Promise}\n        */\n        makeUserActive: function (params, options) {\n            return patchUserActiveField(params, true, options);\n        },\n\n        /**\n        * Set a particular end user as `inactive`. Inactive end users are not assigned to [worlds](../world-manager/) in multiplayer games during automatic assignment.\n        *\n        * **Example**\n        *\n        *       var ma = new F.service.Member({ token: 'user-or-project-access-token' });\n        *       ma.makeUserInactive({ userId: '42836d4b-5b61-4fe4-80eb-3136e956ee5c',\n        *                           groupId: '80257a25-aa10-4959-968b-fd053901f72f' });\n        *\n        * **Parameters**\n        * @param {object} params The end user and group information.\n        * @param {string} params.userId The id of the end user to make inactive.\n        * @param {string} params.groupId The id of the group to which this end user belongs, and in which the end user should become inactive.\n        * @param {object} options (Optional) Overrides for configuration options.\n        * @return {Promise}\n        */\n        makeUserInactive: function (params, options) {\n            return patchUserActiveField(params, false, options);\n        }\n    };\n\n    $.extend(this, publicAPI);\n};\n","/**\n *\n * ## Presence API Service\n *\n * The Presence API Service provides methods to get and set the presence of an end user in a project, that is, to indicate whether the end user is online. This happens automatically: in projects that use [channels](../epicenter-channel-manager/), the end user's presence is published automatically on a \"presence\" channel that is specific to each group. You can also use the Presence API Service to do this explicitly: you can make a call to indicate that a particular end user is online or offline. \n *\n * The Presence API Service is only needed for Authenticated projects, that is, team projects with [end users and groups](../../../groups_and_end_users/). It is typically used only in multiplayer projects, to facilitate end users communicating with each other. It is based on the query capabilities of the underlying RESTful [Presence API](../../../rest_apis/multiplayer/presence/).\n *\n *      var pr = new F.service.Presence();\n *      pr.markOnline('example-userId');\n *      pr.markOffline('example-userId');\n *      pr.getStatus();\n */\n\nvar ConfigService = require('./configuration-service');\nvar TransportFactory = require('../transport/http-transport-factory');\nvar SessionManager = require('../store/session-manager');\nvar apiEndpoint = 'presence';\nvar ChannelManager = require('../managers/epicenter-channel-manager');\n\nmodule.exports = function (config) {\n    var defaults = {\n        /**\n         * The account id. In the Epicenter UI, this is the **Team ID** (for team projects) or **User ID** (for personal projects). Defaults to undefined. If left undefined, taken from the URL or session manager.\n         * @type {String}\n         */\n        account: undefined,\n\n        /**\n         * The project id. Defaults to undefined. If left undefined, taken from the URL or session manager.\n         * @type {String}\n         */\n        project: undefined,\n\n        /**\n         * Epicenter group name. Defaults to undefined. Note that this is the group *name*, not the group *id*. If left blank, taken from the session manager.\n         * @type {string}\n         */\n        groupName: undefined,\n\n        /**\n         * Options to pass on to the underlying transport layer. All jquery.ajax options at http://api.jquery.com/jQuery.ajax/ are available. Defaults to empty object.\n         * @type {object}\n         */\n        transport: {},\n    };\n    this.sessionManager = new SessionManager();\n    var serviceOptions = this.sessionManager.getMergedOptions(defaults, config);\n    var urlConfig = new ConfigService(serviceOptions).get('server');\n    if (serviceOptions.account) {\n        urlConfig.accountPath = serviceOptions.account;\n    }\n    if (serviceOptions.project) {\n        urlConfig.projectPath = serviceOptions.project;\n    }\n\n    var transportOptions = $.extend(true, {}, serviceOptions.transport, {\n        url: urlConfig.getAPIPath(apiEndpoint)\n    });\n\n    if (serviceOptions.token) {\n        transportOptions.headers = {\n            Authorization: 'Bearer ' + serviceOptions.token\n        };\n    }\n    var http = new TransportFactory(transportOptions, serviceOptions);\n\n\n    var getFinalParams = function (params) {\n        if (typeof params === 'object') {\n            return $.extend(true, {}, serviceOptions, params);\n        }\n        return serviceOptions;\n    };\n\n    var publicAPI = {\n        /**\n         * Marks an end user as online.\n         *\n         *\n         * **Example**\n         *\n         *     var pr = new F.service.Presence();\n         *     pr.markOnline('0000015a68d806bc09cd0a7d207f44ba5f74')\n         *          .then(function(presenceObj) {\n         *               console.log('user ', presenceObj.userId, \n         *                    ' now online, as of ', presenceObj.lastModified);\n         *          });\n         *\n         * **Return Value**\n         *\n         * Promise with presence information for user marked online.\n         *\n         * **Parameters**\n         *\n         * @param  {String} userId (optional) If not provided, taken from session cookie.\n         * @param  {Object} options Additional options to change the presence service defaults.\n         * @return {Promise} promise\n         */\n        markOnline: function (userId, options) {\n            options = options || {};\n            var isString = typeof userId === 'string';\n            var objParams = getFinalParams(userId);\n            if (!objParams.groupName && !options.groupName) {\n                throw new Error('No groupName specified.');\n            }\n            userId = isString ? userId : objParams.userId;\n            var groupName = options.groupName || objParams.groupName;\n            var httpOptions = $.extend(true, {}, serviceOptions, options,\n                { url: urlConfig.getAPIPath(apiEndpoint) + groupName + '/' + userId }\n            );\n            return http.post({ message: 'online' }, httpOptions);\n        },\n\n        /**\n         * Marks an end user as offline.\n         *\n         *\n         * **Example**\n         *\n         *     var pr = new F.service.Presence();\n         *     pr.markOffline('0000015a68d806bc09cd0a7d207f44ba5f74');\n         *\n         * **Return Value**\n         *\n         * Promise to remove presence record for end user.\n         *\n         * **Parameters**\n         *\n         * @param  {String} userId (optional) If not provided, taken from session cookie.\n         * @param  {Object} options Additional options to change the presence service defaults.\n         * @return {Promise} promise\n         */\n        markOffline: function (userId, options) {\n            options = options || {};\n            var isString = typeof userId === 'string';\n            var objParams = getFinalParams(userId);\n            if (!objParams.groupName && !options.groupName) {\n                throw new Error('No groupName specified.');\n            }\n            userId = isString ? userId : objParams.userId;\n            var groupName = options.groupName || objParams.groupName;\n            var httpOptions = $.extend(true, {}, serviceOptions, options,\n                { url: urlConfig.getAPIPath(apiEndpoint) + groupName + '/' + userId }\n            );\n            return http.delete({}, httpOptions);\n        },\n\n        /**\n         * Returns a list of all end users in this group that are currently online.\n         *\n         *\n         * **Example**\n         *\n         *     var pr = new F.service.Presence();\n         *     pr.getStatus('groupName').then(function(onlineUsers) {\n         *          for (var i=0; i < onlineUsers.length; i++) {\n         *               console.log('user ', onlineUsers[i].userId, \n         *                    ' is online as of ', onlineUsers[i].lastModified);\n         *          }\n         *     });\n         *\n         * **Return Value**\n         *\n         * Promise with response of online users\n         *\n         * **Parameters**\n         *\n         * @param  {String} groupName (optional) If not provided, taken from session cookie.\n         * @param  {Object} options Additional options to change the presence service defaults.\n         * @return {Promise} promise\n         */\n        getStatus: function (groupName, options) {\n            options = options || {};\n            var objParams = getFinalParams(groupName);\n            if (!groupName && !objParams.groupName) {\n                throw new Error('No groupName specified.');\n            }\n            groupName = groupName || objParams.groupName;\n            var httpOptions = $.extend(true, {}, serviceOptions, options,\n                { url: urlConfig.getAPIPath(apiEndpoint) + groupName }\n            );\n            return http.get({}, httpOptions);\n        },\n\n        /**\n         * End users are automatically marked online and offline in a \"presence\" channel that is specific to each group. Gets this channel (an instance of the [Channel Service](../channel-service/)) for the given group. (Note that this Channel Service instance is also available from the [Epicenter Channel Manager getPresenceChannel()](../epicenter-channel-manager/#getPresenceChannel).)\n         *\n         *\n         * **Example**\n         *\n         *     var pr = new F.service.Presence();\n         *     var cm = pr.getChannel('group1');\n         *     cm.publish('', 'a message to presence channel');\n         *\n         * **Return Value**\n         *\n         * Channel instance for Presence channel\n         *\n         * **Parameters**\n         *\n         * @param  {String} groupName (optional) If not provided, taken from session cookie.\n         * @param  {Object} options Additional options to change the presence service defaults\n         * @return {Channel} Channel instance\n         */\n        getChannel: function (groupName, options) {\n            options = options || {};\n            var isString = typeof groupName === 'string';\n            var objParams = getFinalParams(groupName);\n            if (!isString && !objParams.groupName) {\n                throw new Error('No groupName specified.');\n            }\n            groupName = isString ? groupName : objParams.groupName;\n            var cm = new ChannelManager(options);\n            return cm.getPresenceChannel(groupName);\n        }\n    };\n\n    $.extend(this, publicAPI);\n};\n","/**\n *\n * ## Run API Service\n *\n * The Run API Service allows you to perform common tasks around creating and updating runs, variables, and data.\n *\n * When building interfaces to show run one at a time (as for standard end users), typically you first instantiate a [Run Manager](../run-manager/) and then access the Run Service that is automatically part of the manager, rather than instantiating the Run Service directly. This is because the Run Manager (and associated [run strategies](../strategies/)) gives you control over run creation depending on run states.\n *\n * The Run API Service is useful for building an interface where you want to show data across multiple runs (this is easy using the `filter()` and `query()` methods). For instance, you would probably use a Run Service to build a page for a facilitator. This is because a facilitator typically wants to evaluate performance from multiple end users, each of whom have been working with their own run.\n *\n * To use the Run API Service, instantiate it by passing in:\n *\n * * `account`: Epicenter account id (**Team ID** for team projects, **User ID** for personal projects).\n * * `project`: Epicenter project id.\n *\n * If you know in advance that you would like to work with particular, existing run(s), you can optionally pass in:\n *\n * * `filter`: (Optional) Criteria by which to filter for existing runs. \n * * `id`: (Optional) The run id of an existing run. This is a convenience alias for using filter, in the case where you only want to work with one run.\n *\n * For example,\n *\n *       var rs = new F.service.Run({\n *            account: 'acme-simulations',\n *            project: 'supply-chain-game',\n *      });\n *      rs.create('supply_chain_game.py').then(function(run) {\n *             rs.do('someOperation');\n *      });\n *\n *\n * Additionally, all API calls take in an `options` object as the last parameter. The options can be used to extend/override the Run API Service defaults listed below. In particular, passing `{ id: 'a-run-id' }` in this `options` object allows you to make calls to an existing run.\n *\n * Note that in addition to the `account`, `project`, and `model`, the Run Service parameters optionally include a `server` object, whose `host` field contains the URI of the Forio server. This is automatically set, but you can pass it explicitly if desired. It is most commonly used for clarity when you are [hosting an Epicenter project on your own server](../../../how_to/self_hosting/).\n *\n *       var rm = new F.manager.RunManager({\n *           run: {\n *               account: 'acme-simulations',\n *               project: 'supply-chain-game',\n *               model: 'supply_chain_game.py',\n *               server: { host: 'api.forio.com' }\n *           }\n *       });\n *       rm.getRun()\n *           .then(function(run) {\n *               // the RunManager.run contains the instantiated Run Service,\n *               // so any Run Service method is valid here\n *               var rs = rm.run;\n *               rs.do('someOperation');\n *       })\n *\n */\n\n'use strict';\n\nvar ConfigService = require('./configuration-service');\nvar qutil = require('../util/query-util');\nvar rutil = require('../util/run-util');\nvar _pick = require('../util/object-util')._pick;\nvar TransportFactory = require('../transport/http-transport-factory');\nvar VariablesService = require('./variables-api-service');\nvar IntrospectionService = require('./introspection-api-service');\nvar SessionManager = require('../store/session-manager');\n\nmodule.exports = function (config) {\n    var defaults = {\n        /**\n         * For projects that require authentication, pass in the user access token (defaults to undefined). If the user is already logged in to Epicenter, the user access token is already set in a cookie and automatically loaded from there. (See [more background on access tokens](../../../project_access/)).\n         * @see [Authentication API Service](../auth-api-service/) for getting tokens.\n         * @type {String}\n         */\n        token: undefined,\n\n        /**\n         * The account id. In the Epicenter UI, this is the **Team ID** (for team projects) or **User ID** (for personal projects). Defaults to undefined. If left undefined, taken from the URL.\n         * @type {String}\n         */\n        account: undefined,\n\n        /**\n         * The project id. Defaults to undefined. If left undefined, taken from the URL.\n         * @type {String}\n         */\n        project: undefined,\n\n        /**\n         * Criteria by which to filter runs. Defaults to empty string.\n         * @type {String}\n         */\n        filter: '',\n\n        /**\n         * Convenience alias for filter. Pass in an existing run id to interact with a particular run.\n         * @type {String}\n         */\n        id: '',\n\n        /**\n         * Flag determines if `X-AutoRestore: true` header is sent to Epicenter, meaning runs are automatically pulled from the Epicenter backend database if not currently in memory on the Epicenter servers. Defaults to `true`.\n         * @type {boolean}\n         */\n        autoRestore: true,\n\n        /**\n         * Called when the call completes successfully. Defaults to `$.noop`.\n         * @type {function}\n         */\n        success: $.noop,\n\n        /**\n         * Called when the call fails. Defaults to `$.noop`.\n         * @type {function}\n         */\n        error: $.noop,\n\n        /**\n         * Options to pass on to the underlying transport layer. All jquery.ajax options at http://api.jquery.com/jQuery.ajax/ are available. Defaults to empty object.\n         * @type {Object}\n         */\n        transport: {}\n    };\n\n    this.sessionManager = new SessionManager();\n    var serviceOptions = this.sessionManager.getMergedOptions(defaults, config);\n    if (serviceOptions.id) {\n        serviceOptions.filter = serviceOptions.id;\n    }\n\n    function updateURLConfig(opts) {\n        var urlConfig = new ConfigService(opts).get('server');\n        if (opts.account) {\n            urlConfig.accountPath = opts.account;\n        }\n        if (opts.project) {\n            urlConfig.projectPath = opts.project;\n        }\n\n        urlConfig.filter = ';';\n        urlConfig.getFilterURL = function (filter) {\n            var url = urlConfig.getAPIPath('run');\n            var filterMatrix = qutil.toMatrixFormat(filter || opts.filter);\n\n            if (filterMatrix) {\n                url += filterMatrix + '/';\n            }\n            return url;\n        };\n\n        urlConfig.addAutoRestoreHeader = function (options) {\n            var filter = opts.filter;\n            // The semicolon separated filter is used when filter is an object\n            var isFilterRunId = filter && $.type(filter) === 'string';\n            if (opts.autoRestore && isFilterRunId) {\n                // By default autoreplay the run by sending this header to epicenter\n                // https://forio.com/epicenter/docs/public/rest_apis/aggregate_run_api/#retrieving\n                var autorestoreOpts = {\n                    headers: {\n                        'X-AutoRestore': true\n                    }\n                };\n                return $.extend(true, autorestoreOpts, options);\n            }\n\n            return options;\n        };\n        return urlConfig;\n    }\n\n    var http;\n    var httpOptions; //FIXME: Make this side-effect-less\n    function updateHTTPConfig(serviceOptions, urlConfig) {\n        httpOptions = $.extend(true, {}, serviceOptions.transport, {\n            url: urlConfig.getFilterURL\n        });\n\n        if (serviceOptions.token) {\n            httpOptions.headers = {\n                Authorization: 'Bearer ' + serviceOptions.token\n            };\n        }\n        http = new TransportFactory(httpOptions);\n        http.splitGet = rutil.splitGetFactory(httpOptions);\n    }\n\n    var urlConfig = updateURLConfig(serviceOptions); //making a function so #updateConfig can call this; change when refactored\n    updateHTTPConfig(serviceOptions, urlConfig);\n   \n\n    function setFilterOrThrowError(options) {\n        if (options.id) {\n            serviceOptions.filter = serviceOptions.id = options.id;\n        }\n        if (options.filter) {\n            serviceOptions.filter = serviceOptions.id = options.filter;\n        }\n        if (!serviceOptions.filter) {\n            throw new Error('No filter specified to apply operations against');\n        }\n    }\n\n    var publicAsyncAPI = {\n        urlConfig: urlConfig,\n\n        /**\n         * Create a new run.\n         *\n         * NOTE: Typically this is not used! Use `RunManager.getRun()` with a `strategy` of `reuse-never`, or use `RunManager.reset()`. See [Run Manager](../run-manager/) for more details.\n         *\n         *  **Example**\n         *\n         *      rs.create('hello_world.jl');\n         *\n         *  **Parameters**\n         * @param {String|Object} params If a string, the name of the primary [model file](../../../writing_your_model/). This is the one file in the project that explicitly exposes variables and methods, and it must be stored in the Model folder of your Epicenter project. If an object, may include `model`, `scope`, and `files`. (See the [Run Manager](../run_manager/) for more information on `scope` and `files`.)\n         * @param {Object} options (Optional) Overrides for configuration options.\n         * @return {Promise}\n         */\n        create: function (params, options) {\n            var createOptions = $.extend(true, {}, serviceOptions, options, { url: urlConfig.getAPIPath('run') });\n            var runApiParams = ['model', 'scope', 'files', 'ephemeral', 'cinFiles'];\n            if (typeof params === 'string') {\n                // this is just the model name\n                params = { model: params };\n            } else {\n                // whitelist the fields that we actually can send to the api\n                params = _pick(params, runApiParams);\n            }\n\n            var oldSuccess = createOptions.success;\n            createOptions.success = function (response) {\n                serviceOptions.filter = response.id; //all future chained calls to operate on this id\n                serviceOptions.id = response.id;\n                return oldSuccess.apply(this, arguments);\n            };\n\n            return http.post(params, createOptions);\n        },\n\n        /**\n         * Returns particular runs, based on conditions specified in the `qs` object.\n         *\n         * The elements of the `qs` object are ANDed together within a single call to `.query()`.\n         *\n         * **Example**\n         *\n         *      // returns runs with saved = true and variables.price > 1,\n         *      // where variables.price has been persisted (recorded)\n         *      // in the model.\n         *     rs.query({\n         *          'saved': 'true',\n         *          '.price': '>1'\n         *       },\n         *       {\n         *          startrecord: 2,\n         *          endrecord: 5\n         *       });\n         *\n         * **Parameters**\n         * @param {Object} qs Query object. Each key can be a property of the run or the name of variable that has been saved in the run (prefaced by `variables.`). Each value can be a literal value, or a comparison operator and value. (See [more on filtering](../../../rest_apis/aggregate_run_api/#filters) allowed in the underlying Run API.) Querying for variables is available for runs [in memory](../../../run_persistence/#runs-in-memory) and for runs [in the database](../../../run_persistence/#runs-in-memory) if the variables are persisted (e.g. that have been `record`ed in your model or marked for saving in your [model context file](../../../model_code/context/)).\n         * @param {Object} outputModifier (Optional) Available fields include: `startrecord`, `endrecord`, `sort`, and `direction` (`asc` or `desc`).\n         * @param {Object} options (Optional) Overrides for configuration options.\n         * @return {Promise}\n         */\n        query: function (qs, outputModifier, options) {\n            var httpOptions = $.extend(true, {}, serviceOptions, { url: urlConfig.getFilterURL(qs) }, options);\n            httpOptions = urlConfig.addAutoRestoreHeader(httpOptions);\n\n            return http.splitGet(outputModifier, httpOptions).then(function (r) {\n                return ($.isPlainObject(r) && Object.keys(r).length === 0) ? [] : r;\n            });\n        },\n\n        /**\n         * Returns particular runs, based on conditions specified in the `qs` object.\n         *\n         * Similar to `.query()`.\n         *\n         * **Parameters**\n         * @param {Object} filter Filter object. Each key can be a property of the run or the name of variable that has been saved in the run (prefaced by `variables.`). Each value can be a literal value, or a comparison operator and value. (See [more on filtering](../../../rest_apis/aggregate_run_api/#filters) allowed in the underlying Run API.) Filtering for variables is available for runs [in memory](../../../run_persistence/#runs-in-memory) and for runs [in the database](../../../run_persistence/#runs-in-memory) if the variables are persisted (e.g. that have been `record`ed in your model or marked for saving in your [model context file](../../../model_code/context/)).\n         * @param {Object} outputModifier (Optional) Available fields include: `startrecord`, `endrecord`, `sort`, and `direction` (`asc` or `desc`).\n         * @param {Object} options (Optional) Overrides for configuration options.\n         * @return {Promise}\n         */\n        filter: function (filter, outputModifier, options) {\n            if ($.isPlainObject(serviceOptions.filter)) {\n                $.extend(serviceOptions.filter, filter);\n            } else {\n                serviceOptions.filter = filter;\n            }\n            var httpOptions = $.extend(true, {}, serviceOptions, options);\n            httpOptions = urlConfig.addAutoRestoreHeader(httpOptions);\n            return http.splitGet(outputModifier, httpOptions).then(function (r) {\n                return ($.isPlainObject(r) && Object.keys(r).length === 0) ? [] : r;\n            });\n        },\n\n        /**\n         * Get data for a specific run. This includes standard run data such as the account, model, project, and created and last modified dates. To request specific model variables or run record variables, pass them as part of the `filters` parameter.\n         *\n         * Note that if the run is [in memory](../../../run_persistence/#runs-in-memory), any model variables are available; if the run is [in the database](../../../run_persistence/#runs-in-db), only model variables that have been persisted &mdash; that is, `record`ed or saved in your model &mdash; are available.\n         *\n         * **Example**\n         *\n         *     rs.load('bb589677-d476-4971-a68e-0c58d191e450', { include: ['.price', '.sales'] });\n         *\n         * **Parameters**\n         * @param {String} runID The run id.\n         * @param {Object} filters (Optional) Object containing filters and operation modifiers. Use key `include` to list model variables that you want to include in the response. Other available fields include: `startrecord`, `endrecord`, `sort`, and `direction` (`asc` or `desc`).\n         * @param {Object} options (Optional) Overrides for configuration options.\n         * @return {Promise}\n         */\n        load: function (runID, filters, options) {\n            if (runID) {\n                serviceOptions.filter = runID; //shouldn't be able to over-ride\n            }\n            var httpOptions = $.extend(true, {}, serviceOptions, options);\n            httpOptions = urlConfig.addAutoRestoreHeader(httpOptions);\n            return http.get(filters, httpOptions);\n        },\n\n        /**\n         * Removes specified runid from memory\n         *\n         * See [details on run persistence](../../../run_persistence/#runs-in-memory)\n         * @param  {String} [runID]   id of run to remove\n         * @param  {Object} [filters] (Optional) Object containing filters and operation modifiers. Use key `include` to list model variables that you want to include in the response. Other available fields include: `startrecord`, `endrecord`, `sort`, and `direction` (`asc` or `desc`).\n         * @param  {Object} [options] (Optional) Overrides for configuration options.\n         * @return {Promise}\n         */\n        removeFromMemory: function (runID, filters, options) {\n            var httpOptions = $.extend(true, {}, serviceOptions, options);\n            if (runID) {\n                httpOptions.url = urlConfig.getAPIPath('run') + runID;\n            }\n            return http.delete({}, httpOptions);\n        },\n\n        /**\n         * Save attributes (data, model variables) of the run.\n         *\n         * **Examples**\n         *\n         *     // add 'completed' field to run record\n         *     rs.save({ completed: true });\n         *\n         *     // update 'saved' field of run record, and update values of model variables for this run\n         *     rs.save({ saved: true, variables: { a: 23, b: 23 } });\n         *\n         *     // update 'saved' field of run record for a particular run\n         *     rs.save({ saved: true }, { id: '0000015bf2a04995880df6b868d23eb3d229' });\n         *\n         * **Parameters**\n         * @param {Object} attributes The run data and variables to save.\n         * @param {Object} attributes.variables Model variables must be included in a `variables` field within the `attributes` object. (Otherwise they are treated as run data and added to the run record directly.)\n         * @param {Object} options (Optional) Overrides for configuration options.\n         * @return {Promise}\n         */\n        save: function (attributes, options) {\n            var httpOptions = $.extend(true, {}, serviceOptions, options);\n            setFilterOrThrowError(httpOptions);\n            return http.patch(attributes, httpOptions);\n        },\n\n        /**\n         * Call an operation from the model.\n         *\n         * Depending on the language in which you have written your model, the operation (function or method) may need to be exposed (e.g. `export` for a Julia model) in the model file in order to be called through the API. See [Writing your Model](../../../writing_your_model/)).\n         *\n         * The `params` argument is normally an array of arguments to the `operation`. In the special case where `operation` only takes one argument, you are not required to put that argument into an array.\n         *\n         * Note that you can combine the `operation` and `params` arguments into a single object if you prefer, as in the last example.\n         *\n         * **Examples**\n         *\n         *      // operation \"solve\" takes no arguments\n         *     rs.do('solve');\n         *      // operation \"echo\" takes one argument, a string\n         *     rs.do('echo', ['hello']);\n         *      // operation \"echo\" takes one argument, a string\n         *     rs.do('echo', 'hello');\n         *      // operation \"sumArray\" takes one argument, an array\n         *     rs.do('sumArray', [[4,2,1]]);\n         *      // operation \"add\" takes two arguments, both integers\n         *     rs.do({ name:'add', params:[2,4] });\n         *      // call operation \"solve\" on a different run \n         *     rs.do('solve', { id: '0000015bf2a04995880df6b868d23eb3d229' });\n         *\n         * **Parameters**\n         * @param {String} operation Name of operation.\n         * @param {Array} params (Optional) Any parameters the operation takes, passed as an array. In the special case where `operation` only takes one argument, you are not required to put that argument into an array, and can just pass it directly.\n         * @param {Object} options (Optional) Overrides for configuration options.\n         * @return {Promise}\n         */\n        do: function (operation, params, options) {\n            // console.log('do', operation, params);\n            var opsArgs;\n            var postOptions;\n            if (options) {\n                opsArgs = params;\n                postOptions = options;\n            } else if ($.isPlainObject(params)) {\n                opsArgs = null;\n                postOptions = params;\n            } else {\n                opsArgs = params;\n            }\n            var result = rutil.normalizeOperations(operation, opsArgs);\n            var httpOptions = $.extend(true, {}, serviceOptions, postOptions);\n\n            setFilterOrThrowError(httpOptions);\n\n            var prms = (result.args[0].length && (result.args[0] !== null && result.args[0] !== undefined)) ? result.args[0] : [];\n            return http.post({ arguments: prms }, $.extend(true, {}, httpOptions, {\n                url: urlConfig.getFilterURL() + 'operations/' + result.ops[0] + '/'\n            }));\n        },\n\n        /**\n         * Call several operations from the model, sequentially.\n         *\n         * Depending on the language in which you have written your model, the operation (function or method) may need to be exposed (e.g. `export` for a Julia model) in the model file in order to be called through the API. See [Writing your Model](../../../writing_your_model/)).\n         *\n         * **Examples**\n         *\n         *      // operations \"initialize\" and \"solve\" do not take any arguments\n         *     rs.serial(['initialize', 'solve']);\n         *      // operations \"init\" and \"reset\" take two arguments each\n         *     rs.serial([  { name: 'init', params: [1,2] },\n         *                  { name: 'reset', params: [2,3] }]);\n         *      // operation \"init\" takes two arguments,\n         *      // operation \"runmodel\" takes none\n         *     rs.serial([  { name: 'init', params: [1,2] },\n         *                  { name: 'runmodel', params: [] }]);\n         *\n         * **Parameters**\n         * @param {Array} operations If none of the operations take parameters, pass an array of the operation names (strings). If any of the operations do take parameters, pass an array of objects, each of which contains an operation name and its own (possibly empty) array of parameters.\n         * @param {*} params Parameters to pass to operations.\n         * @param {Object} options (Optional) Overrides for configuration options.\n         * @return {Promise} The parameter to the callback is an array. Each array element is an object containing the results of one operation.\n         */\n        serial: function (operations, params, options) {\n            var opParams = rutil.normalizeOperations(operations, params);\n            var ops = opParams.ops;\n            var args = opParams.args;\n            var me = this;\n\n            var $d = $.Deferred();\n            var postOptions = $.extend(true, {}, serviceOptions, options);\n\n            var responses = [];\n            var doSingleOp = function () {\n                var op = ops.shift();\n                var arg = args.shift();\n\n                me.do(op, arg, {\n                    success: function (result) {\n                        responses.push(result);\n                        if (ops.length) {\n                            doSingleOp();\n                        } else {\n                            $d.resolve(responses);\n                            postOptions.success(responses, me);\n                        }\n                    },\n                    error: function (err) {\n                        responses.push(err);\n                        $d.reject(responses);\n                        postOptions.error(responses, me);\n                    }\n                });\n            };\n\n            doSingleOp();\n\n            return $d.promise();\n        },\n\n        /**\n         * Call several operations from the model, executing them in parallel.\n         *\n         * Depending on the language in which you have written your model, the operation (function or method) may need to be exposed (e.g. `export` for a Julia model) in the model file in order to be called through the API. See [Writing your Model](../../../writing_your_model/)).\n         *\n         * **Example**\n         *\n         *      // operations \"solve\" and \"reset\" do not take any arguments\n         *     rs.parallel(['solve', 'reset']);\n         *      // operations \"add\" and \"subtract\" take two arguments each\n         *     rs.parallel([ { name: 'add', params: [1,2] },\n         *                   { name: 'subtract', params:[2,3] }]);\n         *      // operations \"add\" and \"subtract\" take two arguments each\n         *     rs.parallel({ add: [1,2], subtract: [2,4] });\n         *\n         * **Parameters**\n         * @param {Array|Object} operations If none of the operations take parameters, pass an array of the operation names (as strings). If any of the operations do take parameters, you have two options. You can pass an array of objects, each of which contains an operation name and its own (possibly empty) array of parameters. Alternatively, you can pass a single object with the operation name and a (possibly empty) array of parameters.\n         * @param {*} params Parameters to pass to operations.\n         * @param {Object} options (Optional) Overrides for configuration options.\n         * @return {Promise} The parameter to the callback is an array. Each array element is an object containing the results of one operation.\n         */\n        parallel: function (operations, params, options) {\n            var $d = $.Deferred();\n\n            var opParams = rutil.normalizeOperations(operations, params);\n            var ops = opParams.ops;\n            var args = opParams.args;\n            var postOptions = $.extend(true, {}, serviceOptions, options);\n\n            var queue = [];\n            for (var i = 0; i < ops.length; i++) {\n                queue.push(\n                    this.do(ops[i], args[i])\n                );\n            }\n\n            var me = this;\n            $.when.apply(this, queue)\n                .then(function () {\n                    var args = Array.prototype.slice.call(arguments);\n                    var actualResponse = args.map(function (a) {\n                        return a[0];\n                    });\n                    $d.resolve(actualResponse);\n                    postOptions.success(actualResponse, me);\n                })\n                .fail(function () {\n                    var args = Array.prototype.slice.call(arguments);\n                    var actualResponse = args.map(function (a) {\n                        return a[0];\n                    });\n                    $d.reject(actualResponse);\n                    postOptions.error(actualResponse, me);\n                });\n\n            return $d.promise();\n        },\n\n        /**\n         * Shortcut to using the [Introspection API Service](../introspection-api-service/). Allows you to view a list of the variables and operations in a model.\n         *\n         * **Example**\n         *\n         *     rs.introspect({ runID: 'cbf85437-b539-4977-a1fc-23515cf071bb' }).then(function (data) {\n         *          console.log(data.functions);\n         *          console.log(data.variables);\n         *     });\n         *\n         * **Parameters**\n         * @param  {Object} options Options can either be of the form `{ runID: <runid> }` or `{ model: <modelFileName> }`. Note that the `runID` is optional if the Run Service is already associated with a particular run (because `id` was passed in when the Run Service was initialized). If provided, the `runID` overrides the `id` currently associated with the Run Service.\n         * @param  {Object} introspectionConfig (Optional) Service options for Introspection Service\n         * @return {Promise}\n         */\n        introspect: function (options, introspectionConfig) {\n            var introspection = new IntrospectionService($.extend(true, {}, serviceOptions, introspectionConfig));\n            if (options) {\n                if (options.runID) {\n                    return introspection.byRunID(options.runID);\n                } else if (options.model) {\n                    return introspection.byModel(options.model);\n                }\n            } else if (serviceOptions.id) {\n                return introspection.byRunID(serviceOptions.id);\n            } else {\n                throw new Error('Please specify either the model or runid to introspect');\n            }\n        }\n    };\n\n    var publicSyncAPI = {\n        getCurrentConfig: function () {\n            return serviceOptions;\n        },\n        updateConfig: function (config) {\n            if (config && config.id) {\n                config.filter = config.id;\n            } else if (config && config.filter) {\n                config.id = config.filter;\n            }\n            serviceOptions = $.extend(true, {}, serviceOptions, config);\n            urlConfig = updateURLConfig(serviceOptions);\n            this.urlConfig = urlConfig;\n            updateHTTPConfig(serviceOptions, urlConfig);\n        },\n        /**\n          * Returns a Variables Service instance. Use the variables instance to load, save, and query for specific model variables. See the [Variable API Service](../variables-api-service/) for more information.\n          *\n          * **Example**\n          *\n          *      var vs = rs.variables();\n          *      vs.save({ sample_int: 4 });\n          *\n          * **Parameters**\n          * @param {Object} config (Optional) Overrides for configuration options.\n          * @return {Object} variablesService Instance\n          */\n        variables: function (config) {\n            var vs = new VariablesService($.extend(true, {}, serviceOptions, config, {\n                runService: this\n            }));\n            return vs;\n        }\n    };\n\n    $.extend(this, publicAsyncAPI);\n    $.extend(this, publicSyncAPI);\n};\n","'use strict';\n\nvar ConfigService = require('./configuration-service');\nvar SessionManager = require('../store/session-manager');\nvar objectAssign = require('object-assign');\n\nvar serviceUtils = {\n    /*\n    * Gets the default options for a api service.\n    * It will merge:\n    * - The Session options (Using the Session Manager)\n    * - The Authorization Header from the token option\n    * - The full url from the endpoint option\n    * With the supplied overrides and defaults\n    *\n    */\n    getDefaultOptions: function (defaults) {\n        var rest = Array.prototype.slice.call(arguments, 1);\n        var sessionManager = new SessionManager();\n        var serviceOptions = sessionManager.getMergedOptions.apply(sessionManager, [defaults].concat(rest));\n\n        serviceOptions.transport = objectAssign({}, serviceOptions.transport, {\n            url: this.getApiUrl(serviceOptions.apiEndpoint, serviceOptions)\n        });\n\n        if (serviceOptions.token) {\n            serviceOptions.transport.headers = {\n                Authorization: 'Bearer ' + serviceOptions.token\n            };\n        }\n        return serviceOptions;\n    },\n\n    getApiUrl: function (apiEndpoint, serviceOptions) {\n        var urlConfig = new ConfigService(serviceOptions).get('server');\n        return urlConfig.getAPIPath(apiEndpoint);\n    }\n};\n\nmodule.exports = serviceUtils;","'use strict';\n/**\n * ## State API Adapter\n *\n * The State API Adapter allows you to view the history of a run, and to replay or clone runs. \n *\n * The State API Adapter brings existing, persisted run data from the database back into memory, using the same run id (`replay`) or a new run id (`clone`). Runs must be in memory in order for you to update variables or call operations on them.\n *\n * Specifically, the State API Adapter works by \"re-running\" the run (user interactions) from the creation of the run up to the time it was last persisted in the database. This process uses the current version of the run's model. Therefore, if the model has changed since the original run was created, the retrieved run will use the new model — and may end up having different values or behavior as a result. Use with care!\n *\n * To use the State API Adapter, instantiate it and then call its methods:\n *\n *      var sa = new F.service.State();\n *      sa.replay({runId: '1842bb5c-83ad-4ba8-a955-bd13cc2fdb4f'});\n *\n * The constructor takes an optional `options` parameter in which you can specify the `account` and `project` if they are not already available in the current context.\n *\n */\n\nvar ConfigService = require('./configuration-service');\nvar TransportFactory = require('../transport/http-transport-factory');\nvar _pick = require('../util/object-util')._pick;\nvar SessionManager = require('../store/session-manager');\nvar apiEndpoint = 'model/state';\n\nmodule.exports = function (config) {\n\n    var defaults = {\n\n    };\n\n    this.sessionManager = new SessionManager();\n    var serviceOptions = this.sessionManager.getMergedOptions(defaults, config);\n    var urlConfig = new ConfigService(serviceOptions).get('server');\n\n    var transportOptions = $.extend(true, {}, serviceOptions.transport, {\n        url: urlConfig.getAPIPath(apiEndpoint)\n    });\n\n    if (serviceOptions.token) {\n        transportOptions.headers = {\n            Authorization: 'Bearer ' + serviceOptions.token\n        };\n    }\n\n    var http = new TransportFactory(transportOptions);\n    var parseRunIdOrError = function (params) {\n        if ($.isPlainObject(params) && params.runId) {\n            return params.runId;\n        } else {\n            throw new Error('Please pass in a run id');\n        }\n    };\n\n    var publicAPI = {\n\n        /**\n        * View the history of a run.\n        * \n        *  **Example**\n        *\n        *      var sa = new F.service.State();\n        *      sa.load('0000015a06bb58613b28b57365677ec89ec5').then(function(history) {\n        *            console.log('history = ', history);\n        *      });\n        *\n        *  **Parameters**\n        * @param {string} runId The id of the run.\n        * @param {object} options (Optional) Overrides for configuration options.\n        * @return {Promise}\n        */\n        load: function (runId, options) {\n            var httpParams = $.extend(true, {},\n                serviceOptions,\n                options,\n                { url: urlConfig.getAPIPath(apiEndpoint) + runId }\n            );\n            return http.get('', httpParams);\n        },\n\n        /**\n        * Replay a run. After this call, the run, with its original run id, is now available [in memory](../../../run_persistence/#runs-in-memory). (It continues to be persisted into the Epicenter database at regular intervals.)\n        *\n        *  **Example**\n        *\n        *      var sa = new F.service.State();\n        *      sa.replay({runId: '1842bb5c-83ad-4ba8-a955-bd13cc2fdb4f', stopBefore: 'calculateScore'});\n        *\n        *  **Parameters**\n        * @param {object} params Parameters object.\n        * @param {string} params.runId The id of the run to bring back to memory.\n        * @param {string} params.stopBefore (Optional) The run is advanced only up to the first occurrence of this method.\n        * @param {array} params.exclude (Optional) Array of methods to exclude when advancing the run.\n        * @param {object} options (Optional) Overrides for configuration options.\n        * @return {Promise}\n        */\n        replay: function (params, options) {\n            var runId = parseRunIdOrError(params);\n\n            var replayOptions = $.extend(true, {},\n                serviceOptions,\n                options,\n                { url: urlConfig.getAPIPath(apiEndpoint) + runId }\n            );\n\n            params = $.extend(true, { action: 'replay' }, _pick(params, ['stopBefore', 'exclude']));\n\n            return http.post(params, replayOptions);\n        },\n\n        /**\n        * Clone a given run and return a new run in the same state as the given run.\n        *\n        * The new run id is now available [in memory](../../../run_persistence/#runs-in-memory). The new run includes a copy of all of the data from the original run, EXCEPT:\n        *\n        * * The `saved` field in the new run record is not copied from the original run record. It defaults to `false`.\n        * * The `initialized` field in the new run record is not copied from the original run record. It defaults to `false` but may change to `true` as the new run is advanced. For example, if there has been a call to the `step` function (for Vensim models), the `initialized` field is set to `true`.\n        * * The `created` field in the new run record is the date and time at which the clone was created (not the time that the original run was created.)\n        *\n        * The original run remains only [in the database](../../../run_persistence/#runs-in-db).\n        *\n        *  **Example**\n        *\n        *      var sa = new F.service.State();\n        *      sa.clone({runId: '1842bb5c-83ad-4ba8-a955-bd13cc2fdb4f', stopBefore: 'calculateScore', exclude: ['interimCalculation'] });\n        *\n        *  **Parameters**\n        * @param {object} params Parameters object.\n        * @param {string} params.runId The id of the run to clone from memory.\n        * @param {string} params.stopBefore (Optional) The newly cloned run is advanced only up to the first occurrence of this method.\n        * @param {array} params.exclude (Optional) Array of methods to exclude when advancing the newly cloned run.\n        * @param {object} options (Optional) Overrides for configuration options.\n        * @return {Promise}\n        */\n        clone: function (params, options) {\n            var runId = parseRunIdOrError(params);\n\n            var replayOptions = $.extend(true, {},\n                serviceOptions,\n                options,\n                { url: urlConfig.getAPIPath(apiEndpoint) + runId }\n            );\n\n            params = $.extend(true, { action: 'clone' }, _pick(params, ['stopBefore', 'exclude']));\n\n            return http.post(params, replayOptions);\n        }\n    };\n\n    $.extend(this, publicAPI);\n};\n","'use strict';\n\nvar epiVersion = require('../api-version.json');\n\n//TODO: urlutils to get host, since no window on node\nvar defaults = {\n    host: window.location.host,\n    pathname: window.location.pathname\n};\n\nfunction getLocalHost(existingFn, host) {\n    var localHostFn;\n    if (existingFn !== undefined) {\n        if (!$.isFunction(existingFn)) {\n            localHostFn = function () { return existingFn; };\n        } else {\n            localHostFn = existingFn;\n        }\n    } else {\n        localHostFn = function () {\n            var isLocal = !host || //phantomjs\n                host === '127.0.0.1' || \n                host.indexOf('local.') === 0 || \n                host.indexOf('localhost') === 0;\n            return isLocal;\n        };\n    }\n    return localHostFn;\n}\n\nvar UrlConfigService = function (config) {\n    var envConf = UrlConfigService.defaults;\n\n    if (!config) {\n        config = {};\n    }\n    // console.log(this.defaults);\n    var overrides = $.extend({}, envConf, config);\n    var options = $.extend({}, defaults, overrides);\n\n    overrides.isLocalhost = options.isLocalhost = getLocalHost(options.isLocalhost, options.host);\n    \n    // console.log(isLocalhost(), '___________');\n    var actingHost = config && config.host;\n    if (!actingHost && options.isLocalhost()) {\n        actingHost = 'forio.com';\n    } else {\n        actingHost = options.host;\n    }\n\n    var API_PROTOCOL = 'https';\n    var HOST_API_MAPPING = {\n        'forio.com': 'api.forio.com',\n        'foriodev.com': 'api.epicenter.foriodev.com'\n    };\n\n    var publicExports = {\n        protocol: API_PROTOCOL,\n\n        api: '',\n\n        //TODO: this should really be called 'apihost', but can't because that would break too many things\n        host: (function () {\n            var apiHost = (HOST_API_MAPPING[actingHost]) ? HOST_API_MAPPING[actingHost] : actingHost;\n            // console.log(actingHost, config, apiHost);\n            return apiHost;\n        }()),\n\n        isCustomDomain: (function () {\n            var path = options.pathname.split('\\/');\n            var pathHasApp = path && path[1] === 'app';\n            return (!options.isLocalhost() && !pathHasApp);\n        }()),\n\n        appPath: (function () {\n            var path = options.pathname.split('\\/');\n\n            return path && path[1] || '';\n        }()),\n\n        accountPath: (function () {\n            var accnt = '';\n            var path = options.pathname.split('\\/');\n            if (path && path[1] === 'app') {\n                accnt = path[2];\n            }\n            return accnt;\n        }()),\n\n        projectPath: (function () {\n            var prj = '';\n            var path = options.pathname.split('\\/');\n            if (path && path[1] === 'app') {\n                prj = path[3]; //eslint-disable-line no-magic-numbers\n            }\n            return prj;\n        }()),\n\n        versionPath: (function () {\n            var version = epiVersion.version ? epiVersion.version + '/' : '';\n            return version;\n        }()),\n\n        getAPIPath: function (api) {\n            var PROJECT_APIS = ['run', 'data', 'file', 'presence'];\n            var apiMapping = {\n                channel: 'channel/subscribe'\n            };\n            var apiEndpoint = apiMapping[api] || api;\n            \n            if (apiEndpoint === 'config') {\n                var actualProtocol = window.location.protocol.replace(':', '');\n                var configProtocol = (options.isLocalhost()) ? this.protocol : actualProtocol;\n                return configProtocol + '://' + actingHost + '/epicenter/' + this.versionPath + 'config';\n            }\n            var apiPath = this.protocol + '://' + this.host + '/' + this.versionPath + apiEndpoint + '/';\n\n            if ($.inArray(apiEndpoint, PROJECT_APIS) !== -1) {\n                apiPath += this.accountPath + '/' + this.projectPath + '/';\n            }\n            return apiPath;\n        }\n    };\n\n\n    $.extend(publicExports, overrides);\n    return publicExports;\n};\n// This data can be set by external scripts, for loading from an env server for eg;\nUrlConfigService.defaults = {};\n\nmodule.exports = UrlConfigService;\n","'use strict';\n/**\n* ## User API Adapter\n*\n* The User API Adapter allows you to retrieve details about end users in your team (account). It is based on the querying capabilities of the underlying RESTful [User API](../../../rest_apis/user_management/user/).\n*\n* To use the User API Adapter, instantiate it and then call its methods.\n*\n*       var ua = new F.service.User({\n*           account: 'acme-simulations',\n*           token: 'user-or-project-access-token'\n*       });\n*       ua.getById('42836d4b-5b61-4fe4-80eb-3136e956ee5c');\n*       ua.get({ userName: 'jsmith' });\n*       ua.get({ id: ['42836d4b-5b61-4fe4-80eb-3136e956ee5c',\n*                   '4ea75631-4c8d-4872-9d80-b4600146478e'] });\n*\n* The constructor takes an optional `options` parameter in which you can specify the `account` and `token` if they are not already available in the current context.\n*/\n\nvar ConfigService = require('./configuration-service');\nvar TransportFactory = require('../transport/http-transport-factory');\nvar SessionManager = require('../store/session-manager');\nvar qutil = require('../util/query-util');\n\nmodule.exports = function (config) {\n    var defaults = {\n\n        /**\n         * The account id. In the Epicenter UI, this is the **Team ID** (for team projects) or **User ID** (for personal projects). Defaults to empty string.\n         * @type {String}\n         */\n        account: undefined,\n\n        /**\n         * The access token to use when searching for end users. (See [more background on access tokens](../../../project_access/)).\n         * @type {String}\n         */\n        token: undefined,\n\n        /**\n         * Options to pass on to the underlying transport layer. All jquery.ajax options at http://api.jquery.com/jQuery.ajax/ are available. Defaults to empty object.\n         * @type {Object}\n         */\n        transport: {}\n    };\n\n    this.sessionManager = new SessionManager();\n    var serviceOptions = this.sessionManager.getMergedOptions(defaults, config);\n    var urlConfig = new ConfigService(serviceOptions).get('server');\n    var transportOptions = $.extend(true, {}, serviceOptions.transport, {\n        url: urlConfig.getAPIPath('user')\n    });\n\n    if (serviceOptions.token) {\n        transportOptions.headers = {\n            Authorization: 'Bearer ' + serviceOptions.token\n        };\n    }\n\n    var http = new TransportFactory(transportOptions);\n\n    var publicAPI = {\n\n        /**\n        * Retrieve details about particular end users in your team, based on user name or user id.\n        *\n        * **Example**\n        *\n        *       var ua = new F.service.User({\n        *           account: 'acme-simulations',\n        *           token: 'user-or-project-access-token'\n        *       });\n        *       ua.get({ userName: 'jsmith' });\n        *       ua.get({ id: ['42836d4b-5b61-4fe4-80eb-3136e956ee5c',\n        *                   '4ea75631-4c8d-4872-9d80-b4600146478e'] });\n        *\n        * **Parameters**\n        * @param {object} filter Object with field `userName` and value of the username. Alternatively, object with field `id` and value of an array of user ids.\n        * @param {object} options (Optional) Overrides for configuration options.\n        * @return {Promise}\n        */\n\n        get: function (filter, options) {\n            options = options || {};\n            filter = filter || {};\n\n            var getOptions = $.extend(true, {},\n                serviceOptions,\n                options\n            );\n\n            var toQFilter = function (filter) {\n                var res = {};\n\n                // API only supports filtering by username for now\n                if (filter.userName) {\n                    res.q = filter.userName;\n                }\n\n                return res;\n            };\n\n            var toIdFilters = function (id) {\n                if (!id) {\n                    return '';\n                }\n\n                id = $.isArray(id) ? id : [id];\n                return 'id=' + id.join('&id=');\n            };\n\n            var getFilters = [\n                'account=' + getOptions.account,\n                toIdFilters(filter.id),\n                qutil.toQueryFormat(toQFilter(filter))\n            ].join('&');\n\n            // special case for queries with large number of ids\n            // make it as a post with GET semantics\n            var threshold = 30;\n            if (filter.id && $.isArray(filter.id) && filter.id.length >= threshold) {\n                getOptions.url = urlConfig.getAPIPath('user') + '?_method=GET';\n                return http.post({ id: filter.id }, getOptions);\n            } else {\n                return http.get(getFilters, getOptions);\n            }\n        },\n\n        /**\n        * Retrieve details about a single end user in your team, based on user id.\n        *\n        * **Example**\n        *\n        *       var ua = new F.service.User({\n        *           account: 'acme-simulations',\n        *           token: 'user-or-project-access-token'\n        *       });\n        *       ua.getById('42836d4b-5b61-4fe4-80eb-3136e956ee5c');\n        *\n        * **Parameters**\n        * @param {string} userId The user id for the end user in your team.\n        * @param {object} options (Optional) Overrides for configuration options.\n        * @return {Promise}\n        */\n\n        getById: function (userId, options) {\n            return publicAPI.get({ id: userId }, options);\n        }\n    };\n\n    $.extend(this, publicAPI);\n};\n\n\n","/**\n *\n * ## Variables API Service\n *\n * Used in conjunction with the [Run API Service](../run-api-service/) to read, write, and search for specific model variables.\n *\n *     var rm = new F.manager.RunManager({\n *           run: {\n *               account: 'acme-simulations',\n *               project: 'supply-chain-game',\n *               model: 'supply-chain-model.jl'\n *           }\n *      });\n *     rm.getRun()\n *       .then(function() {\n *          var vs = rm.run.variables();\n *          vs.save({sample_int: 4});\n *        });\n *\n */\n\n\n 'use strict';\n\n var TransportFactory = require('../transport/http-transport-factory');\n var rutil = require('../util/run-util');\n\n module.exports = function (config) {\n     var defaults = {\n        /**\n         * The runs object to which the variable filters apply. Defaults to null.\n         * @type {runService}\n         */\n         runService: null\n     };\n     var serviceOptions = $.extend({}, defaults, config);\n\n     var getURL = function () {\n        //TODO: Replace with getCurrentconfig instead?\n         return serviceOptions.runService.urlConfig.getFilterURL() + 'variables/';\n     };\n\n     var addAutoRestoreHeader = function (options) {\n         return serviceOptions.runService.urlConfig.addAutoRestoreHeader(options);\n     };\n\n     var httpOptions = {\n         url: getURL\n     };\n     if (serviceOptions.token) {\n         httpOptions.headers = {\n             Authorization: 'Bearer ' + serviceOptions.token\n         };\n     }\n     var http = new TransportFactory(httpOptions);\n     http.splitGet = rutil.splitGetFactory(httpOptions);\n\n     var publicAPI = {\n\n        /**\n         * Get values for a variable.\n         *\n         * **Example**\n         *\n         *      vs.load('sample_int')\n         *          .then(function(val){\n         *              // val contains the value of sample_int\n         *          });\n         *\n         * **Parameters**\n         * @param {String} variable Name of variable to load.\n         * @param {Object} outputModifier (Optional) Available fields include: `startrecord`, `endrecord`, `sort`, and `direction` (`asc` or `desc`).\n         * @param {Object} options (Optional) Overrides for configuration options.\n         * @return {Promise}\n         */\n         load: function (variable, outputModifier, options) {\n             var httpOptions = $.extend(true, {}, serviceOptions, options);\n             httpOptions = addAutoRestoreHeader(httpOptions);\n             return http.get(outputModifier, $.extend({}, httpOptions, {\n                 url: getURL() + variable + '/'\n             }));\n         },\n\n        /**\n         * Returns particular variables, based on conditions specified in the `query` object.\n         *\n         * **Example**\n         *\n         *      vs.query(['price', 'sales'])\n         *          .then(function(val) {\n         *              // val is an object with the values of the requested variables: val.price, val.sales\n         *          });\n         *\n         *      vs.query({ include:['price', 'sales'] });\n         *\n         * **Parameters**\n         * @param {Object|Array} query The names of the variables requested.\n         * @param {Object} outputModifier (Optional) Available fields include: `startrecord`, `endrecord`, `sort`, and `direction` (`asc` or `desc`).\n         * @param {Object} options (Optional) Overrides for configuration options.\n         * @return {Promise}\n         */\n         query: function (query, outputModifier, options) {\n            //Query and outputModifier are both querystrings in the url; only calling them out separately here to be consistent with the other calls\n             var httpOptions = $.extend(true, {}, serviceOptions, options);\n             httpOptions = addAutoRestoreHeader(httpOptions);\n\n             if ($.isArray(query)) {\n                 query = { include: query };\n             }\n             $.extend(query, outputModifier);\n             return http.splitGet(query, httpOptions);\n         },\n\n        /**\n         * Save values to model variables. Overwrites existing values. Note that you can only update model variables if the run is [in memory](../../../run_persistence/#runs-in-memory). (An alternate way to update model variables is to call a method from the model and make sure that the method persists the variables. See `do`, `serial`, and `parallel` in the [Run API Service](../run-api-service/) for calling methods from the model.)\n         *\n         * **Example**\n         *\n         *      vs.save('price', 4);\n         *      vs.save({ price: 4, quantity: 5, products: [2,3,4] });\n         *\n         * **Parameters**\n         * @param {Object|String} variable An object composed of the model variables and the values to save. Alternatively, a string with the name of the variable.\n         * @param {Object} val (Optional) If passing a string for `variable`, use this argument for the value to save.\n         * @param {Object} options (Optional) Overrides for configuration options.\n         * @return {Promise}\n         */\n         save: function (variable, val, options) {\n             var attrs;\n             if (typeof variable === 'object') {\n                 attrs = variable;\n                 options = val;\n             } else {\n                 (attrs = {})[variable] = val;\n             }\n             var httpOptions = $.extend(true, {}, serviceOptions, options);\n\n             return http.patch.call(this, attrs, httpOptions);\n         }\n\n        // Not Available until underlying API supports PUT. Otherwise save would be PUT and merge would be PATCH\n        // *\n        //  * Save values to the api. Merges arrays, but otherwise same as save\n        //  * @param {Object|String} variable Object with attributes, or string key\n        //  * @param {Object} val Optional if prev parameter was a string, set value here\n        //  * @param {Object} options Overrides for configuration options\n        //  *\n        //  * @example\n        //  *     vs.merge({ price: 4, quantity: 5, products: [2,3,4] })\n        //  *     vs.merge('price', 4);\n\n        // merge: function (variable, val, options) {\n        //     var attrs;\n        //     if (typeof variable === 'object') {\n        //       attrs = variable;\n        //       options = val;\n        //     } else {\n        //       (attrs = {})[variable] = val;\n        //     }\n        //     var httpOptions = $.extend(true, {}, serviceOptions, options);\n\n        //     return http.patch.call(this, attrs, httpOptions);\n        // }\n     };\n     $.extend(this, publicAPI);\n };\n","/**\n * ## World API Adapter\n *\n * A [run](../../../glossary/#run) is a collection of end user interactions with a project and its model -- including setting variables, making decisions, and calling operations. For building multiplayer simulations you typically want multiple end users to share the same set of interactions, and work within a common state. Epicenter allows you to create \"worlds\" to handle such cases. Only [team projects](../../../glossary/#team) can be multiplayer.\n *\n * The World API Adapter allows you to create, access, and manipulate multiplayer worlds within your Epicenter project. You can use this to add and remove end users from the world, and to create, access, and remove their runs. Because of this, typically the World Adapter is used for facilitator pages in your project. (The related [World Manager](../world-manager/) provides an easy way to access runs and worlds for particular end users, so is typically used in pages that end users will interact with.)\n *\n * As with all the other [API Adapters](../../), all methods take in an \"options\" object as the last parameter. The options can be used to extend/override the World API Service defaults.\n *\n * To use the World Adapter, instantiate it and then access the methods provided. Instantiating requires the account id (**Team ID** in the Epicenter user interface), project id (**Project ID**), and group (**Group Name**).\n *\n *       var wa = new F.service.World({\n *          account: 'acme-simulations',\n *          project: 'supply-chain-game',\n *          group: 'team1' });\n *       wa.create()\n *          .then(function(world) {\n *              // call methods, e.g. wa.addUsers()\n *          });\n */\n\n'use strict';\n\nvar ConfigService = require('./configuration-service');\n// var qutil = require('../util/query-util');\nvar TransportFactory = require('../transport/http-transport-factory');\nvar SessionManager = require('../store/session-manager');\nvar _pick = require('../util/object-util')._pick;\n\nvar apiBase = 'multiplayer/';\nvar assignmentEndpoint = apiBase + 'assign';\nvar apiEndpoint = apiBase + 'world';\nvar projectEndpoint = apiBase + 'project';\n\nmodule.exports = function (config) {\n    var defaults = {\n        /**\n         * For projects that require authentication, pass in the user access token (defaults to empty string). If the user is already logged in to Epicenter, the user access token is already set in a cookie and automatically loaded from there. (See [more background on access tokens](../../../project_access/)).\n         * @see [Authentication API Service](../auth-api-service/) for getting tokens.\n         * @type {String}\n         */\n        token: undefined,\n\n        /**\n         * The project id. If left undefined, taken from the URL.\n         * @type {String}\n         */\n        project: undefined,\n\n        /**\n         * The account id. In the Epicenter UI, this is the **Team ID** (for team projects). If left undefined, taken from the URL.\n         * @type {String}\n         */\n        account: undefined,\n\n        /**\n         * The group name. Defaults to undefined.\n         * @type {String}\n         */\n        group: undefined,\n\n        /**\n         * The model file to use to create runs in this world. Defaults to undefined.\n         * @type {String}\n         */\n        model: undefined,\n\n        /**\n         * Criteria by which to filter world. Currently only supports world-ids as filters.\n         * @type {String}\n         */\n        filter: '',\n\n        /**\n         * Convenience alias for filter\n         * @type {String}\n         */\n        id: '',\n\n        /**\n         * Options to pass on to the underlying transport layer. All jquery.ajax options at http://api.jquery.com/jQuery.ajax/ are available. Defaults to empty object.\n         * @type {Object}\n         */\n        transport: {},\n\n        /**\n         * Called when the call completes successfully. Defaults to `$.noop`.\n         * @type {function}\n         */\n        success: $.noop,\n\n        /**\n         * Called when the call fails. Defaults to `$.noop`.\n         * @type {function}\n         */\n        error: $.noop\n    };\n\n    this.sessionManager = new SessionManager();\n    var serviceOptions = this.sessionManager.getMergedOptions(defaults, config);\n    if (serviceOptions.id) {\n        serviceOptions.filter = serviceOptions.id;\n    }\n\n    var urlConfig = new ConfigService(serviceOptions).get('server');\n\n    if (!serviceOptions.account) {\n        serviceOptions.account = urlConfig.accountPath;\n    }\n\n    if (!serviceOptions.project) {\n        serviceOptions.project = urlConfig.projectPath;\n    }\n\n    var transportOptions = $.extend(true, {}, serviceOptions.transport, {\n        url: urlConfig.getAPIPath(apiEndpoint)\n    });\n\n    if (serviceOptions.token) {\n        transportOptions.headers = {\n            Authorization: 'Bearer ' + serviceOptions.token\n        };\n    }\n\n    var http = new TransportFactory(transportOptions);\n\n    var setIdFilterOrThrowError = function (options) {\n        if (options.id) {\n            serviceOptions.filter = options.id;\n        }\n        if (options.filter) {\n            serviceOptions.filter = options.filter;\n        }\n        if (!serviceOptions.filter) {\n            throw new Error('No world id specified to apply operations against. This could happen if the user is not assigned to a world and is trying to work with runs from that world.');\n        }\n    };\n\n    var validateModelOrThrowError = function (options) {\n        if (!options.model) {\n            throw new Error('No model specified to get the current run');\n        }\n    };\n\n    var publicAPI = {\n\n        /**\n        * Creates a new World.\n        *\n        * Using this method is rare. It is more common to create worlds automatically while you `autoAssign()` end users to worlds. (In this case, configuration data for the world, such as the roles, are read from the project-level world configuration information, for example by `getProjectSettings()`.)\n        *\n        *  **Example**\n        *\n        *      var wa = new F.service.World({\n        *           account: 'acme-simulations',\n        *           project: 'supply-chain-game',\n        *           group: 'team1' });\n        *      wa.create({\n        *           roles: ['VP Marketing', 'VP Sales', 'VP Engineering']\n        *       });\n        *\n        *  **Parameters**\n        * @param {object} params Parameters to create the world.\n        * @param {string} params.group (Optional) The **Group Name** to create this world under. Only end users in this group are eligible to join the world. Optional here; required when instantiating the service (`new F.service.World()`).\n        * @param {object} params.roles (Optional) The list of roles (strings) for this world. Some worlds have specific roles that **must** be filled by end users. Listing the roles allows you to autoassign users to worlds and ensure that all roles are filled in each world.\n        * @param {object} params.optionalRoles (Optional) The list of optional roles (strings) for this world. Some worlds have specific roles that **may** be filled by end users. Listing the optional roles as part of the world object allows you to autoassign users to worlds and ensure that all roles are filled in each world.\n        * @param {integer} params.minUsers (Optional) The minimum number of users for the world. Including this number allows you to autoassign end users to worlds and ensure that the correct number of users are in each world.\n        * @param {object} options (Optional) Options object to override global options.\n        * @return {Promise}\n        */\n        create: function (params, options) {\n            var createOptions = $.extend(true, {}, serviceOptions, options, { url: urlConfig.getAPIPath(apiEndpoint) });\n            var worldApiParams = ['scope', 'files', 'roles', 'optionalRoles', 'minUsers', 'group', 'name'];\n            var validParams = _pick(serviceOptions, ['account', 'project', 'group']);\n            // whitelist the fields that we actually can send to the api\n            params = _pick(params, worldApiParams);\n\n            // account and project go in the body, not in the url\n            params = $.extend({}, validParams, params);\n\n            var oldSuccess = createOptions.success;\n            createOptions.success = function (response) {\n                serviceOptions.filter = response.id; //all future chained calls to operate on this id\n                return oldSuccess.apply(this, arguments);\n            };\n\n            return http.post(params, createOptions);\n        },\n\n        /**\n        * Updates a World, for example to replace the roles in the world.\n        *\n        * Typically, you complete world configuration at the project level, rather than at the world level. For example, each world in your project probably has the same roles for end users. And your project is probably either configured so that all end users share the same world (and run), or smaller sets of end users share worlds — but not both. However, this method is available if you need to update the configuration of a particular world.\n        *\n        *  **Example**\n        *\n        *      var wa = new F.service.World({\n        *           account: 'acme-simulations',\n        *           project: 'supply-chain-game',\n        *           group: 'team1' });\n        *      wa.create()\n        *           .then(function(world) {\n        *               wa.update({ roles: ['VP Marketing', 'VP Sales', 'VP Engineering'] });\n        *           });\n        *\n        *  **Parameters**\n        * @param {object} params Parameters to update the world.\n        * @param {string} params.name A string identifier for the linked end users, for example, \"name\": \"Our Team\".\n        * @param {object} params.roles (Optional) The list of roles (strings) for this world. Some worlds have specific roles that **must** be filled by end users. Listing the roles allows you to autoassign users to worlds and ensure that all roles are filled in each world.\n        * @param {object} params.optionalRoles (Optional) The list of optional roles (strings) for this world. Some worlds have specific roles that **may** be filled by end users. Listing the optional roles as part of the world object allows you to autoassign users to worlds and ensure that all roles are filled in each world.\n        * @param {integer} params.minUsers (Optional) The minimum number of users for the world. Including this number allows you to autoassign end users to worlds and ensure that the correct number of users are in each world.\n        * @param {object} options (Optional) Options object to override global options.\n        * @return {Promise}\n        */\n        update: function (params, options) {\n            var whitelist = ['roles', 'optionalRoles', 'minUsers'];\n            options = options || {};\n            setIdFilterOrThrowError(options);\n\n            var updateOptions = $.extend(true, {},\n                serviceOptions,\n                options,\n                { url: urlConfig.getAPIPath(apiEndpoint) + serviceOptions.filter }\n            );\n\n            params = _pick(params || {}, whitelist);\n\n            return http.patch(params, updateOptions);\n        },\n\n        /**\n        * Deletes an existing world.\n        *\n        * This function optionally takes one argument. If the argument is a string, it is the id of the world to delete. If the argument is an object, it is the override for global options.\n        *\n        *  **Example**\n        *\n        *      var wa = new F.service.World({\n        *           account: 'acme-simulations',\n        *           project: 'supply-chain-game',\n        *           group: 'team1' });\n        *      wa.create()\n        *           .then(function(world) {\n        *               wa.delete();\n        *           });\n        *\n        *  **Parameters**\n        * @param {String|Object} options (Optional) The id of the world to delete, or options object to override global options.\n        * @return {Promise}\n        */\n        delete: function (options) {\n            options = (options && (typeof options === 'string')) ? { filter: options } : {};\n            setIdFilterOrThrowError(options);\n\n            var deleteOptions = $.extend(true, {},\n                serviceOptions,\n                options,\n                { url: urlConfig.getAPIPath(apiEndpoint) + serviceOptions.filter }\n            );\n\n            return http.delete(null, deleteOptions);\n        },\n\n        /**\n        * Updates the configuration for the current instance of the World API Adapter (including all subsequent function calls, until the configuration is updated again).\n        *\n        * **Example**\n        *\n        *      var wa = new F.service.World({...}).updateConfig({ filter: '123' }).addUser({ userId: '123' });\n        *\n        * **Parameters**\n        * @param {object} config The configuration object to use in updating existing configuration.\n        * @return {Object} reference to current instance\n        */\n        updateConfig: function (config) {\n            $.extend(serviceOptions, config);\n\n            return this;\n        },\n\n        /**\n        * Lists all worlds for a given account, project, and group. All three are required, and if not specified as parameters, are read from the service.\n        *\n        *  **Example**\n        *\n        *      var wa = new F.service.World({\n        *           account: 'acme-simulations',\n        *           project: 'supply-chain-game',\n        *           group: 'team1' });\n        *      wa.create()\n        *           .then(function(world) {\n        *               // lists all worlds in group \"team1\"\n        *               wa.list();\n        *\n        *               // lists all worlds in group \"other-group-name\"\n        *               wa.list({ group: 'other-group-name' });\n        *           });\n        *\n        *  **Parameters**\n        * @param {object} options (Optional) Options object to override global options.\n        * @return {Promise}\n        */\n        list: function (options) {\n            options = options || {};\n\n            var getOptions = $.extend(true, {},\n                serviceOptions,\n                options,\n                { url: urlConfig.getAPIPath(apiEndpoint) }\n            );\n\n            var filters = _pick(getOptions, ['account', 'project', 'group']);\n\n            return http.get(filters, getOptions);\n        },\n\n        /**\n        * Gets all worlds that an end user belongs to for a given account (team), project, and group.\n        *\n        *  **Example**\n        *\n        *      var wa = new F.service.World({\n        *           account: 'acme-simulations',\n        *           project: 'supply-chain-game',\n        *           group: 'team1' });\n        *      wa.create()\n        *           .then(function(world) {\n        *               wa.getWorldsForUser('b1c19dda-2d2e-4777-ad5d-3929f17e86d3')\n        *           });\n        *\n        * ** Parameters **\n        * @param {string} userId The `userId` of the user whose worlds are being retrieved.\n        * @param {object} options (Optional) Options object to override global options.\n        * @return {Promise}\n        */\n        getWorldsForUser: function (userId, options) {\n            options = options || {};\n\n            var getOptions = $.extend(true, {},\n                serviceOptions,\n                options,\n                { url: urlConfig.getAPIPath(apiEndpoint) }\n            );\n\n            var filters = $.extend(\n                _pick(getOptions, ['account', 'project', 'group']),\n                { userId: userId }\n            );\n\n            return http.get(filters, getOptions);\n        },\n\n        /**\n         * Load information for a specific world. All further calls to the world service will use the id provided.\n         *\n         * **Parameters**\n         * @param {String} worldId The id of the world to load.\n         * @param {Object} options (Optional) Options object to override global options.\n         * @return {Promise}\n         */\n        load: function (worldId, options) {\n            if (worldId) {\n                serviceOptions.filter = worldId;\n            }\n            if (!serviceOptions.filter) {\n                throw new Error('Please provide a worldid to load');\n            }\n            var httpOptions = $.extend(true, {}, serviceOptions, options, { url: urlConfig.getAPIPath(apiEndpoint) + serviceOptions.filter + '/' });\n            return http.get('', httpOptions);\n        },\n\n        /**\n        * Adds an end user or list of end users to a given world. The end user must be a member of the `group` that is associated with this world.\n        *\n        *  **Example**\n        *\n        *      var wa = new F.service.World({\n        *           account: 'acme-simulations',\n        *           project: 'supply-chain-game',\n        *           group: 'team1' });\n        *      wa.create()\n        *           .then(function(world) {\n        *               // add one user\n        *               wa.addUsers('b1c19dda-2d2e-4777-ad5d-3929f17e86d3');\n        *               wa.addUsers(['b1c19dda-2d2e-4777-ad5d-3929f17e86d3']);\n        *               wa.addUsers({ userId: 'b1c19dda-2d2e-4777-ad5d-3929f17e86d3', role: 'VP Sales' });\n        *\n        *               // add several users\n        *               wa.addUsers([\n        *                   { userId: 'a6fe0c1e-f4b8-4f01-9f5f-01ccf4c2ed44',\n        *                     role: 'VP Marketing' },\n        *                   { userId: '8f2604cf-96cd-449f-82fa-e331530734ee',\n        *                     role: 'VP Engineering' }\n        *               ]);\n        *\n        *               // add one user to a specific world\n        *               wa.addUsers('b1c19dda-2d2e-4777-ad5d-3929f17e86d3', world.id);\n        *               wa.addUsers('b1c19dda-2d2e-4777-ad5d-3929f17e86d3', { filter: world.id });\n        *           });\n        *\n        * ** Parameters **\n        * @param {string|object|array} users User id, array of user ids, object, or array of objects of the users to add to this world.\n        * @param {string} users.role The `role` the user should have in the world. It is up to the caller to ensure, if needed, that the `role` passed in is one of the `roles` or `optionalRoles` of this world.\n        * @param {string} worldId The world to which the users should be added. If not specified, the filter parameter of the `options` object is used.\n        * @param {object} options (Optional) Options object to override global options.\n        * @return {Promise}\n        */\n        addUsers: function (users, worldId, options) {\n\n            if (!users) {\n                throw new Error('Please provide a list of users to add to the world');\n            }\n\n            // normalize the list of users to an array of user objects\n            users = $.map([].concat(users), function (u) {\n                var isObject = $.isPlainObject(u);\n\n                if (typeof u !== 'string' && !isObject) {\n                    throw new Error('Some of the users in the list are not in the valid format: ' + u);\n                }\n\n                return isObject ? u : { userId: u };\n            });\n\n            // check if options were passed as the second parameter\n            if ($.isPlainObject(worldId) && !options) {\n                options = worldId;\n                worldId = null;\n            }\n\n            options = options || {};\n\n            // we must have options by now\n            if (typeof worldId === 'string') {\n                options.filter = worldId;\n            }\n\n            setIdFilterOrThrowError(options);\n\n            var updateOptions = $.extend(true, {},\n                serviceOptions,\n                options,\n                { url: urlConfig.getAPIPath(apiEndpoint) + serviceOptions.filter + '/users' }\n            );\n\n            return http.post(users, updateOptions);\n        },\n\n        /**\n        * Updates the role of an end user in a given world. (You can only update one end user at a time.)\n        *\n        * **Example**\n        *\n        *      var wa = new F.service.World({\n        *           account: 'acme-simulations',\n        *           project: 'supply-chain-game',\n        *           group: 'team1' });\n        *\n        *      wa.create().then(function(world) {\n        *           wa.addUsers('b1c19dda-2d2e-4777-ad5d-3929f17e86d3');\n        *           wa.updateUser({ userId: 'b1c19dda-2d2e-4777-ad5d-3929f17e86d3', role: 'leader' });\n        *      });\n        *\n        * **Parameters**\n        * @param {object} user User object with `userId` and the new `role`.\n        * @param {object} options (Optional) Options object to override global options.\n        * @return {Promise}\n        */\n        updateUser: function (user, options) {\n            options = options || {};\n\n            if (!user || !user.userId) {\n                throw new Error('You need to pass a userId to update from the world');\n            }\n\n            setIdFilterOrThrowError(options);\n\n            var patchOptions = $.extend(true, {},\n                serviceOptions,\n                options,\n                { url: urlConfig.getAPIPath(apiEndpoint) + serviceOptions.filter + '/users/' + user.userId }\n            );\n\n            return http.patch(_pick(user, 'role'), patchOptions);\n        },\n\n        /**\n        * Removes an end user from a given world.\n        *\n        *  **Example**\n        *\n        *      var wa = new F.service.World({\n        *           account: 'acme-simulations',\n        *           project: 'supply-chain-game',\n        *           group: 'team1' });\n        *      wa.create()\n        *           .then(function(world) {\n        *               wa.addUsers(['a6fe0c1e-f4b8-4f01-9f5f-01ccf4c2ed44', '8f2604cf-96cd-449f-82fa-e331530734ee']);\n        *               wa.removeUser('a6fe0c1e-f4b8-4f01-9f5f-01ccf4c2ed44');\n        *               wa.removeUser({ userId: '8f2604cf-96cd-449f-82fa-e331530734ee' });\n        *           });\n        *\n        * ** Parameters **\n        * @param {object|string} user The `userId` of the user to remove from the world, or an object containing the `userId` field.\n        * @param {object} options (Optional) Options object to override global options.\n        * @return {Promise}\n        */\n        removeUser: function (user, options) {\n            options = options || {};\n\n            if (typeof user === 'string') {\n                user = { userId: user };\n            }\n\n            if (!user.userId) {\n                throw new Error('You need to pass a userId to remove from the world');\n            }\n\n            setIdFilterOrThrowError(options);\n\n            var getOptions = $.extend(true, {},\n                serviceOptions,\n                options,\n                { url: urlConfig.getAPIPath(apiEndpoint) + serviceOptions.filter + '/users/' + user.userId }\n            );\n\n            return http.delete(null, getOptions);\n        },\n\n        /**\n        * Gets the run id of current run for the given world. If the world does not have a run, creates a new one and returns the run id.\n        *\n        * Remember that a [run](../../glossary/#run) is a collection of interactions with a project and its model. In the case of multiplayer projects, the run is shared by all end users in the world.\n        *\n        *  **Example**\n        *\n        *      var wa = new F.service.World({\n        *           account: 'acme-simulations',\n        *           project: 'supply-chain-game',\n        *           group: 'team1' });\n        *      wa.create()\n        *           .then(function(world) {\n        *               wa.getCurrentRunId({ model: 'model.py' });\n        *           });\n        *\n        * ** Parameters **\n        * @param {object} options (Optional) Options object to override global options.\n        * @param {object} options.model The model file to use to create a run if needed.\n        * @return {Promise}\n        */\n        getCurrentRunId: function (options) {\n            options = options || {};\n\n            setIdFilterOrThrowError(options);\n\n            var getOptions = $.extend(true, {},\n                serviceOptions,\n                options,\n                { url: urlConfig.getAPIPath(apiEndpoint) + serviceOptions.filter + '/run' }\n            );\n\n            validateModelOrThrowError(getOptions);\n            return http.post(_pick(getOptions, 'model'), getOptions);\n        },\n\n        /**\n        * Gets the current (most recent) world for the given end user in the given group. Brings this most recent world into memory if needed.\n        *\n        *  **Example**\n        *\n        *      var wa = new F.service.World({\n        *           account: 'acme-simulations',\n        *           project: 'supply-chain-game',\n        *           group: 'team1' });\n        *      wa.getCurrentWorldForUser('8f2604cf-96cd-449f-82fa-e331530734ee')\n        *           .then(function(world) {\n        *               // use data from world\n        *           });\n        *\n        * ** Parameters **\n        * @param {string} userId The `userId` of the user whose current (most recent) world is being retrieved.\n        * @param {string} groupName (Optional) The name of the group. If not provided, defaults to the group used to create the service.\n        * @return {Promise}\n        */\n        getCurrentWorldForUser: function (userId, groupName) {\n            var dtd = $.Deferred();\n            var me = this;\n            this.getWorldsForUser(userId, { group: groupName })\n                .then(function (worlds) {\n                    // assume the most recent world as the 'active' world\n                    worlds.sort(function (a, b) { return new Date(b.lastModified) - new Date(a.lastModified); });\n                    var currentWorld = worlds[0];\n\n                    if (currentWorld) {\n                        serviceOptions.filter = currentWorld.id;\n                    }\n\n                    dtd.resolveWith(me, [currentWorld]);\n                })\n                .fail(dtd.reject);\n\n            return dtd.promise();\n        },\n\n        /**\n        * Deletes the current run from the world.\n        *\n        * (Note that the world id remains part of the run record, indicating that the run was formerly an active run for the world.)\n        *\n        *  **Example**\n        *\n        *      var wa = new F.service.World({\n        *           account: 'acme-simulations',\n        *           project: 'supply-chain-game',\n        *           group: 'team1' });\n        *\n        *      wa.deleteRun('sample-world-id');\n        *\n        *  **Parameters**\n        * @param {string} worldId The `worldId` of the world from which the current run is being deleted.\n        * @param {object} options (Optional) Options object to override global options.\n        * @return {Promise}\n        */\n        deleteRun: function (worldId, options) {\n            options = options || {};\n\n            if (worldId) {\n                options.filter = worldId;\n            }\n\n            setIdFilterOrThrowError(options);\n\n            var deleteOptions = $.extend(true, {},\n                serviceOptions,\n                options,\n                { url: urlConfig.getAPIPath(apiEndpoint) + serviceOptions.filter + '/run' }\n            );\n\n            return http.delete(null, deleteOptions);\n        },\n\n        /**\n        * Creates a new run for the world.\n        *\n        *  **Example**\n        *\n        *      var wa = new F.service.World({\n        *           account: 'acme-simulations',\n        *           project: 'supply-chain-game',\n        *           group: 'team1' });\n        *\n        *      wa.getCurrentWorldForUser('8f2604cf-96cd-449f-82fa-e331530734ee')\n        *           .then(function (world) {\n        *                   wa.newRunForWorld(world.id);\n        *           });\n        *\n        *  **Parameters**\n        * @param {string} worldId worldId in which we create the new run.\n        * @param {object} options (Optional) Options object to override global options.\n        * @param {object} options.model The model file to use to create a run if needed.\n        * @return {Promise}\n        */\n        newRunForWorld: function (worldId, options) {\n            var currentRunOptions = $.extend(true, {},\n                serviceOptions,\n                options,\n                { filter: worldId || serviceOptions.filter }\n            );\n            var me = this;\n\n            validateModelOrThrowError(currentRunOptions);\n\n            return this.deleteRun(worldId, options)\n                .then(function () {\n                    return me.getCurrentRunId(currentRunOptions);\n                });\n        },\n\n        /**\n        * Assigns end users to worlds, creating new worlds as appropriate, automatically. Assigns all end users in the group, and creates new worlds as needed based on the project-level world configuration (roles, optional roles, and minimum end users per world).\n        *\n        * **Example**\n        *\n        *      var wa = new F.service.World({\n        *           account: 'acme-simulations',\n        *           project: 'supply-chain-game',\n        *           group: 'team1' });\n        *\n        *      wa.autoAssign();\n        *\n        * **Parameters**\n        * @param {object} options (Optional) Options object to override global options.\n        * @param {number} options.maxUsers Sets the maximum number of users in a world.\n        * @param {string[]} options.userIds A list of users to be assigned be assigned instead of all end users in the group.\n        * @return {Promise}\n        *\n        */\n        autoAssign: function (options) {\n            options = options || {};\n\n            var opt = $.extend(true, {},\n                serviceOptions,\n                options,\n                { url: urlConfig.getAPIPath(assignmentEndpoint) }\n            );\n\n            var params = {\n                account: opt.account,\n                project: opt.project,\n                group: opt.group\n            };\n\n            if (opt.maxUsers) {\n                params.maxUsers = opt.maxUsers;\n            }\n\n            if (opt.userIds) {\n                params.userIds = opt.userIds;\n            }\n\n            return http.post(params, opt);\n        },\n\n        /**\n        * Gets the project's world configuration.\n        *\n        * Typically, every interaction with your project uses the same configuration of each world. For example, each world in your project probably has the same roles for end users. And your project is probably either configured so that all end users share the same world (and run), or smaller sets of end users share worlds — but not both.\n        *\n        * (The [Multiplayer Project REST API](../../../rest_apis/multiplayer/multiplayer_project/) allows you to set these project-level world configurations. The World Adapter simply retrieves them, for example so they can be used in auto-assignment of end users to worlds.)\n        *\n        * **Example**\n        *\n        *      var wa = new F.service.World({\n        *           account: 'acme-simulations',\n        *           project: 'supply-chain-game',\n        *           group: 'team1' });\n        *\n        *      wa.getProjectSettings()\n        *           .then(function(settings) {\n        *               console.log(settings.roles);\n        *               console.log(settings.optionalRoles);\n        *           });\n        *\n        * **Parameters**\n        * @param {object} options (Optional) Options object to override global options.\n        * @return {Promise}\n        */\n        getProjectSettings: function (options) {\n            options = options || {};\n\n            var opt = $.extend(true, {},\n                serviceOptions,\n                options,\n                { url: urlConfig.getAPIPath(projectEndpoint) }\n            );\n\n            opt.url += [opt.account, opt.project].join('/');\n            return http.get(null, opt);\n        }\n\n    };\n\n    $.extend(this, publicAPI);\n};\n","/**\n * @class Cookie Storage Service\n *\n * @example\n *      var people = require('cookie-store')({ root: 'people' });\n        people\n            .save({lastName: 'smith' })\n\n */\n\n\n'use strict';\n\n// Thin document.cookie wrapper to allow unit testing\nvar Cookie = function () {\n    this.get = function () {\n        return document.cookie;\n    };\n\n    this.set = function (newCookie) {\n        document.cookie = newCookie;\n    };\n};\n\nmodule.exports = function (config) {\n    var host = window.location.hostname;\n    var validHost = host.split('.').length > 1;\n    var domain = validHost ? '.' + host : null;\n\n    var defaults = {\n        /**\n         * Name of collection\n         * @type { string}\n         */\n        root: '/',\n\n        domain: domain,\n        cookie: new Cookie()\n    };\n    this.serviceOptions = $.extend({}, defaults, config);\n\n    var publicAPI = {\n        // * TBD\n        //  * Query collection; uses MongoDB syntax\n        //  * @see  <TBD: Data API URL>\n        //  *\n        //  * @param { string} qs Query Filter\n        //  * @param { string} limiters @see <TBD: url for limits, paging etc>\n        //  *\n        //  * @example\n        //  *     cs.query(\n        //  *      { name: 'John', className: 'CSC101' },\n        //  *      {limit: 10}\n        //  *     )\n\n        // query: function (qs, limiters) {\n\n        // },\n\n        /**\n         * Save cookie value\n         * @param  { string|Object} key   If given a key save values under it, if given an object directly, save to top-level api\n         * @param  {Object} value (Optional)\n         * @param {Object} options Overrides for service options\n         *\n         * @return {*} The saved value\n         *\n         * @example\n         *     cs.set('person', { firstName: 'john', lastName: 'smith' });\n         *     cs.set({ name:'smith', age:'32' });\n         */\n        set: function (key, value, options) {\n            var setOptions = $.extend(true, {}, this.serviceOptions, options);\n\n            var domain = setOptions.domain;\n            var path = setOptions.root;\n            var cookie = setOptions.cookie;\n\n            cookie.set(encodeURIComponent(key) + '=' +\n                                encodeURIComponent(value) +\n                                (domain ? '; domain=' + domain : '') +\n                                (path ? '; path=' + path : '')\n            );\n\n            return value;\n        },\n\n        /**\n         * Load cookie value\n         * @param  { string|Object} key   If given a key save values under it, if given an object directly, save to top-level api\n         * @return {*} The value stored\n         *\n         * @example\n         *     cs.get('person');\n         */\n        get: function (key) {\n            var cookie = this.serviceOptions.cookie;\n            var cookieReg = new RegExp('(?:^|;)\\\\s*' + encodeURIComponent(key).replace(/[\\-\\.\\+\\*]/g, '\\\\$&') + '\\\\s*\\\\=\\\\s*([^;]*).*$');\n            var res = cookieReg.exec(cookie.get());\n            var val = res ? decodeURIComponent(res[1]) : null;\n            return val;\n        },\n\n        /**\n         * Removes key from collection\n         * @param { string} key key to remove\n         * @param {object} options (optional) overrides for service options\n         * @return { string} key The key removed\n         *\n         * @example\n         *     cs.remove('person');\n         */\n        remove: function (key, options) {\n            var remOptions = $.extend(true, {}, this.serviceOptions, options);\n\n            var domain = remOptions.domain;\n            var path = remOptions.root;\n            var cookie = remOptions.cookie;\n\n            cookie.set(encodeURIComponent(key) +\n                            '=; expires=Thu, 01 Jan 1970 00:00:00 GMT' +\n                            (domain ? '; domain=' + domain : '') +\n                            (path ? '; path=' + path : '')\n            );\n            return key;\n        },\n\n        /**\n         * Removes collection being referenced\n         * @return { array} keys All the keys removed\n         */\n        destroy: function () {\n            var cookie = this.serviceOptions.cookie;\n            var aKeys = cookie.get().replace(/((?:^|\\s*;)[^\\=]+)(?=;|$)|^\\s*|\\s*(?:\\=[^;]*)?(?:\\1|$)/g, '').split(/\\s*(?:\\=[^;]*)?;\\s*/);\n            for (var nIdx = 0; nIdx < aKeys.length; nIdx++) {\n                var cookieKey = decodeURIComponent(aKeys[nIdx]);\n                this.remove(cookieKey);\n            }\n            return aKeys;\n        }\n    };\n\n    $.extend(this, publicAPI);\n};\n","'use strict';\n\nvar keyNames = require('../managers/key-names');\nvar StorageFactory = require('./store-factory');\nvar optionUtils = require('../util/option-utils');\n\nvar EPI_SESSION_KEY = keyNames.EPI_SESSION_KEY;\nvar EPI_MANAGER_KEY = 'epicenter.token'; //can't be under key-names, or logout will clear this too\n\nvar defaults = {\n    /**\n     * Where to store user access tokens for temporary access. Defaults to storing in a cookie in the browser.\n     * @type {string}\n     */\n    store: { synchronous: true }\n};\n\nvar SessionManager = function (managerOptions) {\n    managerOptions = managerOptions || {};\n    function getBaseOptions(overrides) {\n        overrides = overrides || {};\n        var libOptions = optionUtils.getOptions();\n        var finalOptions = $.extend(true, {}, defaults, libOptions, managerOptions, overrides);\n        return finalOptions;\n    }\n\n    function getStore(overrides) {\n        var baseOptions = getBaseOptions(overrides);\n        var storeOpts = baseOptions.store || {};\n        var isEpicenterDomain = !baseOptions.isLocal && !baseOptions.isCustomDomain;\n        if (storeOpts.root === undefined && baseOptions.account && baseOptions.project && isEpicenterDomain) {\n            storeOpts.root = '/app/' + baseOptions.account + '/' + baseOptions.project;\n        }\n        return new StorageFactory(storeOpts);\n    }\n\n    var publicAPI = {\n        saveSession: function (userInfo, options) {\n            var serialized = JSON.stringify(userInfo);\n            getStore(options).set(EPI_SESSION_KEY, serialized);\n        },\n        getSession: function (options) {\n            var store = getStore(options);\n            var finalOpts = store.serviceOptions;\n            var serialized = store.get(EPI_SESSION_KEY) || '{}';\n            var session = JSON.parse(serialized);\n            // If the url contains the project and account\n            // validate the account and project in the session\n            // and override project, groupName, groupId and isFac\n            // Otherwise (i.e. localhost) use the saved session values\n            var account = finalOpts.account;\n            var project = finalOpts.project;\n            if (account && session.account !== account) {\n                // This means that the token was not used to login to the same account\n                return {};\n            }\n            if (session.groups && account && project) {\n                var group = session.groups[project] || { groupId: '', groupName: '', isFac: false };\n                $.extend(session, { project: project }, group);\n            }\n            return session;\n        },\n        removeSession: function (options) {\n            var store = getStore(options);\n            Object.keys(keyNames).forEach(function (cookieKey) {\n                var cookieName = keyNames[cookieKey];\n                store.remove(cookieName);\n            });\n            return true;\n        },\n        getStore: function (options) {\n            return getStore(options);\n        },\n\n        getMergedOptions: function () {\n            var args = Array.prototype.slice.call(arguments);\n            var overrides = $.extend.apply($, [true, {}].concat(args));\n            var baseOptions = getBaseOptions(overrides);\n            var session = this.getSession(overrides);\n\n            var token = session.auth_token;\n            if (!token) {\n                var factory = new StorageFactory();\n                token = factory.get(EPI_MANAGER_KEY);\n            }\n\n            var sessionDefaults = {\n                /**\n                 * For projects that require authentication, pass in the user access token (defaults to empty string). If the user is already logged in to Epicenter, the user access token is already set in a cookie and automatically loaded from there. (See [more background on access tokens](../../../project_access/)).\n                 * @see [Authentication API Service](../auth-api-service/) for getting tokens.\n                 * @type {String}\n                 */\n                token: token,\n\n                /**\n                 * The account. If left undefined, taken from the cookie session.\n                 * @type {String}\n                 */\n                account: session.account,\n\n                /**\n                 * The project. If left undefined, taken from the cookie session.\n                 * @type {String}\n                 */\n                project: session.project,\n\n\n                /**\n                 * The group name. If left undefined, taken from the cookie session.\n                 * @type {String}\n                 */\n                group: session.groupName,\n                /**\n                 * Alias for group. \n                 * @type {String}\n                 */\n                groupName: session.groupName, //It's a little weird that it's called groupName in the cookie, but 'group' in all the service options, so normalize for both\n                /**\n                 * The group id. If left undefined, taken from the cookie session.\n                 * @type {String}\n                 */\n                groupId: session.groupId,\n                userId: session.userId,\n                userName: session.userName,\n            };\n            return $.extend(true, sessionDefaults, baseOptions);\n        }\n    };\n    $.extend(this, publicAPI);\n};\n\nmodule.exports = SessionManager;","/**\n    Decides type of store to provide\n*/\n\n'use strict';\n// var isNode = false; FIXME: Browserify/minifyify has issues with the next link\n// var store = (isNode) ? require('./session-store') : require('./cookie-store');\nvar store = require('./cookie-store');\n\nmodule.exports = store;\n","'use strict';\n\nvar qutils = require('../util/query-util');\n\nmodule.exports = function (config) {\n\n    var defaults = {\n        url: '',\n\n        contentType: 'application/json',\n        headers: {},\n        statusCode: {\n            404: $.noop\n        },\n\n        /**\n         * ONLY for strings in the url. All GET & DELETE params are run through this\n         * @type {[type] }\n         */\n        parameterParser: qutils.toQueryFormat,\n\n        // To allow epicenter.token and other session cookies to be passed\n        // with the requests\n        xhrFields: {\n            withCredentials: true\n        }\n    };\n\n    var transportOptions = $.extend({}, defaults, config);\n\n    var result = function (d) {\n        return ($.isFunction(d)) ? d() : d;\n    };\n\n    var connect = function (method, params, connectOptions) {\n        params = result(params);\n        params = ($.isPlainObject(params) || $.isArray(params)) ? JSON.stringify(params) : params;\n\n        var options = $.extend(true, {}, transportOptions, connectOptions, {\n            type: method,\n            data: params\n        });\n        var ALLOWED_TO_BE_FUNCTIONS = ['data', 'url'];\n        $.each(options, function (key, value) {\n            if ($.isFunction(value) && $.inArray(key, ALLOWED_TO_BE_FUNCTIONS) !== -1) {\n                options[key] = value();\n            }\n        });\n\n        if (options.logLevel && options.logLevel === 'DEBUG') {\n            console.log(options.url);\n            var oldSuccessFn = options.success || $.noop;\n            options.success = function (response, ajaxStatus, ajaxReq) {\n                console.log(response);\n                oldSuccessFn.apply(this, arguments);\n            };\n        }\n\n        var beforeSend = options.beforeSend;\n        options.beforeSend = function (xhr, settings) {\n            xhr.requestUrl = (connectOptions || {}).url;\n            if (beforeSend) {\n                beforeSend.apply(this, arguments);\n            }\n        };\n\n        return $.ajax(options);\n    };\n\n    var publicAPI = {\n        get: function (params, ajaxOptions) {\n            var options = $.extend({}, transportOptions, ajaxOptions);\n            params = options.parameterParser(result(params));\n            return connect.call(this, 'GET', params, options);\n        },\n        splitGet: function () {\n\n        },\n        post: function () {\n            return connect.apply(this, ['post'].concat([].slice.call(arguments)));\n        },\n        patch: function () {\n            return connect.apply(this, ['patch'].concat([].slice.call(arguments)));\n        },\n        put: function () {\n            return connect.apply(this, ['put'].concat([].slice.call(arguments)));\n        },\n        delete: function (params, ajaxOptions) {\n            //DELETE doesn't support body params, but jQuery thinks it does.\n            var options = $.extend({}, transportOptions, ajaxOptions);\n            params = options.parameterParser(result(params));\n            if ($.trim(params)) {\n                var delimiter = (result(options.url).indexOf('?') === -1) ? '?' : '&';\n                options.url = result(options.url) + delimiter + params;\n            }\n            return connect.call(this, 'DELETE', null, options);\n        },\n        head: function () {\n            return connect.apply(this, ['head'].concat([].slice.call(arguments)));\n        },\n        options: function () {\n            return connect.apply(this, ['options'].concat([].slice.call(arguments)));\n        }\n    };\n\n    return $.extend(this, publicAPI);\n};\n","'use strict';\n\n// var isNode = false; FIXME: Browserify/minifyify has issues with the next link\n// var transport = (isNode) ? require('./node-http-transport') : require('./ajax-http-transport');\nvar transport = require('./ajax-http-transport');\nmodule.exports = transport;\n","/**\n/* Inherit from a class (using prototype borrowing)\n*/\n'use strict';\n\nfunction inherit(C, P) {\n    var F = function () {};\n    F.prototype = P.prototype;\n    C.prototype = new F();\n    C.__super = P.prototype;\n    C.prototype.constructor = C;\n}\n\n/**\n* Shallow copy of an object\n* @param {Object} dest object to extend\n* @return {Object} extended object\n*/\nvar extend = function (dest /*, var_args*/) {\n    var obj = Array.prototype.slice.call(arguments, 1);\n    var current;\n    for (var j = 0; j < obj.length; j++) {\n        if (!(current = obj[j])) { //eslint-disable-line\n            continue;\n        }\n\n        // do not wrap inner in dest.hasOwnProperty or bad things will happen\n        for (var key in current) { //eslint-disable-line\n            dest[key] = current[key];\n        }\n    }\n\n    return dest;\n};\n\nmodule.exports = function (base, props, staticProps) {\n    var parent = base;\n    var child;\n\n    child = props && props.hasOwnProperty('constructor') ? props.constructor : function () { return parent.apply(this, arguments); };\n\n    // add static properties to the child constructor function\n    extend(child, parent, staticProps);\n\n    // associate prototype chain\n    inherit(child, parent);\n\n    // add instance properties\n    if (props) {\n        extend(child.prototype, props);\n    }\n\n    // done\n    return child;\n};\n","'use strict';\n\nmodule.exports = {\n    _pick: function (obj, props) {\n        var res = {};\n        for (var p in obj) {\n            if (props.indexOf(p) !== -1) {\n                res[p] = obj[p];\n            }\n        }\n\n        return res;\n    },\n    isEmpty: function isEmpty(value) {\n        return (!value || ($.isPlainObject(value) && Object.keys(value).length === 0));\n    }\n};\n","'use strict';\n\nvar ConfigService = require('../service/configuration-service');\n\nvar urlConfig = new ConfigService().get('server');\nvar customDefaults = {};\nvar libDefaults = {\n    /**\n     * The account id. In the Epicenter UI, this is the **Team ID** (for team projects) or **User ID** (for personal projects). Defaults to empty string. If left undefined, taken from the URL.\n     * @type {String}\n     */\n    account: urlConfig.accountPath || undefined,\n    /**\n     * The account id. In the Epicenter UI, this is the **Team ID** (for team projects) or **User ID** (for personal projects). Defaults to empty string. If left undefined, taken from the URL.\n     * @type {String}\n     */\n    project: urlConfig.projectPath || undefined,\n    isLocal: urlConfig.isLocalhost(),\n    isCustomDomain: urlConfig.isCustomDomain,\n    store: {}\n};\n\nvar optionUtils = {\n    /**\n     * Gets the final options by overriding the global options set with\n     * optionUtils#setDefaults() and the lib defaults.\n     * @param {object} options The final options object.\n     * @return {object} Extended object\n     */\n    getOptions: function (options) {\n        return $.extend(true, {}, libDefaults, customDefaults, options);\n    },\n    /**\n     * Sets the global defaults for the optionUtils#getOptions() method.\n     * @param {object} defaults The defaults object.\n     */\n    setDefaults: function (defaults) {\n        customDefaults = defaults;\n    }\n};\nmodule.exports = optionUtils;\n","/**\n * Utilities for working with query strings\n*/\n'use strict';\n\nmodule.exports = (function () {\n\n    return {\n        /**\n         * Converts to matrix format\n         * @param  {Object} qs Object to convert to query string\n         * @return { string}    Matrix-format query parameters\n         */\n        toMatrixFormat: function (qs) {\n            if (qs === null || qs === undefined || qs === '') {\n                return ';';\n            }\n            if (typeof qs === 'string' || qs instanceof String) {\n                return qs;\n            }\n\n            var returnArray = [];\n            var OPERATORS = ['<', '>', '!'];\n            $.each(qs, function (key, value) {\n                if (typeof value !== 'string' || $.inArray($.trim(value).charAt(0), OPERATORS) === -1) {\n                    value = '=' + value;\n                }\n                returnArray.push(key + value);\n            });\n\n            var mtrx = ';' + returnArray.join(';');\n            return mtrx;\n        },\n\n        /**\n         * Converts strings/arrays/objects to type 'a=b&b=c'\n         * @param  { string|Array|Object} qs\n         * @return { string}\n         */\n        toQueryFormat: function (qs) {\n            if (qs === null || qs === undefined) {\n                return '';\n            }\n            if (typeof qs === 'string' || qs instanceof String) {\n                return qs;\n            }\n\n            var returnArray = [];\n            $.each(qs, function (key, value) {\n                if ($.isArray(value)) {\n                    value = value.join(',');\n                }\n                if ($.isPlainObject(value)) {\n                    //Mostly for data api\n                    value = JSON.stringify(value);\n                }\n                returnArray.push(key + '=' + value);\n            });\n\n            var result = returnArray.join('&');\n            return result;\n        },\n\n        /**\n         * Converts strings of type 'a=b&b=c' to { a:b, b:c}\n         * @param  { string} qs\n         * @return {object}\n         */\n        qsToObject: function (qs) {\n            if (qs === null || qs === undefined || qs === '') {\n                return {};\n            }\n\n            var qsArray = qs.split('&');\n            var returnObj = {};\n            $.each(qsArray, function (index, value) {\n                var qKey = value.split('=')[0];\n                var qVal = value.split('=')[1];\n\n                if (qVal.indexOf(',') !== -1) {\n                    qVal = qVal.split(',');\n                }\n\n                returnObj[qKey] = qVal;\n            });\n\n            return returnObj;\n        },\n\n        /**\n         * Normalizes and merges strings of type 'a=b', { b:c} to { a:b, b:c}\n         * @param  { string|Array|Object} qs1\n         * @param  { string|Array|Object} qs2\n         * @return {Object}\n         */\n        mergeQS: function (qs1, qs2) {\n            var obj1 = this.qsToObject(this.toQueryFormat(qs1));\n            var obj2 = this.qsToObject(this.toQueryFormat(qs2));\n            return $.extend(true, {}, obj1, obj2);\n        },\n\n        addTrailingSlash: function (url) {\n            if (!url) {\n                return '';\n            }\n            return (url.charAt(url.length - 1) === '/') ? url : (url + '/');\n        }\n    };\n}());\n\n\n","/**\n * Utilities for working with the run service\n*/\n'use strict';\nvar qutil = require('./query-util');\nvar MAX_URL_LENGTH = 2048;\n\nmodule.exports = (function () {\n    return {\n        /**\n         * normalizes different types of operation inputs\n         * @param  {Object|Array|String} operations operations to perform\n         * @param  {Array} args arguments for operation\n         * @return {String} operations of the form `{ ops: [], args: [] }`\n         */\n        normalizeOperations: function (operations, args) {\n            if (!args) {\n                args = [];\n            }\n            var returnList = {\n                ops: [],\n                args: []\n            };\n\n            var _concat = function (arr) {\n                return (arr !== null && arr !== undefined) ? [].concat(arr) : [];\n            };\n\n            //{ add: [1,2], subtract: [2,4] }\n            var _normalizePlainObjects = function (operations, returnList) {\n                if (!returnList) {\n                    returnList = { ops: [], args: [] };\n                }\n                $.each(operations, function (opn, arg) {\n                    returnList.ops.push(opn);\n                    returnList.args.push(_concat(arg));\n                });\n                return returnList;\n            };\n            //{ name: 'add', params: [1] }\n            var _normalizeStructuredObjects = function (operation, returnList) {\n                if (!returnList) {\n                    returnList = { ops: [], args: [] };\n                }\n                returnList.ops.push(operation.name);\n                returnList.args.push(_concat(operation.params));\n                return returnList;\n            };\n\n            var _normalizeObject = function (operation, returnList) {\n                return ((operation.name) ? _normalizeStructuredObjects : _normalizePlainObjects)(operation, returnList);\n            };\n\n            var _normalizeLiterals = function (operation, args, returnList) {\n                if (!returnList) {\n                    returnList = { ops: [], args: [] };\n                }\n                returnList.ops.push(operation);\n                returnList.args.push(_concat(args));\n                return returnList;\n            };\n\n\n            var _normalizeArrays = function (operations, arg, returnList) {\n                if (!returnList) {\n                    returnList = { ops: [], args: [] };\n                }\n                $.each(operations, function (index, opn) {\n                    if ($.isPlainObject(opn)) {\n                        _normalizeObject(opn, returnList);\n                    } else {\n                        _normalizeLiterals(opn, args[index], returnList);\n                    }\n                });\n                return returnList;\n            };\n\n            if ($.isPlainObject(operations)) {\n                _normalizeObject(operations, returnList);\n            } else if ($.isArray(operations)) {\n                _normalizeArrays(operations, args, returnList);\n            } else {\n                _normalizeLiterals(operations, args, returnList);\n            }\n\n            return returnList;\n        },\n\n        splitGetFactory: function (httpOptions) {\n            return function (params, options) {\n                var http = this; //eslint-disable-line\n                var getValue = function (name) {\n                    var value = options[name] || httpOptions[name];\n                    if (typeof value === 'function') {\n                        value = value();\n                    }\n                    return value;\n                };\n                var getFinalUrl = function (params) {\n                    var url = getValue('url', options);\n                    var data = params;\n                    // There is easy (or known) way to get the final URL jquery is going to send so\n                    // we're replicating it. The process might change at some point but it probably will not.\n                    // 1. Remove hash\n                    url = url.replace(/#.*$/, '');\n                    // 1. Append query string\n                    var queryParams = qutil.toQueryFormat(data);\n                    var questionIdx = url.indexOf('?');\n                    if (queryParams && questionIdx > -1) {\n                        return url + '&' + queryParams;\n                    } else if (queryParams) {\n                        return url + '?' + queryParams;\n                    }\n                    return url;\n                };\n                var url = getFinalUrl(params);\n                // We must split the GET in multiple short URL's\n                // The only property allowed to be split is \"include\"\n                if (params && params.include && encodeURI(url).length > MAX_URL_LENGTH) {\n                    var dtd = $.Deferred();\n                    var paramsCopy = $.extend(true, {}, params);\n                    delete paramsCopy.include;\n                    var urlNoIncludes = getFinalUrl(paramsCopy);\n                    var diff = MAX_URL_LENGTH - urlNoIncludes.length;\n                    var oldSuccess = options.success || httpOptions.success || $.noop;\n                    var oldError = options.error || httpOptions.error || $.noop;\n                    // remove the original success and error callbacks\n                    options.success = $.noop;\n                    options.error = $.noop;\n\n                    var include = params.include;\n                    var currIncludes = [];\n                    var includeOpts = [currIncludes];\n                    var currLength = encodeURIComponent('?include=').length;\n                    var variable = include.pop();\n                    while (variable) {\n                        var varLenght = encodeURIComponent(variable).length;\n                        // Use a greedy approach for now, can be optimized to be solved in a more\n                        // efficient way\n                        // + 1 is the comma\n                        if (currLength + varLenght + 1 < diff) {\n                            currIncludes.push(variable);\n                            currLength += varLenght + 1;\n                        } else {\n                            currIncludes = [variable];\n                            includeOpts.push(currIncludes);\n                            currLength = '?include='.length + varLenght;\n                        }\n                        variable = include.pop();\n                    }\n                    var reqs = $.map(includeOpts, function (include) {\n                        var reqParams = $.extend({}, params, { include: include });\n                        return http.get(reqParams, options);\n                    });\n                    $.when.apply($, reqs).then(function () {\n                        // Each argument are arrays of the arguments of each done request\n                        // So the first argument of the first array of arguments is the data\n                        var isValid = arguments[0] && arguments[0][0];\n                        if (!isValid) {\n                            // Should never happen...\n                            oldError();\n                            return dtd.reject();\n                        }\n                        var firstResponse = arguments[0][0];\n                        var isObject = $.isPlainObject(firstResponse);\n                        var isRunAPI = (isObject && $.isPlainObject(firstResponse.variables)) || !isObject;\n                        if (isRunAPI) {\n                            if (isObject) {\n                                // aggregate the variables property only\n                                var aggregateRun = arguments[0][0];\n                                $.each(arguments, function (idx, args) {\n                                    var run = args[0];\n                                    $.extend(true, aggregateRun.variables, run.variables);\n                                });\n                                oldSuccess(aggregateRun, arguments[0][1], arguments[0][2]);\n                                dtd.resolve(aggregateRun, arguments[0][1], arguments[0][2]);\n                            } else {\n                                // array of runs\n                                // Agregate variables in each run\n                                var aggregatedRuns = {};\n                                $.each(arguments, function (idx, args) {\n                                    var runs = args[0];\n                                    if (!$.isArray(runs)) {\n                                        return;\n                                    }\n                                    $.each(runs, function (idxRun, run) {\n                                        if (run.id && !aggregatedRuns[run.id]) {\n                                            run.variables = run.variables || {};\n                                            aggregatedRuns[run.id] = run;\n                                        } else if (run.id) {\n                                            $.extend(true, aggregatedRuns[run.id].variables, run.variables);\n                                        }\n                                    });\n                                });\n                                // turn it into an array\n                                aggregatedRuns = $.map(aggregatedRuns, function (run) { return run; });\n                                oldSuccess(aggregatedRuns, arguments[0][1], arguments[0][2]);\n                                dtd.resolve(aggregatedRuns, arguments[0][1], arguments[0][2]);\n                            }\n                        } else {\n                            // is variables API\n                            // aggregate the response\n                            var aggregatedVariables = {};\n                            $.each(arguments, function (idx, args) {\n                                var vars = args[0];\n                                $.extend(true, aggregatedVariables, vars);\n                            });\n                            oldSuccess(aggregatedVariables, arguments[0][1], arguments[0][2]);\n                            dtd.resolve(aggregatedVariables, arguments[0][1], arguments[0][2]);\n                        }\n                    }, function () {\n                        oldError.apply(http, arguments);\n                        dtd.reject.apply(dtd, arguments);\n                    });\n                    return dtd.promise();\n                } else {\n                    return http.get(params, options);\n                }\n            };\n        }\n    };\n}());\n"]} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["node_modules/browser-pack/_prelude.js","node_modules/Base64/base64.js","node_modules/object-assign/index.js","src/api-version.json","src/app.js","src/env-load.js","src/managers/auth-manager.js","src/managers/channel-manager.js","src/managers/epicenter-channel-manager.js","src/managers/key-names.js","src/managers/run-manager.js","src/managers/run-strategies/conditional-creation-strategy.js","src/managers/run-strategies/deprecated/new-if-initialized-strategy.js","src/managers/run-strategies/deprecated/new-if-persisted-strategy.js","src/managers/run-strategies/index.js","src/managers/run-strategies/multiplayer-strategy.js","src/managers/run-strategies/none-strategy.js","src/managers/run-strategies/reuse-across-sessions.js","src/managers/run-strategies/reuse-last-initialized.js","src/managers/run-strategies/reuse-never.js","src/managers/run-strategies/reuse-per-session.js","src/managers/saved-runs-manager.js","src/managers/scenario-manager.js","src/managers/scenario-strategies/baseline-strategy.js","src/managers/scenario-strategies/reuse-last-unsaved.js","src/managers/special-operations.js","src/managers/strategy-utils.js","src/managers/world-manager.js","src/service/admin-file-service.js","src/service/asset-api-adapter.js","src/service/auth-api-service.js","src/service/channel-service.js","src/service/configuration-service.js","src/service/data-api-service.js","src/service/group-api-service.js","src/service/introspection-api-service.js","src/service/member-api-adapter.js","src/service/presence-api-service.js","src/service/run-api-service.js","src/service/service-utils.js","src/service/state-api-adapter.js","src/service/url-config-service.js","src/service/user-api-adapter.js","src/service/variables-api-service.js","src/service/world-api-adapter.js","src/store/cookie-store.js","src/store/session-manager.js","src/store/store-factory.js","src/transport/ajax-http-transport.js","src/transport/http-transport-factory.js","src/util/inherit.js","src/util/object-util.js","src/util/option-utils.js","src/util/query-util.js","src/util/run-util.js"],"names":[],"mappings":"AAAA;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7DA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnFA;AACA;AACA;AACA;;;ACHA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;ACvEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5ZA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChQA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACraA;AACA;AACA;AACA;AACA;AACA;;ACLA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1OA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnHA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3EA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/DA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1JA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1OA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9DA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACRA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9MA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1XA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxHA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjPA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChSA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5LA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5NA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChmBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvJA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3JA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5vBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChJA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACVA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3GA;AACA;AACA;AACA;AACA;AACA;AACA;;ACNA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/GA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"generated.js","sourceRoot":"","sourcesContent":["(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require==\"function\"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error(\"Cannot find module '\"+o+\"'\");throw f.code=\"MODULE_NOT_FOUND\",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require==\"function\"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})",";(function () {\n\n  var object = typeof exports != 'undefined' ? exports : self; // #8: web workers\n  var chars = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=';\n\n  function InvalidCharacterError(message) {\n    this.message = message;\n  }\n  InvalidCharacterError.prototype = new Error;\n  InvalidCharacterError.prototype.name = 'InvalidCharacterError';\n\n  // encoder\n  // [https://gist.github.com/999166] by [https://github.com/nignag]\n  object.btoa || (\n  object.btoa = function (input) {\n    var str = String(input);\n    for (\n      // initialize result and counter\n      var block, charCode, idx = 0, map = chars, output = '';\n      // if the next str index does not exist:\n      //   change the mapping table to \"=\"\n      //   check if d has no fractional digits\n      str.charAt(idx | 0) || (map = '=', idx % 1);\n      // \"8 - idx % 1 * 8\" generates the sequence 2, 4, 6, 8\n      output += map.charAt(63 & block >> 8 - idx % 1 * 8)\n    ) {\n      charCode = str.charCodeAt(idx += 3/4);\n      if (charCode > 0xFF) {\n        throw new InvalidCharacterError(\"'btoa' failed: The string to be encoded contains characters outside of the Latin1 range.\");\n      }\n      block = block << 8 | charCode;\n    }\n    return output;\n  });\n\n  // decoder\n  // [https://gist.github.com/1020396] by [https://github.com/atk]\n  object.atob || (\n  object.atob = function (input) {\n    var str = String(input).replace(/=+$/, '');\n    if (str.length % 4 == 1) {\n      throw new InvalidCharacterError(\"'atob' failed: The string to be decoded is not correctly encoded.\");\n    }\n    for (\n      // initialize result and counters\n      var bc = 0, bs, buffer, idx = 0, output = '';\n      // get next character\n      buffer = str.charAt(idx++);\n      // character found in table? initialize bit storage and add its ascii value;\n      ~buffer && (bs = bc % 4 ? bs * 64 + buffer : buffer,\n        // and if not first of each 4 characters,\n        // convert the first 8 bits to one ascii character\n        bc++ % 4) ? output += String.fromCharCode(255 & bs >> (-2 * bc & 6)) : 0\n    ) {\n      // try to find character in table (0-63, not found => -1)\n      buffer = chars.indexOf(buffer);\n    }\n    return output;\n  });\n\n}());\n","'use strict';\n/* eslint-disable no-unused-vars */\nvar hasOwnProperty = Object.prototype.hasOwnProperty;\nvar propIsEnumerable = Object.prototype.propertyIsEnumerable;\n\nfunction toObject(val) {\n\tif (val === null || val === undefined) {\n\t\tthrow new TypeError('Object.assign cannot be called with null or undefined');\n\t}\n\n\treturn Object(val);\n}\n\nfunction shouldUseNative() {\n\ttry {\n\t\tif (!Object.assign) {\n\t\t\treturn false;\n\t\t}\n\n\t\t// Detect buggy property enumeration order in older V8 versions.\n\n\t\t// https://bugs.chromium.org/p/v8/issues/detail?id=4118\n\t\tvar test1 = new String('abc');  // eslint-disable-line\n\t\ttest1[5] = 'de';\n\t\tif (Object.getOwnPropertyNames(test1)[0] === '5') {\n\t\t\treturn false;\n\t\t}\n\n\t\t// https://bugs.chromium.org/p/v8/issues/detail?id=3056\n\t\tvar test2 = {};\n\t\tfor (var i = 0; i < 10; i++) {\n\t\t\ttest2['_' + String.fromCharCode(i)] = i;\n\t\t}\n\t\tvar order2 = Object.getOwnPropertyNames(test2).map(function (n) {\n\t\t\treturn test2[n];\n\t\t});\n\t\tif (order2.join('') !== '0123456789') {\n\t\t\treturn false;\n\t\t}\n\n\t\t// https://bugs.chromium.org/p/v8/issues/detail?id=3056\n\t\tvar test3 = {};\n\t\t'abcdefghijklmnopqrst'.split('').forEach(function (letter) {\n\t\t\ttest3[letter] = letter;\n\t\t});\n\t\tif (Object.keys(Object.assign({}, test3)).join('') !==\n\t\t\t\t'abcdefghijklmnopqrst') {\n\t\t\treturn false;\n\t\t}\n\n\t\treturn true;\n\t} catch (e) {\n\t\t// We don't expect any of the above to throw, but better to be safe.\n\t\treturn false;\n\t}\n}\n\nmodule.exports = shouldUseNative() ? Object.assign : function (target, source) {\n\tvar from;\n\tvar to = toObject(target);\n\tvar symbols;\n\n\tfor (var s = 1; s < arguments.length; s++) {\n\t\tfrom = Object(arguments[s]);\n\n\t\tfor (var key in from) {\n\t\t\tif (hasOwnProperty.call(from, key)) {\n\t\t\t\tto[key] = from[key];\n\t\t\t}\n\t\t}\n\n\t\tif (Object.getOwnPropertySymbols) {\n\t\t\tsymbols = Object.getOwnPropertySymbols(from);\n\t\t\tfor (var i = 0; i < symbols.length; i++) {\n\t\t\t\tif (propIsEnumerable.call(from, symbols[i])) {\n\t\t\t\t\tto[symbols[i]] = from[symbols[i]];\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\t}\n\n\treturn to;\n};\n","module.exports={\n    \"version\": \"v2\"\n}\n","/**\n * Epicenter Javascript libraries\n * v<%= version %>\n * https://github.com/forio/epicenter-js-libs\n */\n\nvar F = {\n    _private: {}, //need this hook now because tests expect everything to be global. Delete once tests are browserified\n    util: {},\n    factory: {},\n    transport: {},\n    store: {},\n    service: {},\n    manager: {\n        strategy: {}\n    },\n\n};\n\nF.load = require('./env-load');\n\nif (!global.SKIP_ENV_LOAD) {\n    F.load();\n}\n\nF.util.query = require('./util/query-util');\nF.util.run = require('./util/run-util');\nF.util.classFrom = require('./util/inherit');\nF._private.strategyutils = require('./managers/strategy-utils');\n\nF.factory.Transport = require('./transport/http-transport-factory');\nF.transport.Ajax = require('./transport/ajax-http-transport');\n\nF.service.URL = require('./service/url-config-service');\nF.service.Config = require('./service/configuration-service');\nF.service.Run = require('./service/run-api-service');\nF.service.File = require('./service/admin-file-service');\nF.service.Variables = require('./service/variables-api-service');\nF.service.Data = require('./service/data-api-service');\nF.service.Auth = require('./service/auth-api-service');\nF.service.World = require('./service/world-api-adapter');\nF.service.State = require('./service/state-api-adapter');\nF.service.User = require('./service/user-api-adapter');\nF.service.Member = require('./service/member-api-adapter');\nF.service.Asset = require('./service/asset-api-adapter');\nF.service.Group = require('./service/group-api-service');\nF.service.Introspect = require('./service/introspection-api-service');\nF.service.Presence = require('./service/presence-api-service');\n\nF.store.Cookie = require('./store/cookie-store');\nF.factory.Store = require('./store/store-factory');\n\nF.manager.ScenarioManager = require('./managers/scenario-manager');\nF.manager.RunManager = require('./managers/run-manager');\nF.manager.AuthManager = require('./managers/auth-manager');\nF.manager.WorldManager = require('./managers/world-manager');\nF.manager.SavedRunsManager = require('./managers/saved-runs-manager');\n\nvar strategies = require('./managers/run-strategies');\nF.manager.strategy = strategies.list; //TODO: this is not really a manager so namespace this better\n\nF.manager.ChannelManager = require('./managers/epicenter-channel-manager');\nF.service.Channel = require('./service/channel-service');\n\nF.version = '<%= version %>';\nF.api = require('./api-version.json');\n\nF.constants = require('./managers/key-names');\n\nglobal.F = F;\nmodule.exports = F;\n","'use strict';\n\nvar URLConfigService = require('./service/url-config-service');\n\nvar envLoad = function (callback) {\n    var urlService = new URLConfigService();\n    var infoUrl = urlService.getAPIPath('config');\n    var envPromise = $.ajax({ url: infoUrl, async: false });\n    envPromise = envPromise.then(function (res) {\n        var overrides = res.api;\n        URLConfigService.defaults = $.extend(URLConfigService.defaults, overrides);\n    });\n    return envPromise.then(callback).fail(callback);\n};\n\nmodule.exports = envLoad;\n","/**\n* ## Authorization Manager\n*\n* The Authorization Manager provides an easy way to manage user authentication (logging in and out) and authorization (keeping track of tokens, sessions, and groups) for projects.\n*\n* The Authorization Manager is most useful for [team projects](../../../glossary/#team) with an access level of [Authenticated](../../../glossary/#access). These projects are accessed by [end users](../../../glossary/#users) who are members of one or more [groups](../../../glossary/#groups).\n*\n* #### Using the Authorization Manager\n*\n* To use the Authorization Manager, instantiate it. Then, make calls to any of the methods you need:\n*\n*       var authMgr = new F.manager.AuthManager({\n*           account: 'acme-simulations',\n*           userName: 'enduser1',\n*           password: 'passw0rd'\n*       });\n*       authMgr.login().then(function () {\n*           authMgr.getCurrentUserSessionInfo();\n*       });\n*\n*\n* The `options` object passed to the `F.manager.AuthManager()` call can include:\n*\n*   * `account`: The account id for this `userName`. In the Epicenter UI, this is the **Team ID** (for team projects) or the **User ID** (for personal projects).\n*   * `userName`: Email or username to use for logging in.\n*   * `password`: Password for specified `userName`.\n*   * `project`: The **Project ID** for the project to log this user into. Optional.\n*   * `groupId`: Id of the group to which `userName` belongs. Required for end users if the `project` is specified.\n*\n* If you prefer starting from a template, the Epicenter JS Libs [Login Component](../../#components) uses the Authorization Manager as well. This sample HTML page (and associated CSS and JS files) provides a login form for team members and end users of your project. It also includes a group selector for end users that are members of multiple groups.\n*/\n\n'use strict';\nvar AuthAdapter = require('../service/auth-api-service');\nvar MemberAdapter = require('../service/member-api-adapter');\nvar GroupService = require('../service/group-api-service');\nvar SessionManager = require('../store/session-manager');\nvar _pick = require('../util/object-util')._pick;\nvar objectAssign = require('object-assign');\n\nvar atob = window.atob || require('Base64').atob;\n\nvar defaults = {\n    requiresGroup: true\n};\n\nfunction AuthManager(options) {\n    options = $.extend(true, {}, defaults, options);\n    this.sessionManager = new SessionManager(options);\n    this.options = this.sessionManager.getMergedOptions();\n\n    this.authAdapter = new AuthAdapter(this.options);\n}\n\nvar _findUserInGroup = function (members, id) {\n    for (var j = 0; j < members.length; j++) {\n        if (members[j].userId === id) {\n            return members[j];\n        }\n    }\n    return null;\n};\n\nAuthManager.prototype = $.extend(AuthManager.prototype, {\n\n    /**\n    * Logs user in.\n    *\n    * **Example**\n    *\n    *       authMgr.login({\n    *           account: 'acme-simulations',\n    *           project: 'supply-chain-game',\n    *           userName: 'enduser1',\n    *           password: 'passw0rd'\n    *       })\n    *           .then(function(statusObj) {\n    *               // if enduser1 belongs to exactly one group\n    *               // (or if the login() call is modified to include the group id)\n    *               // continue here\n    *           })\n    *           .fail(function(statusObj) {\n    *               // if enduser1 belongs to multiple groups,\n    *               // the login() call fails\n    *               // and returns all groups of which the user is a member\n    *               for (var i=0; i < statusObj.userGroups.length; i++) {\n    *                   console.log(statusObj.userGroups[i].name, statusObj.userGroups[i].groupId);\n    *               }\n    *           });\n    *\n    * **Parameters**\n    *\n    * @param {Object} options (Optional) Overrides for configuration options. If not passed in when creating an instance of the manager (`F.manager.AuthManager()`), these options should include:\n    * @param {string} options.account The account id for this `userName`. In the Epicenter UI, this is the **Team ID** (for team projects) or the **User ID** (for personal projects).\n    * @param {string} options.userName Email or username to use for logging in.\n    * @param {string} options.password Password for specified `userName`.\n    * @param {string} options.project (Optional) The **Project ID** for the project to log this user into.\n    * @param {string} options.groupId The id of the group to which `userName` belongs. Required for [end users](../../../glossary/#users) if the `project` is specified and if the end users are members of multiple [groups](../../../glossary/#groups), otherwise optional.\n    * @return {Promise}\n    */\n    login: function (options) {\n        var me = this;\n        var $d = $.Deferred();\n        var sessionManager = this.sessionManager;\n        var adapterOptions = sessionManager.getMergedOptions({ success: $.noop, error: $.noop }, options);\n        var outSuccess = adapterOptions.success;\n        var outError = adapterOptions.error;\n        var groupId = adapterOptions.groupId;\n\n        var decodeToken = function (token) {\n            var encoded = token.split('.')[1];\n            while (encoded.length % 4 !== 0) { //eslint-disable-line\n                encoded += '=';\n            }\n            return JSON.parse(atob(encoded));\n        };\n\n        var handleGroupError = function (message, statusCode, data, type) {\n            // logout the user since it's in an invalid state with no group selected\n            me.logout().then(function () {\n                var error = $.extend(true, {}, data, { statusText: message, status: statusCode, type: type });\n                $d.reject(error);\n            });\n        };\n\n        var handleSuccess = function (response) {\n            var token = response.access_token;\n            var userInfo = decodeToken(token);\n            var oldGroups = sessionManager.getSession(adapterOptions).groups || {};\n            var userGroupOpts = $.extend(true, {}, adapterOptions, { success: $.noop });\n            var data = { auth: response, user: userInfo };\n            var project = adapterOptions.project;\n            var isTeamMember = userInfo.parent_account_id === null;\n            var requiresGroup = adapterOptions.requiresGroup && project;\n\n            var userName = (userInfo.user_name || '').split('/')[0]; //of form <user>/<team>\n            var sessionInfo = {\n                auth_token: token,\n                account: adapterOptions.account,\n                project: project,\n                userId: userInfo.user_id,\n                groups: oldGroups,\n                isTeamMember: isTeamMember,\n                userName: userName,\n            };\n            // The group is not required if the user is not logging into a project\n            if (!requiresGroup) {\n                sessionManager.saveSession(sessionInfo);\n                outSuccess.apply(this, [data]);\n                $d.resolve(data);\n                return;\n            }\n\n            var handleGroupList = function (groupList) {\n                data.userGroups = groupList;\n\n                var group = null;\n                if (groupList.length === 0) {\n                    handleGroupError('The user has no groups associated in this account', 403, data, 'NO_GROUPS');\n                    return;\n                } else if (groupList.length === 1) {\n                    // Select the only group\n                    group = groupList[0];\n                } else if (groupList.length > 1) {\n                    if (groupId) {\n                        var filteredGroups = $.grep(groupList, function (resGroup) {\n                            return resGroup.groupId === groupId;\n                        });\n                        group = filteredGroups.length === 1 ? filteredGroups[0] : null;\n                    }\n                }\n\n                if (group) {\n                    // A team member does not get the group members because is calling the Group API\n                    // but it's automatically a fac user\n                    var isFac = isTeamMember ? true : _findUserInGroup(group.members, userInfo.user_id).role === 'facilitator';\n                    var groupData = {\n                        groupId: group.groupId,\n                        groupName: group.name,\n                        isFac: isFac\n                    };\n                    var sessionInfoWithGroup = objectAssign({}, sessionInfo, groupData);\n                    sessionInfo.groups[project] = groupData;\n                    me.sessionManager.saveSession(sessionInfoWithGroup, adapterOptions);\n                    outSuccess.apply(this, [data]);\n                    $d.resolve(data);\n                } else {\n                    handleGroupError('This user is associated with more than one group. Please specify a group id to log into and try again', 403, data, 'MULTIPLE_GROUPS');\n                }\n            };\n\n            if (!isTeamMember) {\n                me.getUserGroups({ userId: userInfo.user_id, token: token }, userGroupOpts)\n                    .then(handleGroupList, $d.reject);\n            } else {\n                var opts = objectAssign({}, userGroupOpts, { token: token });\n                var groupService = new GroupService(opts);\n                groupService.getGroups({ account: adapterOptions.account, project: project })\n                    .then(function (groups) {\n                        // Group API returns id instead of groupId\n                        groups.forEach(function (group) {\n                            group.groupId = group.id;\n                        });\n\n                        if (groups.length) {\n                            handleGroupList(groups);\n                        } else {\n                            //either it's a private project or there are no groups\n                            sessionManager.saveSession(sessionInfo);\n                            outSuccess.apply(this, [data]);\n                            $d.resolve(data);\n                            return;\n                        }\n                    }, $d.reject);\n            }\n        };\n\n        adapterOptions.success = handleSuccess;\n        adapterOptions.error = function (response) {\n            if (adapterOptions.account) {\n                // Try to login as a system user\n                adapterOptions.account = null;\n                adapterOptions.error = function () {\n                    outError.apply(this, arguments);\n                    $d.reject(response);\n                };\n\n                me.authAdapter.login(adapterOptions);\n                return;\n            }\n\n            outError.apply(this, arguments);\n            $d.reject(response);\n        };\n\n        this.authAdapter.login(adapterOptions);\n        return $d.promise();\n    },\n\n    /**\n    * Logs user out by clearing all session information.\n    *\n    * **Example**\n    *\n    *       authMgr.logout();\n    *\n    * **Parameters**\n    *\n    * @param {Object} options (Optional) Overrides for configuration options.\n    * @return {Promise}\n    */\n    logout: function (options) {\n        var me = this;\n        var adapterOptions = this.sessionManager.getMergedOptions(options);\n\n        var removeCookieFn = function (response) {\n            me.sessionManager.removeSession();\n        };\n\n        return this.authAdapter.logout(adapterOptions).then(removeCookieFn);\n    },\n\n    /**\n     * Returns the existing user access token if the user is already logged in. Otherwise, logs the user in, creating a new user access token, and returns the new token. (See [more background on access tokens](../../../project_access/)).\n     *\n     * **Example**\n     *\n     *      authMgr.getToken()\n     *          .then(function (token) {\n     *              console.log('My token is ', token);\n     *          });\n     *\n     * **Parameters**\n     * @param {Object} options (Optional) Overrides for configuration options.\n     * @return {Promise}\n     */\n    getToken: function (options) {\n        var httpOptions = this.sessionManager.getMergedOptions(options);\n\n        var session = this.sessionManager.getSession(httpOptions);\n        var $d = $.Deferred();\n        if (session.auth_token) {\n            $d.resolve(session.auth_token);\n        } else {\n            this.login(httpOptions).then($d.resolve);\n        }\n        return $d.promise();\n    },\n\n    /**\n     * Returns an array of group records, one for each group of which the current user is a member. Each group record includes the group `name`, `account`, `project`, and `groupId`.\n     *\n     * If some end users in your project are members of multiple groups, this is a useful method to call on your project's login page. When the user attempts to log in, you can use this to display the groups of which the user is member, and have the user select the correct group to log in to for this session.\n     *\n     * **Example**\n     *\n     *      // get groups for current user\n     *      var sessionObj = authMgr.getCurrentUserSessionInfo();\n     *      authMgr.getUserGroups({ userId: sessionObj.userId, token: sessionObj.auth_token })\n     *          .then(function (groups) {\n     *              for (var i=0; i < groups.length; i++)\n     *                  { console.log(groups[i].name); }\n     *          });\n     *\n     *      // get groups for particular user\n     *      authMgr.getUserGroups({userId: 'b1c19dda-2d2e-4777-ad5d-3929f17e86d3', token: savedProjAccessToken });\n     *\n     * **Parameters**\n     * @param {Object} params Object with a userId and token properties.\n     * @param {String} params.userId The userId. If looking up groups for the currently logged in user, this is in the session information. Otherwise, pass a string.\n     * @param {String} params.token The authorization credentials (access token) to use for checking the groups for this user. If looking up groups for the currently logged in user, this is in the session information. A team member's token or a project access token can access all the groups for all end users in the team or project.\n     * @param {Object} options (Optional) Overrides for configuration options.\n     * @return {Promise}\n     */\n    getUserGroups: function (params, options) {\n        var adapterOptions = this.sessionManager.getMergedOptions({ success: $.noop }, options);\n        var $d = $.Deferred();\n        var outSuccess = adapterOptions.success;\n\n        adapterOptions.success = function (memberInfo) {\n            // The member API is at the account scope, we filter by project\n            if (adapterOptions.project) {\n                memberInfo = $.grep(memberInfo, function (group) {\n                    return group.project === adapterOptions.project;\n                });\n            }\n\n            outSuccess.apply(this, [memberInfo]);\n            $d.resolve(memberInfo);\n        };\n\n        var memberAdapter = new MemberAdapter({ token: params.token, server: adapterOptions.server });\n        memberAdapter.getGroupsForUser(params, adapterOptions).fail($d.reject);\n        return $d.promise();\n    },\n\n    /**\n     * Helper method to check if you're currently logged in\n     *\n     *  **Example**\n     *  \n     *      var amILoggedIn = authMgr.isLoggedIn();\n     *\n     * **Parameters**\n     * @param {none} none\n     * @return {Boolean} true if you're logged in\n     */\n    isLoggedIn: function () {\n        var session = this.getCurrentUserSessionInfo();\n        return !!(session && session.userId);\n    },\n\n    /**\n     * Returns session information for the current user, including the `userId`, `account`, `project`, `groupId`, `groupName`, `isFac` (whether the end user is a facilitator of this group), and `auth_token` (user access token).\n     *\n     * *Important*: This method is synchronous. The session information is returned immediately in an object; no callbacks or promises are needed.\n     *\n     * Session information is stored in a cookie in the browser.\n     *\n     * **Example**\n     *\n     *      var sessionObj = authMgr.getCurrentUserSessionInfo();\n     *\n     * **Parameters**\n     * @param {Object} options (Optional) Overrides for configuration options.\n     * @return {Object} session information\n     */\n    getCurrentUserSessionInfo: function (options) {\n        var adapterOptions = this.sessionManager.getMergedOptions({ success: $.noop }, options);\n        return this.sessionManager.getSession(adapterOptions);\n    },\n\n    /*\n     * Adds one or more groups to the current session. \n     *\n     * This method assumes that the project and group exist and the user specified in the session is part of this project and group.\n     *\n     * Returns the new session object.\n     *\n     * **Example**\n     *\n     *      authMgr.addGroups({ project: 'hello-world', groupName: 'groupName', groupId: 'groupId' });\n     *      authMgr.addGroups([{ project: 'hello-world', groupName: 'groupName', groupId: 'groupId' }, { project: 'hello-world', groupName: '...' }]);\n     *\n     * **Parameters**\n     * @param {object|array} groups (Required) The group object must contain the `project` (**Project ID**) and `groupName` properties. If passing an array of such objects, all of the objects must contain *different* `project` (**Project ID**) values: although end users may be logged in to multiple projects at once, they may only be logged in to one group per project at a time.\n     * @param {string} group.isFac (optional) Defaults to `false`. Set to `true` if the user in the session should be a facilitator in this group.\n     * @param {string} group.groupId (optional) Defaults to undefined. Needed mostly for the Members API.\n     * @return {Object} session information\n    */\n    addGroups: function (groups) {\n        var session = this.getCurrentUserSessionInfo();\n        var isArray = Array.isArray(groups);\n        groups = isArray ? groups : [groups];\n\n        $.each(groups, function (index, group) {\n            var extendedGroup = $.extend({}, { isFac: false }, group);\n            var project = extendedGroup.project;\n            var validProps = ['groupName', 'groupId', 'isFac'];\n            if (!project || !extendedGroup.groupName) {\n                throw new Error('No project or groupName specified.');\n            }\n            // filter object\n            extendedGroup = _pick(extendedGroup, validProps);\n            session.groups[project] = extendedGroup;\n        });\n        this.sessionManager.saveSession(session);\n        return session;\n    }\n});\n\nmodule.exports = AuthManager;\n","'use strict';\n\n/**\n * ## Channel Manager\n *\n * There are two main use cases for the channel: event notifications and chat messages.\n *\n * If you are developing with Epicenter.js, you should use the [Epicenter Channel Manager](../epicenter-channel-manager/) rather than this more generic Channel Manager. (The Epicenter Channel Manager is a wrapper that instantiates a Channel Manager with Epicenter-specific defaults.) The Epicenter Channel Manager documentation also has more [background](../epicenter-channel-manager/#background) information on channels and their use. \n *\n * However, you can work directly with the Channel Manager if you like. (This might be useful if you are working through Node.js, for example, `require('manager/channel-manager')`.)\n *\n * The Channel Manager is a wrapper around the default [cometd JavaScript library](http://docs.cometd.org/2/reference/javascript.html), `$.cometd`. It provides a few nice features that `$.cometd` doesn't, including:\n *\n * * Automatic re-subscription to channels if you lose your connection\n * * Online / Offline notifications\n * * 'Events' for cometd notifications (instead of having to listen on specific meta channels)\n *\n * You'll need to include the `epicenter-multiplayer-dependencies.js` library in addition to the `epicenter.js` library in your project to use the Channel Manager. (See [Including Epicenter.js](../../#include).)\n *\n * To use the Channel Manager in client-side JavaScript, instantiate the [Epicenter Channel Manager](../epicenter-channel-manager/), get a particular channel -- that is, an instance of a [Channel Service](../channel-service/) -- then use the channel's `subscribe()` and `publish()` methods to subscribe to topics or publish data to topics.\n *\n *      var cm = new F.manager.ChannelManager();\n *      var gc = cm.getGroupChannel();\n *      // because we used an Epicenter Channel Manager to get the group channel,\n *      // subscribe() and publish() here default to the base topic for the group;\n *      gc.subscribe('', function(data) { console.log(data); });\n *      gc.publish('', { message: 'a new message to the group' });\n *\n * The parameters for instantiating a Channel Manager include:\n *\n * * `options` The options object to configure the Channel Manager. Besides the common options listed here, see http://docs.cometd.org/reference/javascript.html for other supported options.\n * * `options.url` The Cometd endpoint URL.\n * * `options.websocketEnabled` Whether websocket support is active (boolean).\n * * `options.channel` Other defaults to pass on to instances of the underlying Channel Service. See [Channel Service](../channel-service/) for details.\n *\n */\n\nvar Channel = require('../service/channel-service');\nvar SessionManager = require('../store/session-manager');\n\nvar ChannelManager = function (options) {\n    if (!$.cometd) {\n        throw new Error('Cometd library not found. Please include epicenter-multiplayer-dependencies.js');\n    }\n    if (!options || !options.url) {\n        throw new Error('Please provide an url for the cometd server');\n    }\n\n    var defaults = {\n        /**\n         * The Cometd endpoint URL.\n         * @type {string}\n         */\n        url: '',\n\n        /**\n         * The log level for the channel (logs to console).\n         * @type {string}\n         */\n        logLevel: 'info',\n\n        /**\n         * Whether websocket support is active. Defaults to `true`.\n         * @type {boolean}\n         */\n        websocketEnabled: true,\n\n        /**\n         * Whether the ACK extension is enabled. Defaults to `true`. See [https://docs.cometd.org/current/reference/#_extensions_acknowledge](https://docs.cometd.org/current/reference/#_extensions_acknowledge) for more info.\n         * @type {boolean}\n         */\n        ackEnabled: true,\n\n        /**\n         * If false each instance of Channel will have a separate cometd connection to server, which could be noisy. Set to true to re-use the same connection across instances.\n         * @type {boolean}\n         */\n        shareConnection: false,\n\n        /**\n         * Other defaults to pass on to instances of the underlying [Channel Service](../channel-service/), which are created through `getChannel()`.\n         * @type {object}\n         */\n        channel: {\n\n        },\n\n        /**\n         * Options to pass to the channel handshake.\n         *\n         * For example, the [Epicenter Channel Manager](../epicenter-channel-manager/) passes `ext` and authorization information. More information on possible options is in the details of the underlying [Push Channel API](../../../rest_apis/multiplayer/channel/).\n         *\n         * @type {object}\n         */\n        handshake: undefined\n    };\n    this.sessionManager = new SessionManager();\n    var defaultCometOptions = this.sessionManager.getMergedOptions(defaults, options);\n    this.currentSubscriptions = [];\n    this.options = defaultCometOptions;\n\n    if (defaultCometOptions.shareConnection && ChannelManager.prototype._cometd) {\n        this.cometd = ChannelManager.prototype._cometd;\n        return this;\n    }\n    var cometd = new $.CometD();\n    ChannelManager.prototype._cometd = cometd;\n\n    cometd.websocketEnabled = defaultCometOptions.websocketEnabled;\n    cometd.ackEnabled = defaultCometOptions.ackEnabled;\n\n    this.isConnected = false;\n    var connectionBroken = function (message) {\n        $(this).trigger('disconnect', message);\n    };\n    var connectionSucceeded = function (message) {\n        $(this).trigger('connect', message);\n    };\n    var me = this;\n\n    cometd.configure(defaultCometOptions);\n\n    cometd.addListener('/meta/connect', function (message) {\n        var wasConnected = this.isConnected;\n        this.isConnected = (message.successful === true);\n        if (!wasConnected && this.isConnected) { //Connecting for the first time\n            connectionSucceeded.call(this, message);\n        } else if (wasConnected && !this.isConnected) { //Only throw disconnected message fro the first disconnect, not once per try\n            connectionBroken.call(this, message);\n        }\n    }.bind(this));\n\n    cometd.addListener('/meta/disconnect', connectionBroken);\n\n    cometd.addListener('/meta/handshake', function (message) {\n        if (message.successful) {\n            //http://docs.cometd.org/reference/javascript_subscribe.html#javascript_subscribe_meta_channels\n            // ^ \"dynamic subscriptions are cleared (like any other subscription) and the application needs to figure out which dynamic subscription must be performed again\"\n            cometd.batch(function () {\n                $(me.currentSubscriptions).each(function (index, subs) {\n                    cometd.resubscribe(subs);\n                });\n            });\n        }\n    });\n\n    //Other interesting events for reference\n    cometd.addListener('/meta/subscribe', function (message) {\n        $(me).trigger('subscribe', message);\n    });\n    cometd.addListener('/meta/unsubscribe', function (message) {\n        $(me).trigger('unsubscribe', message);\n    });\n    cometd.addListener('/meta/publish', function (message) {\n        $(me).trigger('publish', message);\n    });\n    cometd.addListener('/meta/unsuccessful', function (message) {\n        $(me).trigger('error', message);\n    });\n\n    cometd.handshake(defaultCometOptions.handshake);\n\n    this.cometd = cometd;\n};\n\n\nChannelManager.prototype = $.extend(ChannelManager.prototype, {\n\n    /**\n     * Creates and returns a channel, that is, an instance of a [Channel Service](../channel-service/).\n     *\n     * **Example**\n     *\n     *      var cm = new F.manager.ChannelManager();\n     *      var channel = cm.getChannel();\n     *\n     *      channel.subscribe('topic', callback);\n     *      channel.publish('topic', { myData: 100 });\n     *\n     * **Parameters**\n     * @param {Object|String} options (Optional) If string, assumed to be the base channel url. If object, assumed to be configuration options for the constructor.\n     * @return {Channel} Channel instance\n     */\n    getChannel: function (options) {\n        //If you just want to pass in a string\n        if (options && !$.isPlainObject(options)) {\n            options = {\n                base: options\n            };\n        }\n        var defaults = {\n            transport: this.cometd\n        };\n        var channel = new Channel($.extend(true, {}, this.options.channel, defaults, options));\n\n\n        //Wrap subs and unsubs so we can use it to re-attach handlers after being disconnected\n        var subs = channel.subscribe;\n        channel.subscribe = function () {\n            var subid = subs.apply(channel, arguments);\n            this.currentSubscriptions = this.currentSubscriptions.concat(subid);\n            return subid;\n        }.bind(this);\n\n\n        var unsubs = channel.unsubscribe;\n        channel.unsubscribe = function () {\n            var removed = unsubs.apply(channel, arguments);\n            for (var i = 0; i < this.currentSubscriptions.length; i++) {\n                if (this.currentSubscriptions[i].id === removed.id) {\n                    this.currentSubscriptions.splice(i, 1);\n                }\n            }\n            return removed;\n        }.bind(this);\n\n        return channel;\n    },\n\n    /**\n     * Start listening for events on this instance. Signature is same as for jQuery Events: http://api.jquery.com/on/.\n     *\n     * Supported events are: `connect`, `disconnect`, `subscribe`, `unsubscribe`, `publish`, `error`.\n     *\n     * **Parameters**\n     *\n     * @param {string} event The event type. See more detail at jQuery Events: http://api.jquery.com/on/.\n     */\n    on: function (event) {\n        $(this).on.apply($(this), arguments);\n    },\n\n    /**\n     * Stop listening for events on this instance. Signature is same as for jQuery Events: http://api.jquery.com/off/.\n     *\n     * **Parameters**\n     *\n     * @param {string} event The event type. See more detail at jQuery Events: http://api.jquery.com/off/.\n     */\n    off: function (event) {\n        $(this).off.apply($(this), arguments);\n    },\n\n    /**\n     * Trigger events and execute handlers. Signature is same as for jQuery Events: http://api.jquery.com/trigger/.\n     *\n     * **Parameters**\n     *\n     * @param {string} event The event type. See more detail at jQuery Events: http://api.jquery.com/trigger/.\n     */\n    trigger: function (event) {\n        $(this).trigger.apply($(this), arguments);\n    }\n});\n\nmodule.exports = ChannelManager;\n","'use strict';\n\n/**\n * ## Epicenter Channel Manager\n *\n * The Epicenter platform provides a push channel, which allows you to publish and subscribe to messages within a [project](../../../glossary/#projects), [group](../../../glossary/#groups), or [multiplayer world](../../../glossary/#world).\n *\n * <a name=\"background\"></a>\n * ### Channel Background\n *\n * Channel notifications are only available for [team projects](../../../glossary/#team). There are two main use cases for the push channel: event notifications and chat messages.\n *\n * #### Event Notifications\n *\n * Within a [multiplayer simulation or world](../../../glossary/#world), it is often useful for your project's [model](../../../writing_your_model/) to alert the [user interface (browser)](../../../creating_your_interface/) that something new has happened.\n *\n * Usually, this \"something new\" is an event within the project, group, or world, such as:\n *\n * * An end user comes online (logs in) or goes offline. (This is especially interesting in a multiplayer world; only available if you have [enabled authorization](../../../updating_your_settings/#general-settings) for the channel.)\n * * An end user is assigned to a world.\n * * An end user updates a variable / makes a decision.\n * * An end user creates or updates data stored in the [Data API](../data-api-service/).\n * * An operation (method) is called. (This is especially interesting if the model is advanced, for instance, the Vensim `step` operation is called.)\n *\n * When these events occur, you often want to have the user interface for one or more end users automatically update with new information.\n *\n * #### Chat Messages\n *\n * Another reason to use the push channel is to allow players (end users) to send chat messages to other players, and to have those messages appear immediately.\n *\n * #### Getting Started\n *\n * For both the event notification and chat message use cases:\n *\n * * First, enable channel notifications for your project.\n *      * Channel notifications are only available for [team projects](../../../glossary/#team). To enable notifications for your project, [update your project settings](../../../updating_your_settings/#general-settings) to turn on the **Push Channel** setting, and optionally require authorization for the channel.\n * * Then, instantiate an Epicenter Channel Manager.\n * * Next, get the channel with the scope you want (user, world, group, data).\n * * Finally, use the channel's `subscribe()` and `publish()` methods to subscribe to topics or publish data to topics.\n *\n * Here's an example of those last three steps (instantiate, get channel, subscribe):\n *\n *     var cm = new F.manager.ChannelManager();\n *     var gc = cm.getGroupChannel();\n *     gc.subscribe('', function(data) { console.log(data); });\n *     gc.publish('', { message: 'a new message to the group' });\n *\n * For a more detailed example, see a [complete publish and subscribe example](../../../rest_apis/multiplayer/channel/#epijs-example).\n *\n * For details on what data is published automatically to which channels, see [Automatic Publishing of Events](../../../rest_apis/multiplayer/channel/#publish-message-auto).\n *\n * #### Creating an Epicenter Channel Manager\n *\n * The Epicenter Channel Manager is a wrapper around the (more generic) [Channel Manager](../channel-manager/), to instantiate it with Epicenter-specific defaults. If you are interested in including a notification or chat feature in your project, using an Epicenter Channel Manager is the easiest way to get started.\n *\n * You'll need to include the `epicenter-multiplayer-dependencies.js` library in addition to the `epicenter.js` library in your project to use the Epicenter Channel Manager. See [Including Epicenter.js](../../#include).\n *\n * The parameters for instantiating an Epicenter Channel Manager include:\n *\n * * `options` Object with details about the Epicenter project for this Epicenter Channel Manager instance.\n * * `options.account` The Epicenter account id (**Team ID** for team projects, **User ID** for personal projects).\n * * `options.project` Epicenter project id.\n * * `options.userName` Epicenter userName used for authentication.\n * * `options.userId` Epicenter user id used for authentication. Optional; `options.userName` is preferred.\n * * `options.token` Epicenter token used for authentication. (You can retrieve this using `authManager.getToken()` from the [Authorization Manager](../auth-manager/).)\n * * `options.allowAllChannels` If not included or if set to `false`, all channel paths are validated; if your project requires [Push Channel Authorization](../../../updating_your_settings/), you should use this option. If you want to allow other channel paths, set to `true`; this is not common.\n */\n\nvar ChannelManager = require('./channel-manager');\nvar ConfigService = require('../service/configuration-service');\nvar classFrom = require('../util/inherit');\nvar SessionManager = require('../store/session-manager');\n\nvar validTypes = {\n    project: true,\n    group: true,\n    world: true,\n    user: true,\n    data: true,\n    general: true,\n    chat: true\n};\nvar getFromSessionOrError = function (value, sessionKeyName, settings) {\n    if (!value) {\n        if (settings && settings[sessionKeyName]) {\n            value = settings[sessionKeyName];\n        } else {\n            throw new Error(sessionKeyName + ' not found. Please log-in again, or specify ' + sessionKeyName + ' explicitly');\n        }\n    }\n    return value;\n};\n\nvar isPresenceData = function (payload) {\n    return payload.data && payload.data.type === 'user' && payload.data.user;\n};\n\nvar __super = ChannelManager.prototype;\nvar EpicenterChannelManager = classFrom(ChannelManager, {\n    constructor: function (options) {\n        this.sessionManager = new SessionManager(options);\n        var defaultCometOptions = this.sessionManager.getMergedOptions(options);\n\n        var urlConfig = new ConfigService(defaultCometOptions).get('server');\n        if (!defaultCometOptions.url) {\n            defaultCometOptions.url = urlConfig.getAPIPath('channel');\n        }\n\n        if (defaultCometOptions.handshake === undefined) {\n            var userName = defaultCometOptions.userName;\n            var userId = defaultCometOptions.userId;\n            var token = defaultCometOptions.token;\n            if ((userName || userId) && token) {\n                var userProp = userName ? 'userName' : 'userId';\n                var ext = {\n                    authorization: 'Bearer ' + token\n                };\n                ext[userProp] = userName ? userName : userId;\n\n                defaultCometOptions.handshake = {\n                    ext: ext\n                };\n            }\n        }\n\n        this.options = defaultCometOptions;\n        return __super.constructor.call(this, defaultCometOptions);\n    },\n\n    /**\n     * Creates and returns a channel, that is, an instance of a [Channel Service](../channel-service/).\n     *\n     * This method enforces Epicenter-specific channel naming: all channels requested must be in the form `/{type}/{account id}/{project id}/{...}`, where `type` is one of `run`, `data`, `user`, `world`, or `chat`.\n     *\n     * **Example**\n     *\n     *      var cm = new F.manager.EpicenterChannelManager();\n     *      var channel = cm.getChannel('/group/acme/supply-chain-game/');\n     *\n     *      channel.subscribe('topic', callback);\n     *      channel.publish('topic', { myData: 100 });\n     *\n     * **Parameters**\n     * @param {Object|String} options (Optional) If string, assumed to be the base channel url. If object, assumed to be configuration options for the constructor.\n     * @return {Channel} Channel instance\n     */\n    getChannel: function (options) {\n        if (options && typeof options !== 'object') {\n            options = {\n                base: options\n            };\n        }\n        var channelOpts = $.extend({}, this.options, options);\n        var base = channelOpts.base;\n        if (!base) {\n            throw new Error('No base topic was provided');\n        }\n\n        if (!channelOpts.allowAllChannels) {\n            var baseParts = base.split('/');\n            var channelType = baseParts[1];\n            if (baseParts.length < 4) { //eslint-disable-line\n                throw new Error('Invalid channel base name, it must be in the form /{type}/{account id}/{project id}/{...}');\n            }\n            if (!validTypes[channelType]) {\n                throw new Error('Invalid channel type');\n            }\n        }\n        return __super.getChannel.apply(this, arguments);\n    },\n\n    /**\n     * Create and return a publish/subscribe channel (from the underlying [Channel Manager](../channel-manager/)) for the given [group](../../../glossary/#groups). The group must exist in the account (team) and project provided.\n     *\n     * There are no notifications from Epicenter on this channel; all messages are user-originated.\n     *\n     * **Example**\n     *\n     *     var cm = new F.manager.ChannelManager();\n     *     var gc = cm.getGroupChannel();\n     *     gc.subscribe('broadcasts', callback);\n     *\n     * **Return Value**\n     *\n     * * *Channel* Returns the channel (an instance of the [Channel Service](../channel-service/)).\n     *\n     * **Parameters**\n     *\n     * @param  {String} groupName (Optional) Group to broadcast to. If not provided, picks up group from current session if end user is logged in.\n     * @return {Channel} Channel instance\n     */\n    getGroupChannel: function (groupName) {\n        var session = this.sessionManager.getMergedOptions(this.options);\n        groupName = getFromSessionOrError(groupName, 'groupName', session);\n        var account = getFromSessionOrError('', 'account', session);\n        var project = getFromSessionOrError('', 'project', session);\n\n        var baseTopic = ['/group', account, project, groupName].join('/');\n        var channel = __super.getChannel.call(this, { base: baseTopic });\n        var oldsubs = channel.subscribe;\n        channel.subscribe = function (topic, callback, context, options) {\n            var callbackWithoutPresenceData = function (payload) {\n                if (!isPresenceData(payload)) {\n                    callback.call(context, payload);\n                }\n            };\n            return oldsubs.call(channel, topic, callbackWithoutPresenceData, context, options);\n        };\n        return channel;\n    },\n\n    /**\n     * Create and return a publish/subscribe channel (from the underlying [Channel Manager](../channel-manager/)) for the given [world](../../../glossary/#world).\n     *\n     * This is typically used together with the [World Manager](../world-manager).\n     *\n     * **Example**\n     *\n     *     var cm = new F.manager.ChannelManager();\n     *     var worldManager = new F.manager.WorldManager({\n     *         account: 'acme-simulations',\n     *         project: 'supply-chain-game',\n     *         group: 'team1',\n     *         run: { model: 'model.eqn' }\n     *     });\n     *     worldManager.getCurrentWorld().then(function (worldObject, worldAdapter) {\n     *         var worldChannel = cm.getWorldChannel(worldObject);\n     *         worldChannel.subscribe('', function (data) {\n     *             console.log(data);\n     *         });\n     *      });\n     *\n     * **Return Value**\n     *\n     * * *Channel* Returns the channel (an instance of the [Channel Service](../channel-service/)).\n     *\n     * **Parameters**\n     *\n     * @param  {String|Object} world The world object or id.\n     * @param  {String} groupName (Optional) Group the world exists in. If not provided, picks up group from current session if end user is logged in.\n     * @return {Channel} Channel instance\n     */\n    getWorldChannel: function (world, groupName) {\n        var worldid = ($.isPlainObject(world) && world.id) ? world.id : world;\n        if (!worldid) {\n            throw new Error('Please specify a world id');\n        }\n        var session = this.sessionManager.getMergedOptions(this.options);\n\n        groupName = getFromSessionOrError(groupName, 'groupName', session);\n        var account = getFromSessionOrError('', 'account', session);\n        var project = getFromSessionOrError('', 'project', session);\n\n        var baseTopic = ['/world', account, project, groupName, worldid].join('/');\n        return __super.getChannel.call(this, { base: baseTopic });\n    },\n\n    /**\n     * Create and return a publish/subscribe channel (from the underlying [Channel Manager](../channel-manager/)) for the current [end user](../../../glossary/#users) in that user's current [world](../../../glossary/#world).\n     *\n     * This is typically used together with the [World Manager](../world-manager). Note that this channel only gets notifications for worlds currently in memory. (See more background on [persistence](../../../run_persistence).)\n     *\n     * **Example**\n     *\n     *     var cm = new F.manager.ChannelManager();\n     *     var worldManager = new F.manager.WorldManager({\n     *         account: 'acme-simulations',\n     *         project: 'supply-chain-game',\n     *         group: 'team1',\n     *         run: { model: 'model.eqn' }\n     *     });\n     *     worldManager.getCurrentWorld().then(function (worldObject, worldAdapter) {\n     *         var userChannel = cm.getUserChannel(worldObject);\n     *         userChannel.subscribe('', function (data) {\n     *             console.log(data);\n     *         });\n     *      });\n     *\n     *\n     * **Return Value**\n     *\n     * * *Channel* Returns the channel (an instance of the [Channel Service](../channel-service/)).\n     *\n     * **Parameters**\n     *\n     * @param  {String|Object} world World object or id.\n     * @param  {String|Object} user (Optional) User object or id. If not provided, picks up user id from current session if end user is logged in.\n     * @param  {String} groupName (Optional) Group the world exists in. If not provided, picks up group from current session if end user is logged in.\n     * @return {Channel} Channel instance\n     */\n    getUserChannel: function (world, user, groupName) {\n        var worldid = ($.isPlainObject(world) && world.id) ? world.id : world;\n        if (!worldid) {\n            throw new Error('Please specify a world id');\n        }\n        var session = this.sessionManager.getMergedOptions(this.options);\n\n        var userid = ($.isPlainObject(user) && user.id) ? user.id : user;\n        userid = getFromSessionOrError(userid, 'userId', session);\n        groupName = getFromSessionOrError(groupName, 'groupName', session);\n\n        var account = getFromSessionOrError('', 'account', session);\n        var project = getFromSessionOrError('', 'project', session);\n\n        var baseTopic = ['/user', account, project, groupName, worldid, userid].join('/');\n        return __super.getChannel.call(this, { base: baseTopic });\n    },\n\n    /**\n     * Create and return a publish/subscribe channel (from the underlying [Channel Manager](../channel-manager/)) that automatically tracks the presence of an [end user](../../../glossary/#users), that is, whether the end user is currently online in this group. Notifications are automatically sent when the end user comes online, and when the end user goes offline (not present for more than 2 minutes). Useful in multiplayer games for letting each end user know whether other users in their group are also online.\n     *\n     * Note that the presence channel is tracking all end users in a group. In particular, if the project additionally splits each group into [worlds](../world-manager/), this channel continues to show notifications for all end users in the group (not restricted by worlds).\n     *\n     * **Example**\n     *\n     *     var cm = new F.manager.ChannelManager();\n     *     var pc = cm.getPresenceChannel();\n     *     pc.subscribe('', function (data) {\n     *          // 'data' is the entire message object to the channel;\n     *          // parse for information of interest\n     *          if (data.data.subType === 'disconnect') {\n     *               console.log('user ', data.data.user.userName, 'disconnected at ', data.data.date);\n     *          }\n     *          if (data.data.subType === 'connect') {\n     *               console.log('user ', data.data.user.userName, 'connected at ', data.data.date);\n     *          }\n     *     });\n     *\n     *\n     * **Return Value**\n     *\n     * * *Channel* Returns the channel (an instance of the [Channel Service](../channel-service/)).\n     *\n     * **Parameters**\n     *\n     * @param  {String} groupName (Optional) Group the end user is in. If not provided, picks up group from current session if end user is logged in.\n     * @return {Channel} Channel instance\n     */\n    getPresenceChannel: function (groupName) {\n        var session = this.sessionManager.getMergedOptions(this.options);\n        groupName = getFromSessionOrError(groupName, 'groupName', session);\n        var account = getFromSessionOrError('', 'account', session);\n        var project = getFromSessionOrError('', 'project', session);\n\n        var baseTopic = ['/group', account, project, groupName].join('/');\n        var channel = __super.getChannel.call(this, { base: baseTopic });\n        var oldsubs = channel.subscribe;\n        channel.subscribe = function (topic, callback, context, options) {\n            var callbackWithOnlyPresenceData = function (payload) {\n                if (isPresenceData(payload)) {\n                    callback.call(context, payload);\n                }\n            };\n            return oldsubs.call(channel, topic, callbackWithOnlyPresenceData, context, options);\n        };\n        return channel;\n    },\n\n    /**\n     * Create and return a publish/subscribe channel (from the underlying [Channel Manager](../channel-manager/)) for the given collection. (The collection name is specified in the `root` argument when the [Data Service](../data-api-service/) is instantiated.) Must be one of the collections in this account (team) and project.\n     *\n     * There are automatic notifications from Epicenter on this channel when data is created, updated, or deleted in this collection. See more on [automatic messages to the data channel](../../../rest_apis/multiplayer/channel/#data-messages).\n     *\n     * **Example**\n     *\n     *     var cm = new F.manager.ChannelManager();\n     *     var dc = cm.getDataChannel('survey-responses');\n     *     dc.subscribe('', function(data, meta) {\n     *          console.log(data);\n     *\n     *          // meta.date is time of change,\n     *          // meta.subType is the kind of change: new, update, or delete\n     *          // meta.path is the full path to the changed data\n     *          console.log(meta);\n     *     });\n     *\n     * **Return Value**\n     *\n     * * *Channel* Returns the channel (an instance of the [Channel Service](../channel-service/)).\n     *\n     * **Parameters**\n     *\n     * @param  {String} collection Name of collection whose automatic notifications you want to receive.\n     * @return {Channel} Channel instance\n     */\n    getDataChannel: function (collection) {\n        if (!collection) {\n            throw new Error('Please specify a collection to listen on.');\n        }\n\n        var session = this.sessionManager.getMergedOptions(this.options);\n        var account = getFromSessionOrError('', 'account', session);\n        var project = getFromSessionOrError('', 'project', session);\n        var baseTopic = ['/data', account, project, collection].join('/');\n        var channel = __super.getChannel.call(this, { base: baseTopic });\n\n        //TODO: Fix after Epicenter bug is resolved\n        var oldsubs = channel.subscribe;\n        channel.subscribe = function (topic, callback, context, options) {\n            var callbackWithCleanData = function (payload) {\n                var meta = {\n                    path: payload.channel,\n                    subType: payload.data.subType,\n                    date: payload.data.date,\n                    dataPath: payload.data.data.path,\n                };\n                var actualData = payload.data.data;\n                if (actualData.data !== undefined) { //Delete notifications are one data-level behind of course\n                    actualData = actualData.data;\n                }\n\n                callback.call(context, actualData, meta);\n            };\n            return oldsubs.call(channel, topic, callbackWithCleanData, context, options);\n        };\n\n        return channel;\n    }\n});\n\nmodule.exports = EpicenterChannelManager;\n","'use strict';\n\nmodule.exports = {\n    EPI_SESSION_KEY: 'epicenterjs.session',\n    STRATEGY_SESSION_KEY: 'epicenter-scenario'\n};","/**\n* ## Run Manager\n*\n* The Run Manager gives you access to runs for your project. This allows you to read and update variables, call operations, etc. Additionally, the Run Manager gives you control over run creation depending on run states. Specifically, you can select [run creation strategies (rules)](../strategies/) for which runs end users of your project work with when they log in to your project.\n*\n* There are many ways to create new runs, including the Epicenter.js [Run Service](../run-api-service/) and the RESFTful [Run API](../../../rest_apis/aggregate_run_api). However, for some projects it makes more sense to pick up where the user left off, using an existing run. And in some projects, whether to create a new run or use an existing one is conditional, for example based on characteristics of the existing run or your own knowledge about the model. The Run Manager provides this level of control: your call to `getRun()`, rather than always returning a new run, returns a run based on the strategy you've specified.\n*\n*\n* ### Using the Run Manager to create and access runs\n*\n* To use the Run Manager, instantiate it by passing in:\n*\n*   * `run`: (required) Run object. Must contain:\n*       * `account`: Epicenter account id (**Team ID** for team projects, **User ID** for personal projects).\n*       * `project`: Epicenter project id.\n*       * `model`: The name of your primary model file. (See more on [Writing your Model](../../../writing_your_model/).)\n*       * `scope`: (optional) Scope object for the run, for example `scope.group` with value of the name of the group.\n*       * `server`: (optional) An object with one field, `host`. The value of `host` is the string `api.forio.com`, the URI of the Forio server. This is automatically set, but you can pass it explicitly if desired. It is most commonly used for clarity when you are [hosting an Epicenter project on your own server](../../../how_to/self_hosting/).\n*       * `files`: (optional) If and only if you are using a Vensim model and you have additional data to pass in to your model, you can optionally pass a `files` object with the names of the files, for example: `\"files\": {\"data\": \"myExtraData.xls\"}`. (See more on [Using External Data in Vensim](../../../model_code/vensim/vensim_example_xls/).)\n*\n*   * `strategy`: (optional) Run creation strategy for when to create a new run and when to reuse an end user's existing run. This is *optional*; by default, the Run Manager selects `reuse-per-session`, or `reuse-last-initialized` if you also pass in an initial operation. See [below](#using-the-run-manager-to-access-and-register-strategies) for more information on strategies.\n*\n*   * `strategyOptions`: (optional) Additional options passed directly to the [run creation strategy](../strategies/).\n*\n*   * `sessionKey`: (optional) Name of browser cookie in which to store run information, including run id. Many conditional strategies, including the provided strategies, rely on this browser cookie to store the run id and help make the decision of whether to create a new run or use an existing one. The name of this cookie defaults to `epicenter-scenario` and can be set with the `sessionKey` parameter. This can also be a function which returns a string, if you'd like to control this at runtime.\n*\n*\n* After instantiating a Run Manager, make a call to `getRun()` whenever you need to access a run for this end user. The `RunManager.run` contains the instantiated [Run Service](../run-api-service/). The Run Service allows you to access variables, call operations, etc.\n*\n* **Example**\n*\n*       var rm = new F.manager.RunManager({\n*           run: {\n*               account: 'acme-simulations',\n*               project: 'supply-chain-game',\n*               model: 'supply-chain-model.jl',\n*               server: { host: 'api.forio.com' }\n*           },\n*           strategy: 'reuse-never',\n*           sessionKey: 'epicenter-session'\n*       });\n*       rm.getRun()\n*           .then(function(run) {\n*               // the return value of getRun() is a run object\n*               var thisRunId = run.id;\n*               // the RunManager.run also contains the instantiated Run Service,\n*               // so any Run Service method is valid here\n*               rm.run.do('runModel');\n*       })\n*\n*\n* ### Using the Run Manager to access and register strategies\n*\n* The `strategy` for a Run Manager describes when to create a new run and when to reuse an end user's existing run. The Run Manager is responsible for passing a strategy everything it might need to determine the 'correct' run, that is, how to find the best existing run and how to decide when to create a new run.\n*\n* There are several common strategies provided as part of Epicenter.js, which you can list by accessing `F.manager.RunManager.strategies`. You can also create your own strategies, and register them to use with Run Managers. See [Run Manager Strategies](../strategies/) for details.\n* \n*/\n\n'use strict';\nvar strategies = require('./run-strategies');\nvar specialOperations = require('./special-operations');\n\nvar RunService = require('../service/run-api-service');\nvar SessionManager = require('../store/session-manager');\n\nvar util = require('../util/object-util');\nvar keyNames = require('./key-names');\n\nfunction patchRunService(service, manager) {\n    if (service.patched) {\n        return service;\n    }\n\n    var orig = service.do;\n    service.do = function (operation, params, options) {\n        var reservedOps = Object.keys(specialOperations);\n        if (reservedOps.indexOf(operation) === -1) {\n            return orig.apply(service, arguments);\n        } else {\n            return specialOperations[operation].call(service, params, options, manager);\n        }\n    };\n\n    service.patched = true;\n\n    return service;\n}\n\nfunction sessionKeyFromOptions(options, runService) {\n    var config = runService.getCurrentConfig();\n    var sessionKey = $.isFunction(options.sessionKey) ? options.sessionKey(config) : options.sessionKey;\n    return sessionKey;\n}\n\nfunction setRunInSession(sessionKey, run, sessionManager) {\n    if (sessionKey) {\n        delete run.variables;\n        sessionManager.getStore().set(sessionKey, JSON.stringify(run));\n    }\n}\n\nvar defaults = {\n    sessionKey: function (config) { \n        var baseKey = keyNames.STRATEGY_SESSION_KEY;\n        var key = ['account', 'project', 'model'].reduce(function (accum, key) {\n            return config[key] ? accum + '-' + config[key] : accum; \n        }, baseKey);\n        return key;\n    }\n};\n\nfunction RunManager(options) {\n    this.options = $.extend(true, {}, defaults, options);\n\n    if (this.options.run instanceof RunService) {\n        this.run = this.options.run;\n    } else if (!util.isEmpty(this.options.run)) {\n        this.run = new RunService(this.options.run);\n    } else {\n        throw new Error('No run options passed to RunManager');\n    }\n    patchRunService(this.run, this);\n\n    this.strategy = strategies.getBestStrategy(this.options);\n    this.sessionManager = new SessionManager(this.options);\n}\n\nRunManager.prototype = {\n    /**\n     * Returns the run object for the 'correct' run. The correct run is defined by the strategy. \n     *\n     * For example, if the strategy is `reuse-never`, the call\n     * to `getRun()` always returns a newly created run; if the strategy is `reuse-per-session`,\n     * `getRun()` returns the run currently referenced in the browser cookie, and if there is none, creates a new run. \n     * See [Run Manager Strategies](../strategies/) for more on strategies.\n     *\n     *  **Example**\n     *\n     *      rm.getRun().then(function (run) {\n     *          // use the run object\n     *          var thisRunId = run.id;\n     *\n     *          // use the Run Service object\n     *          rm.run.do('runModel');\n     *      });\n     *\n     *      rm.getRun(['sample_int']).then(function (run) {\n     *         // an object whose fields are the name : value pairs of the variables passed to getRun()\n     *         console.log(run.variables);\n     *         // the value of sample_int\n     *         console.log(run.variables.sample_int); \n     *      });\n     *\n     * @param {Array} variables (Optional) The run object is populated with the provided model variables, if provided. Note: `getRun()` does not throw an error if you try to get a variable which doesn't exist. Instead, the variables list is empty, and any errors are logged to the console.\n     * @param {Object} options (Optional) Configuration options; passed on to [RunService#create](../run-api-service/#create) if the strategy does create a new run.\n     * @return {$promise} Promise to complete the call.\n     */\n    getRun: function (variables, options) {\n        var me = this;\n        var sessionStore = this.sessionManager.getStore();\n\n        var sessionContents = sessionStore.get(sessionKeyFromOptions(this.options, me.run));\n        var runSession = JSON.parse(sessionContents || '{}');\n        \n        if (runSession.runId) {\n            //EpiJS < 2.2 used runId as key, so maintain comptaibility. Remove at some future date (Summer `17?)\n            runSession.id = runSession.runId;\n        }\n\n        var authSession = this.sessionManager.getSession();\n        if (this.strategy.requiresAuth && util.isEmpty(authSession)) {\n            console.error('No user-session available', this.options.strategy, 'requires authentication.');\n            return $.Deferred().reject('No user-session available').promise();\n        }\n        return this.strategy\n                .getRun(this.run, authSession, runSession, options).then(function (run) {\n                    if (run && run.id) {\n                        me.run.updateConfig({ filter: run.id });\n                        var sessionKey = sessionKeyFromOptions(me.options, me.run);\n                        setRunInSession(sessionKey, run, me.sessionManager);\n\n                        if (variables && variables.length) {\n                            return me.run.variables().query(variables).then(function (results) {\n                                run.variables = results;\n                                return run;\n                            }).catch(function (err) {\n                                run.variables = {};\n                                console.error(err);\n                                return run;\n                            });\n                        }\n                    }\n                    return run;\n                });\n    },\n\n    /**\n     * Returns the run object for a 'reset' run. The definition of a reset is defined by the strategy, but typically means forcing the creation of a new run. For example, `reset()` for the default strategies `reuse-per-session` and `reuse-last-initialized` both create new runs.\n     *\n     *  **Example**\n     *\n     *      rm.reset().then(function (run) {\n     *          // use the (new) run object\n     *          var thisRunId = run.id;\n     *\n     *          // use the Run Service object\n     *          rm.run.do('runModel');\n     *      });\n     *\n     * **Parameters**\n     * @param {Object} options (Optional) Configuration options; passed on to [RunService#create](../run-api-service/#create).\n     * @return {Promise}\n     */\n    reset: function (options) {\n        var me = this;\n        var authSession = this.sessionManager.getSession();\n        if (this.strategy.requiresAuth && util.isEmpty(authSession)) {\n            console.error('No user-session available', this.options.strategy, 'requires authentication.');\n            return $.Deferred().reject('No user-session available').promise();\n        }\n        return this.strategy.reset(this.run, authSession, options).then(function (run) {\n            if (run && run.id) {\n                me.run.updateConfig({ filter: run.id });\n                var sessionKey = sessionKeyFromOptions(me.options, me.run);\n                setRunInSession(sessionKey, run.id, me.sessionManager);\n            }\n            return run;\n        });\n    }\n};\n\nRunManager.strategies = strategies;\nmodule.exports = RunManager;\n","'use strict';\n\nvar Base = require('./none-strategy');\nvar classFrom = require('../../util/inherit');\n\n/**\n* ## Conditional Creation Strategy\n*\n* This strategy will try to get the run stored in the cookie and\n* evaluate if it needs to create a new run by calling the `condition` function.\n*/\n\nvar Strategy = classFrom(Base, {\n    constructor: function Strategy(condition) {\n        if (condition == null) { //eslint-disable-line\n            throw new Error('Conditional strategy needs a condition to create a run');\n        }\n        this.condition = typeof condition !== 'function' ? function () { return condition; } : condition;\n    },\n\n    /**\n     * Gets a new 'correct' run, or updates the existing one (the definition of 'correct' depends on strategy implementation).\n     * @param  {RunService} runService A Run Service instance for the current run, as determined by the Run Manager.\n     * @param  {Object} userSession Information about the current user session. See [AuthManager#getCurrentUserSessionInfo](../auth-manager/#getcurrentusersessioninfo) for format.\n     * @param  {Object} options (Optional) See [RunService#create](../run-api-service/#create) for supported options.\n     * @return {Promise}             \n     */\n    reset: function (runService, userSession, options) {\n        var group = userSession && userSession.groupName;\n        var opt = $.extend({\n            scope: { group: group }\n        }, runService.getCurrentConfig());\n\n        return runService\n                .create(opt, options)\n                .then(function (run) {\n                    run.freshlyCreated = true;\n                    return run;\n                });\n    },\n\n    /**\n     * Gets the 'correct' run (the definition of 'correct' depends on strategy implementation).\n     * @param  {RunService} runService A Run Service instance for the current run, as determined by the Run Manager.\n     * @param  {Object} userSession Information about the current user session. See [AuthManager#getCurrentUserSessionInfo](../auth-manager/#getcurrentusersessioninfo) for format.\n     * @param  {Object} runSession The Run Manager stores the 'last accessed' run in a cookie and passes it back here.\n     * @param  {Object} options (Optional) See [RunService#create](../run-api-service/#create) for supported options.\n     * @return {Promise}             \n     */\n    getRun: function (runService, userSession, runSession, options) {\n        var me = this;\n        if (runSession && runSession.id) {\n            return this.loadAndCheck(runService, userSession, runSession, options).catch(function () {\n                return me.reset(runService, userSession, options); //if it got the wrong cookie for e.g.\n            });\n        } else {\n            return this.reset(runService, userSession, options);\n        }\n    },\n\n    loadAndCheck: function (runService, userSession, runSession, options) {\n        var shouldCreate = false;\n        var me = this;\n\n        return runService\n            .load(runSession.id, null, {\n                success: function (run, msg, headers) {\n                    shouldCreate = me.condition(run, headers, userSession, runSession);\n                }\n            })\n            .then(function (run) {\n                if (shouldCreate) {\n                    return me.reset(runService, userSession, options);\n                }\n                return run;\n            });\n    }\n});\n\nmodule.exports = Strategy;\n","/**\n * The `new-if-initialized` strategy creates a new run if the current one is in memory or has its `initialized` field set to `true`. The `initialized` field in the run record is automatically set to `true` at run creation, but can be changed.\n * \n * This strategy is useful if your project is structured such that immediately after a run is created, the model is executed completely (for example, a Vensim model is stepped to the end). It is similar to the `new-if-missing` strategy, except that it checks a field of the run record.\n * \n * Specifically, the strategy is:\n *\n * * Check the `sessionKey` cookie. \n *  * This cookie is set by the [Run Manager](../run-manager/) and configurable through its options.\n *  * If the cookie exists, check whether the run is in memory or only persisted in the database. Additionally, check whether the run's `initialized` field is `true`. \n *      * If the run is in memory, create a new run.\n *      * If the run's `initialized` field is `true`, create a new run.\n *      * Otherwise, use the existing run.\n *  * If the cookie does not exist, create a new run for this end user.\n *  \n *  @deprecated Consider using `reuse-last-initialized` instead\n */\n\n'use strict';\nvar classFrom = require('../../../util/inherit');\nvar ConditionalStrategy = require('../conditional-creation-strategy');\n\nvar __super = ConditionalStrategy.prototype;\n\nvar Strategy = classFrom(ConditionalStrategy, {\n    constructor: function (options) {\n        __super.constructor.call(this, this.createIf, options);\n        console.warn('This strategy is deprecated; all runs now default to being initialized by default making this redundant. Consider using `reuse-last-initialized` instead.');\n    },\n\n    createIf: function (run, headers) {\n        return headers.getResponseHeader('pragma') === 'persistent' || run.initialized;\n    }\n});\n\nmodule.exports = Strategy;\n","/**\n * The `new-if-persisted` strategy creates a new run when the current one becomes persisted (end user is idle for a set period), but otherwise uses the current one. \n * \n * Using this strategy means that when end users navigate between pages in your project, or refresh their browsers, they will still be working with the same run. \n * \n * However, if they are idle for longer than your project's **Model Session Timeout** (configured in your project's [Settings](../../../updating_your_settings/)), then their run is persisted; the next time they interact with the project, they will get a new run. (See more background on [Run Persistence](../../../run_persistence/).)\n * \n * This strategy is useful for multi-page projects where end users play through a simulation in one sitting, stepping through the model sequentially (for example, a Vensim model that uses the `step` operation) or calling specific functions until the model is \"complete.\" However, you will need to guarantee that your end users will remain engaged with the project from beginning to end &mdash; or at least, that if they are idle for longer than the **Model Session Timeout**, it is okay for them to start the project from scratch (with an uninitialized model). \n * \n * Specifically, the strategy is:\n *\n * * Check the `sessionKey` cookie.\n *   * This cookie is set by the [Run Manager](../run-manager/) and configurable through its options.\n *   * If the cookie exists, check whether the run is in memory or only persisted in the database. \n *      * If the run is in memory, use the run.\n *      * If the run is only persisted (and not still in memory), create a new run for this end user.\n *      * If the cookie does not exist, create a new run for this end user.\n *\n * @deprecated The run-service now sets a header to automatically bring back runs into memory\n */\n\n'use strict';\nvar classFrom = require('../../../util/inherit');\nvar ConditionalStrategy = require('../conditional-creation-strategy');\n\nvar __super = ConditionalStrategy.prototype;\n\nvar Strategy = classFrom(ConditionalStrategy, {\n    constructor: function (options) {\n        __super.constructor.call(this, this.createIf, options);\n        console.warn('This strategy is deprecated; the run-service now sets a header to automatically bring back runs into memory');\n    },\n\n    createIf: function (run, headers) {\n        return headers.getResponseHeader('pragma') === 'persistent';\n    }\n});\n\nmodule.exports = Strategy;\n","/**\n * ### Working with Run Strategies\n *\n * You can access a list of available strategies using `F.manager.RunManager.strategies.list`. You can also ask for a particular strategy by name.\n *\n * If you decide to [create your own run strategy](#create-your-own), you can register your strategy. Registering your strategy means that:\n *\n * * You can pass the strategy by name to a Run Manager (as opposed to passing the strategy function): `new F.manager.RunManager({ strategy: 'mynewname'})`.\n * * You can pass configuration options to your strategy.\n * * You can specify whether or not your strategy requires authorization (a valid user session) to work.\n */\n\n\nvar list = {\n    'conditional-creation': require('./conditional-creation-strategy'),\n    'new-if-initialized': require('./deprecated/new-if-initialized-strategy'), //deprecated\n    'new-if-persisted': require('./deprecated/new-if-persisted-strategy'), //deprecated\n\n    none: require('./none-strategy'),\n\n    multiplayer: require('./multiplayer-strategy'),\n    'reuse-never': require('./reuse-never'),\n    'reuse-per-session': require('./reuse-per-session'),\n    'reuse-across-sessions': require('./reuse-across-sessions'),\n    'reuse-last-initialized': require('./reuse-last-initialized'),\n};\n\n//Add back older aliases\nlist['always-new'] = list['reuse-never'];\nlist['new-if-missing'] = list['reuse-per-session'];\nlist['persistent-single-player'] = list['reuse-across-sessions'];\n\n\nmodule.exports = {\n    /**\n     * List of available strategies. Within this object, each key is the strategy name and the associated value is the strategy constructor.\n     * @type {Object} \n     */\n    list: list,\n\n    /**\n     * Gets strategy by name.\n     *\n     * **Example**\n     *\n     *      var reuseStrat = F.manager.RunManager.strategies.byName('reuse-across-sessions');\n     *      // shows strategy function\n     *      console.log('reuseStrat = ', reuseStrat);\n     *      // create a new run manager using this strategy\n     *      var rm = new F.manager.RunManager({strategy: reuseStrat, run: { model: 'model.vmf'} });\n     *\n     * **Parameters**\n     * @param  {String} strategyName Name of strategy to get.\n     * @return {Function} Strategy function.\n     */\n    byName: function (strategyName) {\n        return list[strategyName];\n    },\n\n    getBestStrategy: function (options) {\n        var strategy = options.strategy;\n        if (!strategy) {\n            if (options.strategyOptions && options.strategyOptions.initOperation) {\n                strategy = 'reuse-last-initialized';\n            } else {\n                strategy = 'reuse-per-session';\n            }\n        }\n\n        if (strategy.getRun) {\n            return strategy;\n        }\n        var StrategyCtor = typeof strategy === 'function' ? strategy : this.byName(strategy);\n        if (!StrategyCtor) {\n            throw new Error('Specified run creation strategy was invalid:', strategy);\n        }\n\n        var strategyInstance = new StrategyCtor(options);\n        if (!strategyInstance.getRun || !strategyInstance.reset) {\n            throw new Error('All strategies should implement a `getRun` and `reset` interface', options.strategy);\n        }\n        strategyInstance.requiresAuth = StrategyCtor.requiresAuth;\n\n        return strategyInstance;\n    },\n\n    /**\n     * Adds a new strategy.\n     *\n     * **Example**\n     *\n     *      // this \"favorite run\" strategy always returns the same run, no matter what\n     *      // (not a useful strategy, except as an example)\n     *      F.manager.RunManager.strategies.register(\n     *          'favRun', \n     *          function() { \n     *              return { getRun: function() { return '0000015a4cd1700209cd0a7d207f44bac289'; },\n     *                      reset: function() { return '0000015a4cd1700209cd0a7d207f44bac289'; } \n     *              } \n     *          }, \n     *          { requiresAuth: true }\n     *      );\n     *      \n     *      var rm = new F.manager.RunManager({strategy: 'favRun', run: { model: 'model.vmf'} });\n     *\n     * **Parameters**\n     * @param  {String} name Name for strategy. This string can then be passed to a Run Manager as `new F.manager.RunManager({ strategy: 'mynewname'})`.\n     * @param  {Function} strategy The strategy constructor. Will be called with `new` on Run Manager initialization.\n     * @param  {Object} options  Options for strategy.\n     * @param  {Boolean} options.requiresAuth Specify if the strategy requires a valid user session to work.\n     */\n    register: function (name, strategy, options) {\n        strategy.options = options;\n        list[name] = strategy;\n    }\n};","/**\n * The `multiplayer` strategy is for use with [multiplayer worlds](../../../glossary/#world). It checks the current world for this end user, and always returns the current run for that world. This is equivalent to calling `getCurrentWorldForUser()` and then `getCurrentRunId()` from the [World API Adapater](../world-api-adapter/). If you use the [World Manager](../world-manager/), you are automatically using this strategy.\n * \n * Using this strategy means that end users in projects with multiplayer worlds always see the most current world and run. This ensures that they are in sync with the other end users sharing their world and run. In turn, this allows for competitive or collaborative multiplayer projects.\n */\n'use strict';\n\nvar classFrom = require('../../util/inherit');\n\nvar IdentityStrategy = require('./none-strategy');\nvar WorldApiAdapter = require('../../service/world-api-adapter');\n\nvar defaults = {};\n\nvar Strategy = classFrom(IdentityStrategy, {\n    constructor: function (options) {\n        this.options = $.extend(true, {}, defaults, options);\n        this.worldApi = new WorldApiAdapter(this.options.run);\n    },\n\n    reset: function (runService, session, options) {\n        var curUserId = session.userId;\n        var curGroupName = session.groupName;\n\n        return this.worldApi\n            .getCurrentWorldForUser(curUserId, curGroupName)\n            .then(function (world) {\n                return this.worldApi.newRunForWorld(world.id, options).then(function (runid) {\n                    return {\n                        id: runid\n                    };\n                });\n            }.bind(this));\n    },\n\n    getRun: function (runService, session) {\n        var curUserId = session.userId;\n        var curGroupName = session.groupName;\n        var worldApi = this.worldApi;\n        var model = this.options.model;\n        var me = this;\n        var dtd = $.Deferred();\n\n        if (!curUserId) {\n            return dtd.reject({ statusCode: 400, error: 'We need an authenticated user to join a multiplayer world. (ERR: no userId in session)' }, session).promise();\n        }\n\n        var loadRunFromWorld = function (world) {\n            if (!world) {\n                return dtd.reject({ statusCode: 404, error: 'The user is not in any world.' }, { options: me.options, session: session });\n            }\n            return worldApi.getCurrentRunId({ model: model, filter: world.id })\n                .then(function (id) {\n                    return runService.load(id);\n                })\n                .then(dtd.resolve)\n                .fail(dtd.reject);\n        };\n\n        var serverError = function (error) {\n            // is this possible?\n            return dtd.reject(error, session, me.options);\n        };\n\n        this.worldApi\n            .getCurrentWorldForUser(curUserId, curGroupName)\n            .then(loadRunFromWorld)\n            .fail(serverError);\n\n        return dtd.promise();\n    },\n\n});\n\nmodule.exports = Strategy;\n","/**\n * The `none` strategy never returns a run or tries to create a new run. It simply returns the contents of the current [Run Service instance](../run-api-service/).\n * \n * This strategy is useful if you want to manually decide how to create your own runs and don't want any automatic assistance.\n */\n\n'use strict';\n\nvar classFrom = require('../../util/inherit');\nvar Base = {};\n\n// Interface that all strategies need to implement\nmodule.exports = classFrom(Base, {\n    constructor: function (options) {\n\n    },\n\n    reset: function () {\n        // return a newly created run\n        return $.Deferred().resolve().promise();\n    },\n\n    getRun: function (runService) {\n        // return a usable run\n        return $.Deferred().resolve(runService).promise();\n    }\n});\n","/**\n * The `reuse-across-sessions` strategy returns the latest (most recent) run for this user, whether it is in memory or not. If there are no runs for this user, it creates a new one.\n *\n * This strategy is useful if end users are using your project for an extended period of time, possibly over several sessions. This is most common in cases where a user of your project executes the model step by step (as opposed to a project where the model is executed completely, for example, a Vensim model that is immediately stepped to the end).\n *\n * Specifically, the strategy is:\n * \n * * Check if there are any runs for this end user.\n *     * If there are no runs (either in memory or in the database), create a new one.\n *     * If there are runs, take the latest (most recent) one.\n *\n * @name persistent-single-player\n */\n\n'use strict';\n\nvar classFrom = require('../../util/inherit');\nvar IdentityStrategy = require('./none-strategy');\nvar injectFiltersFromSession = require('../strategy-utils').injectFiltersFromSession;\nvar injectScopeFromSession = require('../strategy-utils').injectScopeFromSession;\n\nvar defaults = {\n    /**\n     * (Optional) Additional criteria to use while selecting the last run\n     * @type {Object}\n     */\n    filter: {},\n};\n\nvar Strategy = classFrom(IdentityStrategy, {\n    constructor: function Strategy(options) {\n        var strategyOptions = options ? options.strategyOptions : {};\n        this.options = $.extend(true, {}, defaults, strategyOptions);\n    },\n\n    reset: function (runService, userSession, options) {\n        var opt = injectScopeFromSession(runService.getCurrentConfig(), userSession);\n        return runService\n            .create(opt, options)\n            .then(function (run) {\n                run.freshlyCreated = true;\n                return run;\n            });\n    },\n\n    getRun: function (runService, userSession, runSession, options) {\n        var filter = injectFiltersFromSession(this.options.filter, userSession);\n        var me = this;\n        return runService.query(filter, { \n            startrecord: 0,\n            endrecord: 0,\n            sort: 'created', \n            direction: 'desc'\n        }).then(function (runs) {\n            if (!runs.length) {\n                return me.reset(runService, userSession, options);\n            }\n            return runs[0];\n        });\n    }\n});\n\nmodule.exports = Strategy;\n","/**\n * The `reuse-last-initialized` strategy looks for the most recent run that matches particular criteria; if it cannot find one, it creates a new run and immediately executes a set of \"initialization\" operations. \n *\n * This strategy is useful if you have a time-based model and always want the run you're operating on to start at a particular step. For example:\n *\n *      var rm = new F.manager.RunManager({\n *          strategy: 'reuse-last-initialized',\n *          strategyOptions: {\n *              initOperation: [{ step: 10 }]\n *          }\n *      });\n * \n * This strategy is also useful if you have a custom initialization function in your model, and want to make sure it's always executed for new runs.\n *\n * Specifically, the strategy is:\n *\n * * Look for the most recent run that matches the (optional) `flag` criteria\n * * If there are no runs that match the `flag` criteria, create a new run. Immediately \"initialize\" this new run by:\n *     *  Calling the model operation(s) specified in the `initOperation` array.\n *     *  Optionally, setting a `flag` in the run.\n *\n */\n\n'use strict';\nvar classFrom = require('../../util/inherit');\nvar injectFiltersFromSession = require('../strategy-utils').injectFiltersFromSession;\nvar injectScopeFromSession = require('../strategy-utils').injectScopeFromSession;\n\nvar Base = {};\n\nvar defaults = {\n    /**\n     * Operations to execute in the model for initialization to be considered complete.\n     * @type {Array} Can be in any of the formats [Run Service's `serial()`](../run-api-service/#serial) supports.\n     */\n    initOperation: [],\n\n    /**\n     * (Optional) Flag to set in run after initialization operations are run. You typically would not override this unless you needed to set additional properties as well.\n     * @type {Object}\n     */\n    flag: null,\n};\nmodule.exports = classFrom(Base, {\n    constructor: function (options) {\n        var strategyOptions = options ? options.strategyOptions : {};\n        this.options = $.extend(true, {}, defaults, strategyOptions);\n        if (!this.options.initOperation || !this.options.initOperation.length) {\n            throw new Error('Specifying an init function is required for this strategy');\n        }\n        if (!this.options.flag) {\n            this.options.flag = {\n                isInitComplete: true\n            };\n        }\n    },\n\n    reset: function (runService, userSession, options) {\n        var opt = injectScopeFromSession(runService.getCurrentConfig(), userSession);\n        var me = this;\n        return runService.create(opt, options).then(function (createResponse) {\n            return runService.serial([].concat(me.options.initOperation)).then(function () {\n                return createResponse;\n            });\n        }).then(function (createResponse) {\n            return runService.save(me.options.flag).then(function (patchResponse) {\n                return $.extend(true, {}, createResponse, patchResponse);\n            });\n        });\n    },\n\n    getRun: function (runService, userSession, runSession, options) {\n        var sessionFilter = injectFiltersFromSession(this.options.flag, userSession);\n        var runopts = runService.getCurrentConfig();\n        var filter = $.extend(true, { trashed: false }, sessionFilter, { model: runopts.model });\n        var me = this;\n        return runService.query(filter, { \n            startrecord: 0,\n            endrecord: 0,\n            sort: 'created', \n            direction: 'desc'\n        }).then(function (runs) {\n            if (!runs.length) {\n                return me.reset(runService, userSession, options);\n            }\n            return runs[0];\n        });\n    }\n});","/**\n * The `reuse-never` strategy always creates a new run for this end user irrespective of current state. This is equivalent to calling `F.service.Run.create()` from the [Run Service](../run-api-service/) every time. \n * \n * This strategy means that every time your end users refresh their browsers, they get a new run. \n * \n * This strategy can be useful for basic, single-page projects. This strategy is also useful for prototyping or project development: it creates a new run each time you refresh the page, and you can easily check the outputs of the model. However, typically you will use one of the other strategies for a production project.\n *\n * @name always-new\n */\n\n'use strict';\n\nvar classFrom = require('../../util/inherit');\nvar ConditionalStrategy = require('./conditional-creation-strategy');\n\nvar __super = ConditionalStrategy.prototype;\n\nvar Strategy = classFrom(ConditionalStrategy, {\n    constructor: function (options) {\n        __super.constructor.call(this, this.createIf, options);\n    },\n\n    createIf: function (run, headers) {\n        // always create a new run!\n        return true;\n    }\n});\n\nmodule.exports = Strategy;\n","/**\n * The `reuse-per-session` strategy creates a new run when the current one is not in the browser cookie.\n * \n * Using this strategy means that when end users navigate between pages in your project, or refresh their browsers, they will still be working with the same run. However, if end users log out and return to the project at a later date, a new run is created.\n *\n * This strategy is useful if your project is structured such that immediately after a run is created, the model is executed completely (for example, a Vensim model that is stepped to the end as soon as it is created). In contrast, if end users play with your project for an extended period of time, executing the model step by step, the `reuse-across-sessions` strategy is probably a better choice (it allows end users to pick up where they left off, rather than starting from scratch each browser session).\n * \n * Specifically, the strategy is:\n *\n * * Check the `sessionKey` cookie.\n *     * This cookie is set by the [Run Manager](../run-manager/) and configurable through its options. \n *     * If the cookie exists, use the run id stored there. \n *     * If the cookie does not exist, create a new run for this end user.\n *\n * @name new-if-missing\n */\n\n'use strict';\n\nvar classFrom = require('../../util/inherit');\nvar ConditionalStrategy = require('./conditional-creation-strategy');\n\nvar __super = ConditionalStrategy.prototype;\n\n/*\n*  create a new run only if nothing is stored in the cookie\n*  this is useful for baseRuns.\n*/\nvar Strategy = classFrom(ConditionalStrategy, {\n    constructor: function (options) {\n        __super.constructor.call(this, this.createIf, options);\n    },\n\n    createIf: function (run, headers) {\n        // if we are here, it means that the run exists... so we don't need a new one\n        return false;\n    }\n});\n\nmodule.exports = Strategy;\n","/**\n * ## Saved Runs Manager\n *\n * The Saved Runs Manager is a specific type of [Run Manager](../../run-manager/) which provides access to a list of runs (rather than just one run). It also provides utility functions for dealing with multiple runs (e.g. saving, deleting, listing).\n *\n * An instance of a Saved Runs Manager is included automatically in every instance of a [Scenario Manager](../), and is accessible from the Scenario Manager at `.savedRuns`. See [more information](../#properties) on using `.savedRuns` within the Scenario Manager.\n *\n */\n'use strict';\n\nvar RunService = require('../service/run-api-service');\nvar SessionManager = require('../store/session-manager');\n\nvar injectFiltersFromSession = require('./strategy-utils').injectFiltersFromSession;\n\nvar SavedRunsManager = function (config) {\n    var defaults = {\n        /**\n         * If set, will only pull runs from current group. Defaults to `true`.\n         * @type {Boolean}\n         */\n        scopeByGroup: true,\n\n        /**\n         * If set, will only pull runs from current user. Defaults to `true`.\n         *\n         * For multiplayer run comparison projects, set this to false so that all end users in a group can view the shared set of saved runs.\n         * @type {Boolean}\n         */\n        scopeByUser: true,\n    };\n\n    this.sessionManager = new SessionManager();\n\n    var options = $.extend(true, {}, defaults, config);\n    if (options.run) {\n        if (options.run instanceof RunService) {\n            this.runService = options.run;\n        } else {\n            this.runService = new RunService(options.run);\n        }\n        this.options = options;\n    } else {\n        throw new Error('No run options passed to SavedRunsManager');\n    }\n};\n\nSavedRunsManager.prototype = {\n    /**\n     * Marks a run as saved. \n     *\n     * Note that while any run can be saved, only runs which also match the configuration options `scopeByGroup` and `scopeByUser` are returned by the `getRuns()` method.\n     *\n     * **Example**\n     *\n     *      var sm = new F.manager.ScenarioManager();\n     *      sm.savedRuns.save('0000015a4cd1700209cd0a7d207f44bac289');\n     *\n     * @param  {String|RunService} run Run to save. Pass in either the run id, as a string, or the [Run Service](../../run-api-service/).\n     * @param  {Object} otherFields (Optional) Any other meta-data to save with the run.\n     * @return {Promise}\n     */\n    save: function (run, otherFields) {\n        var param = $.extend(true, {}, otherFields, { saved: true, trashed: false });\n        return this.mark(run, param);\n    },\n    /**\n     * Marks a run as removed; the inverse of marking as saved.\n     *\n     * **Example**\n     *\n     *      var sm = new F.manager.ScenarioManager();\n     *      sm.savedRuns.remove('0000015a4cd1700209cd0a7d207f44bac289');\n     *\n     * @param  {String|RunService} run Run to remove. Pass in either the run id, as a string, or the [Run Service](../../run-api-service/).\n     * @param  {Object} otherFields (Optional) any other meta-data to save with the run.\n     * @return {Promise}\n     */\n    remove: function (run, otherFields) {\n        var param = $.extend(true, {}, otherFields, { saved: false, trashed: true });\n        return this.mark(run, param);\n    },\n\n\n    /**\n     * Sets additional fields on a run. This is a convenience method for [RunService#save](../../run-api-service/#save).\n     *\n     * **Example**\n     *\n     *      var sm = new F.manager.ScenarioManager();\n     *      sm.savedRuns.mark('0000015a4cd1700209cd0a7d207f44bac289', \n     *          { 'myRunName': 'sample policy decisions' });\n     *\n     * @param  {String|RunService} run  Run to operate on. Pass in either the run id, as a string, or the [Run Service](../../run-api-service/).\n     * @param  {Object} toMark Fields to set, as name : value pairs.\n     * @return {Promise}\n     */\n    mark: function (run, toMark) {\n        var rs;\n        var existingOptions = this.runService.getCurrentConfig();\n        if (run instanceof RunService) {\n            rs = run;\n        } else if (run && (typeof run === 'string')) {\n            rs = new RunService($.extend(true, {}, existingOptions, { id: run, autoRestore: false }));\n        } else if ($.isArray(run)) {\n            var me = this;\n            var proms = run.map(function (r) {\n                return me.mark(r, toMark);\n            });\n            return $.when.apply(null, proms);\n        } else {\n            throw new Error('Invalid run object provided');\n        }\n        return rs.save(toMark);\n    },\n\n    /**\n     * Returns a list of saved runs.\n     *\n     * **Example**\n     *\n     *      var sm = new F.manager.ScenarioManager();\n     *      sm.savedRuns.getRuns().then(function (runs) {\n     *          for (var i=0; i<runs.length; i++) {\n     *              console.log('run id of saved run: ', runs[i].id);\n     *          }\n     *      });\n     *\n     * @param  {Array} variables (Optional) If provided, in the returned list of runs, each run will have a `.variables` property with these set.\n     * @param  {Object} filter    (Optional) Any filters to apply while fetching the run. See [RunService#filter](../../run-api-service/#filter) for details.\n     * @param  {Object} modifiers (Optional) Use for paging/sorting etc. See [RunService#filter](../../run-api-service/#filter) for details.\n     * @return {Promise}\n     */\n    getRuns: function (variables, filter, modifiers) {\n        var session = this.sessionManager.getSession(this.runService.getCurrentConfig());\n\n        var runopts = this.runService.getCurrentConfig();\n        var scopedFilter = injectFiltersFromSession($.extend(true, {}, {\n            saved: true, \n            trashed: false,\n            model: runopts.model,\n        }, filter), session, this.options);\n\n        var opModifiers = $.extend(true, {}, {\n            sort: 'created',\n            direction: 'asc',\n        }, modifiers);\n        if (variables) {\n            opModifiers.include = [].concat(variables);\n        }\n        return this.runService.query(scopedFilter, opModifiers);\n    }\n};\nmodule.exports = SavedRunsManager;\n","/**\n* ## Scenario Manager\n*\n* In some projects, often called \"turn-by-turn\" projects, end users advance through the project's model step-by-step, working either individually or together to make decisions at each step. \n*\n* In other projects, often called \"run comparison\" or \"scenario comparison\" projects, end users set some initial decisions, then simulate the model to its end. Typically end users will do this several times, creating several runs, and compare the results. \n*\n* The Scenario Manager makes it easy to create these \"run comparison\" projects. Each Scenario Manager allows you to compare the results of several runs. This is mostly useful for time-based models; by default, you can use the Scenario Manager with [Vensim](../../../model_code/vensim/), [Powersim](../../../model_code/powersim/), and [SimLang](../../../model_code/forio_simlang). (You can use the Scenario Manager with other languages as well, by using the Scenario Manager's [configuration options](#configuration-options) to change the `advanceOperation`.)\n*\n* The Scenario Manager can be thought of as a collection of [Run Managers](../run-manager/) with pre-configured [strategies](../strategies/). Just as the Run Manager provides use case -based abstractions and utilities for managing the [Run Service](../run-api-service/), the Scenario Manager does the same for the Run Manager.\n*\n* There are typically three components to building a run comparison:\n*\n* * A `current` run in which to make decisions; this is defined as a run that hasn't been advanced yet, and so can be used to set initial decisions. The current run maintains state across different sessions.\n* * A list of `saved` runs, that is, all runs that you want to use for comparisons.\n* * A `baseline` run to compare against; this is defined as a run \"advanced to the end\" of your model using just the model defaults. Comparing against a baseline run is optional; you can [configure](#configuration-options) the Scenario Manager to not include one.\n*\n* To satisfy these needs a Scenario Manager instance has three Run Managers: [baseline](./baseline/), [current](./current/), and [savedRuns](./saved/).\n*\n* ### Using the Scenario Manager to create a run comparison project\n*\n* To use the Scenario Manager, instantiate it, then access its Run Managers as needed to create your project's user interface:\n*\n* **Example**\n*\n*       var sm = new F.manager.ScenarioManager({\n*           run: {\n*               model: 'mymodel.vmf'\n*           }\n*       });\n*\n*       // The current is an instance of a Run Manager,\n*       // with a strategy which picks up the most recent unsaved run.\n*       // It is typically used to store the decisions being made by the end user. \n*       var currentRM = sm.current;\n*\n*       // The Run Manager operation, which retrieves the current run.\n*       sm.current.getRun();\n*       // The Run Manager operation, which resets the decisions made on the current run.\n*       sm.current.reset();\n*       // A special method on the current run,\n*       // which clones the current run, then advances and saves this clone\n*       // (it becomes part of the saved runs list).\n*       // The current run is unchanged and can continue to be used\n*       // to store decisions being made by the end user.\n*       sm.current.saveAndAdvance();\n*\n*       // The savedRuns is an instance of a Saved Runs Manager \n*       // (itself a variant of a Run Manager).\n*       // It is typically displayed in the project's UI as part of a run comparison table or chart.\n*       var savedRM = sm.savedRuns;\n*       // Mark a run as saved, adding it to the set of saved runs.\n*       sm.savedRuns.save(run);\n*       // Mark a run as removed, removing it from the set of saved runs.\n*       sm.savedRuns.remove(run);\n*       // List the saved runs, optionally including some specific model variables for each.\n*       sm.savedRuns.getRuns();\n*\n*       // The baseline is an instance of a Run Manager,\n*       // with a strategy which locates the most recent baseline run\n*       // (that is, flagged as `saved` and not `trashed`), or creates a new one.\n*       // It is typically displayed in the project's UI as part of a run comparison table or chart.\n*       var baselineRM = sm.baseline;\n*\n*       // The Run Manager operation, which retrieves the baseline run.\n*       sm.baseline.getRun();\n*       // The Run Manager operation, which resets the baseline run.\n*       // Useful if the model has changed since the baseline run was created.\n*       sm.baseline.reset(); \n*/\n\n'use strict';\n\n// See integration-test-scenario-manager for usage examples\nvar RunManager = require('./run-manager');\nvar SavedRunsManager = require('./saved-runs-manager');\nvar strategyUtils = require('./strategy-utils');\nvar rutil = require('../util/run-util');\n\nvar NoneStrategy = require('./run-strategies/none-strategy');\n\nvar StateService = require('../service/state-api-adapter');\nvar RunService = require('../service/run-api-service');\n\nvar BaselineStrategy = require('./scenario-strategies/baseline-strategy');\nvar LastUnsavedStrategy = require('./scenario-strategies/reuse-last-unsaved');\n\nvar defaults = {\n    /**\n     * Operations to perform on each run to indicate that the run is complete. Operations are executed [serially](../run-api-service/#serial). Defaults to calling the model operation `stepTo('end')`, which advances Vensim, Powersim, and SimLang models to the end. \n     * @type {Array}\n     */\n    advanceOperation: [{ name: 'stepTo', params: ['end'] }],\n\n    /**\n     * Additional options to pass through to run creation (for e.g., `files`, etc.). Defaults to empty object.\n     * @type {Object}\n     */\n    run: {},\n\n    /**\n     * Whether or not to include a baseline run in this Scenario Manager. Defaults to `true`.\n     * @type {Boolean}\n     */\n    includeBaseLine: true,\n\n    /**\n     * Additional configuration for the `baseline` run. \n     *\n     * * `baseline.runName`: Name of the baseline run. Defaults to 'Baseline'. \n     * * `baseline.run`: Additional options to pass through to run creation, specifically for the baseline run. These will override any options provided under `run`. Defaults to empty object. \n     * @type {Object}\n     */\n    baseline: {\n        runName: 'Baseline',\n        run: {}\n    },\n\n    /**\n     * Additional configuration for the `current` run. \n     *\n     * * `current.run`: Additional options to pass through to run creation, specifically for the current run. These will override any options provided under `run`. Defaults to empty object.\n     * @type {Object}\n     */\n    current: {\n        run: {}\n    },\n\n    /**\n     * Options to pass through to the `savedRuns` list. See the [Saved Runs Manager](./saved/) for complete description of available options. Defaults to empty object.\n     * @type {Object}\n     */\n    savedRuns: {}\n};\n\nfunction cookieNameFromOptions(prefix, config) {\n    var key = ['account', 'project', 'model'].reduce(function (accum, key) {\n        return config[key] ? accum + '-' + config[key] : accum; \n    }, prefix);\n    return key;\n}\n\nfunction ScenarioManager(config) {\n    var opts = $.extend(true, {}, defaults, config);\n    if (config && config.advanceOperation) {\n        opts.advanceOperation = config.advanceOperation; //jquery.extend does a poor job trying to merge arrays\n    }\n\n    var BaselineStrategyToUse = opts.includeBaseLine ? BaselineStrategy : NoneStrategy;\n    /**\n     * A [Run Manager](../run-manager/) instance containing a 'baseline' run to compare against; this is defined as a run \"advanced to the end\" of your model using just the model defaults. By default the \"advance\" operation is assumed to be `stepTo: end`, which works for time-based models in [Vensim](../../../model_code/vensim/), [Powersim](../../../model_code/powersim/), and [SimLang](../../../model_code/forio_simlang). If you're using a different language, or need to change this, just pass in a different `advanceOperation` option while creating the Scenario Manager. The baseline run is typically displayed in the project's UI as part of a run comparison table or chart.\n     * @return {RunManager}\n     */\n    this.baseline = new RunManager({\n        strategy: BaselineStrategyToUse,\n        sessionKey: cookieNameFromOptions.bind(null, 'sm-baseline-run'),\n        run: strategyUtils.mergeRunOptions(opts.run, opts.baseline.run),\n        strategyOptions: {\n            baselineName: opts.baseline.runName,\n            initOperation: opts.advanceOperation\n        }\n    });\n\n    /**\n     * A [SavedRunsManager](../saved-runs-manager/) instance containing a list of saved runs, that is, all runs that you want to use for comparisons. The saved runs are typically displayed in the project's UI as part of a run comparison table or chart.\n     * @return {SavedRunsManager}\n     */\n    this.savedRuns = new SavedRunsManager($.extend(true, {}, {\n        run: opts.run,\n    }, opts.savedRuns));\n\n    var origGetRuns = this.savedRuns.getRuns;\n    var me = this;\n    this.savedRuns.getRuns = function () {\n        var args = Array.apply(null, arguments);\n        return me.baseline.getRun().then(function () {\n            return origGetRuns.apply(me.savedRuns, args);\n        });\n    };\n\n    /**\n     * A [Run Manager](../run-manager/) instance containing a 'current' run; this is defined as a run that hasn't been advanced yet, and so can be used to set initial decisions. The current run is typically used to store the decisions being made by the end user.\n     * @return {RunManager}\n     */\n    this.current = new RunManager({\n        strategy: LastUnsavedStrategy,\n        sessionKey: cookieNameFromOptions.bind(null, 'sm-current-run'),\n        run: strategyUtils.mergeRunOptions(opts.run, opts.current.run)\n    });\n\n    /**\n     * Clones the current run, advances this clone by calling the `advanceOperation`, and saves the cloned run (it becomes part of the `savedRuns` list). Additionally, adds any provided metadata to the cloned run; typically used for naming the run. The current run is unchanged and can continue to be used to store decisions being made by the end user.\n     *\n     * Available only for the Scenario Manager's `current` property (Run Manager). \n     *\n     * **Example**\n     *\n     *      var sm = new F.manager.ScenarioManager();\n     *      sm.current.saveAndAdvance({'myRunName': 'sample policy decisions'});\n     *\n     * **Parameters**\n     * @param  {Object} metadata   Metadata to save, for example, the run name.\n     * @return {Promise}\n     */\n    this.current.saveAndAdvance = function (metadata) {\n        function clone(run) {\n            var sa = new StateService();\n            var advanceOpns = rutil.normalizeOperations(opts.advanceOperation); \n            //run i'm cloning shouldn't have the advance operations there by default, but just in case\n            return sa.clone({ runId: run.id, exclude: advanceOpns.ops }).then(function (response) {\n                var rs = new RunService(me.current.run.getCurrentConfig());\n                return rs.load(response.run);\n            });\n        }\n        function markSaved(run) {\n            return me.savedRuns.save(run.id, metadata).then(function (savedResponse) {\n                return $.extend(true, {}, run, savedResponse);\n            });\n        }\n        function advance(run) {\n            var rs = new RunService(run);\n            return rs.serial(opts.advanceOperation).then(function () {\n                return run;\n            });\n        }\n        return me.current\n                .getRun()\n                .then(clone)\n                .then(advance)\n                .then(markSaved);\n    };\n}\n\nmodule.exports = ScenarioManager;\n","/**\n * ## Baseline\n *\n * An instance of a [Run Manager](../../run-manager/) with a baseline strategy is included automatically in every instance of a [Scenario Manager](../), and is accessible from the Scenario Manager at `.baseline`.\n *\n * A baseline is defined as a run \"advanced to the end\" using just the model defaults. The baseline run is typically displayed in the project's UI as part of a run comparison table or chart.\n *\n * The `baseline` strategy looks for the most recent run named as 'Baseline' (or named as specified in the `baseline.runName` [configuration option of the Scenario Manager](../#configuration-options)) that is flagged as `saved` and not `trashed`. If the strategy cannot find such a run, it creates a new run and immediately executes a set of initialization operations. \n *\n * Comparing against a baseline run is optional in a Scenario Manager; you can [configure](../#configuration-options) your Scenario Manager to not include one. See [more information](../#properties) on using `.baseline` within the Scenario Manager.\n *\n * See also: [additional information on run strategies](../../strategies/).\n *\n */\n\n'use strict';\n\nvar ReuseinitStrategy = require('../run-strategies/reuse-last-initialized');\n\nmodule.exports = function (options) {\n    var defaults = {\n        baselineName: 'Baseline',\n        initOperation: [{ stepTo: 'end' }]\n    };\n    var strategyOptions = options ? options.strategyOptions : {};\n    var opts = $.extend({}, defaults, strategyOptions);\n    return new ReuseinitStrategy({\n        strategyOptions: {\n            initOperation: opts.initOperation,\n            flag: {\n                saved: true,\n                trashed: false,\n                name: opts.baselineName\n            }\n        }\n    });\n};\n","/**\n * ## Current (reuse-last-unsaved)\n *\n * An instance of a [Run Manager](../../run-manager/) with this strategy is included automatically in every instance of a [Scenario Manager](../), and is accessible from the Scenario Manager at `.current`.\n *\n * The `reuse-last-unsaved` strategy returns the most recent run that is not flagged as `trashed` and also not flagged as `saved`.\n * \n * Using this strategy means that end users continue working with the most recent run that has not been explicitly flagged by the project. However, if there are no runs for this end user, a new run is created.\n * \n * Specifically, the strategy is:\n *\n * * Check the `saved` and `trashed` fields of the run to determine if the run has been explicitly saved or explicitly flagged as no longer useful.\n *     * Return the most recent run that is not `trashed` and also not `saved`.\n *     * If there are no runs, create a new run for this end user. \n *\n * See [more information](../#properties) on using `.current` within the Scenario Manager.\n *\n * See also: [additional information on run strategies](../../strategies/).\n */\n\n'use strict';\nvar classFrom = require('../../util/inherit');\nvar injectFiltersFromSession = require('../strategy-utils').injectFiltersFromSession;\nvar injectScopeFromSession = require('../strategy-utils').injectScopeFromSession;\n\nvar Base = {};\n\n//TODO: Make a more generic version of this called 'reuse-by-matching-filter';\nmodule.exports = classFrom(Base, {\n    constructor: function (options) {\n        var strategyOptions = options ? options.strategyOptions : {};\n        this.options = strategyOptions;\n    },\n\n    reset: function (runService, userSession, options) {\n        var opt = injectScopeFromSession(runService.getCurrentConfig(), userSession);\n        return runService.create(opt, options).then(function (createResponse) {\n            return $.extend(true, {}, createResponse, { freshlyCreated: true });\n        });\n    },\n\n    getRun: function (runService, userSession, opts) {\n        var runopts = runService.getCurrentConfig();\n        var filter = injectFiltersFromSession({ \n            saved: false,\n            trashed: false,\n            model: runopts.model,\n        }, userSession);\n        var me = this;\n        var outputModifiers = { \n            startrecord: 0,\n            endrecord: 0,\n            sort: 'created', \n            direction: 'desc'\n        };\n        return runService.query(filter, outputModifiers).then(function (runs) {\n            if (!runs.length) {\n                return me.reset(runService, userSession);\n            }\n            return runs[0];\n        });\n    }\n}, { requiresAuth: false });","'use strict';\n\n\nmodule.exports = {\n    reset: function (params, options, manager) {\n        return manager.reset(options);\n    }\n};\n","'use strict';\n\nvar RunService = require('../service/run-api-service');\n\nmodule.exports = {\n    mergeRunOptions: function (run, options) {\n        if (run instanceof RunService) {\n            run.updateConfig(options);\n            return run;\n        } \n        return $.extend(true, {}, run, options);\n    },\n\n    injectFiltersFromSession: function (currentFilter, session, options) {\n        var defaults = {\n            scopeByGroup: true,\n            scopeByUser: true,\n        };\n        var opts = $.extend(true, {}, defaults, options);\n\n        var filter = $.extend(true, {}, currentFilter);\n        if (opts.scopeByGroup && session && session.groupName) {\n            filter['scope.group'] = session.groupName;\n        }\n        if (opts.scopeByUser && session && session.userId) {\n            filter['user.id'] = session.userId;\n        }\n        return filter;\n    },\n\n    injectScopeFromSession: function (currentParams, session) {\n        var group = session && session.groupName;\n        var params = $.extend(true, {}, currentParams);\n        if (group) {\n            $.extend(params, {\n                scope: { group: group }\n            });\n        }\n        return params;\n    }\n};","/**\n* ## World Manager\n*\n* As discussed under the [World API Adapter](../world-api-adapter/), a [run](../../../glossary/#run) is a collection of end user interactions with a project and its model. For building multiplayer simulations you typically want multiple end users to share the same set of interactions, and work within a common state. Epicenter allows you to create \"worlds\" to handle such cases.\n*\n* The World Manager provides an easy way to track and access the current world and run for particular end users. It is typically used in pages that end users will interact with. (The related [World API Adapter](../world-api-adapter/) handles creating multiplayer worlds, and adding and removing end users and runs from a world. Because of this, typically the World Adapter is used for facilitator pages in your project.)\n*\n* ### Using the World Manager\n*\n* To use the World Manager, instantiate it. Then, make calls to any of the methods you need.\n*\n* When you instantiate a World Manager, the world's account id, project id, and group are automatically taken from the session (thanks to the [Authentication Service](../auth-api-service)).\n*\n* Note that the World Manager does *not* create worlds automatically. (This is different than the [Run Manager](../run-manager).) However, you can pass in specific options to any runs created by the manager, using a `run` object.\n*\n* The parameters for creating a World Manager are:\n*\n*   * `account`: The **Team ID** in the Epicenter user interface for this project.\n*   * `project`: The **Project ID** for this project.\n*   * `group`: The **Group Name** for this world.\n*   * `run`: Options to use when creating new runs with the manager, e.g. `run: { files: ['data.xls'] }`.\n*   * `run.model`: The name of the primary model file for this project. Required if you have not already passed it in as part of the `options` parameter for an enclosing call.\n*\n* For example:\n*\n*       var wMgr = new F.manager.WorldManager({\n*          account: 'acme-simulations',\n*          project: 'supply-chain-game',\n*          run: { model: 'supply-chain.py' },\n*          group: 'team1'\n*       });\n*\n*       wMgr.getCurrentRun();\n*/\n\n'use strict';\n\nvar WorldApi = require('../service/world-api-adapter');\nvar RunManager = require('./run-manager');\nvar AuthManager = require('./auth-manager');\nvar worldApi;\n\nfunction buildStrategy(worldId, dtd) {\n\n    return function Ctor(options) {\n        this.options = options;\n\n        $.extend(this, {\n            reset: function () {\n                throw new Error('not implementd. Need api changes');\n            },\n\n            getRun: function (runService) {\n                var me = this;\n                //get or create!\n                // Model is required in the options\n                var model = this.options.run.model || this.options.model;\n                return worldApi.getCurrentRunId({ model: model, filter: worldId })\n                    .then(function (runId) {\n                        return runService.load(runId);\n                    })\n                    .then(function (run) {\n                        dtd.resolveWith(me, [run]);\n                    })\n                    .fail(dtd.reject);\n            }\n        }\n        );\n    };\n}\n\n\nmodule.exports = function (options) {\n    this.options = options || { run: {}, world: {} };\n\n    $.extend(true, this.options, this.options.run);\n    $.extend(true, this.options, this.options.world);\n\n    worldApi = new WorldApi(this.options);\n    this._auth = new AuthManager();\n    var me = this;\n\n    var api = {\n\n        /**\n        * Returns the current world (object) and an instance of the [World API Adapter](../world-api-adapter/).\n        *\n        * **Example**\n        *\n        *       wMgr.getCurrentWorld()\n        *           .then(function(world, worldAdapter) {\n        *               console.log(world.id);\n        *               worldAdapter.getCurrentRunId();\n        *           });\n        *\n        * **Parameters**\n        * @param {string} userId (Optional) The id of the user whose world is being accessed. Defaults to the user in the current session.\n        * @param {string} groupName (Optional) The name of the group whose world is being accessed. Defaults to the group for the user in the current session.\n        * @return {Promise}\n        */\n        getCurrentWorld: function (userId, groupName) {\n            var session = this._auth.getCurrentUserSessionInfo();\n            if (!userId) {\n                userId = session.userId;\n            }\n            if (!groupName) {\n                groupName = session.groupName;\n            }\n            return worldApi.getCurrentWorldForUser(userId, groupName);\n        },\n\n        /**\n        * Returns the current run (object) and an instance of the [Run API Service](../run-api-service/).\n        *\n        * **Example**\n        *\n        *       wMgr.getCurrentRun('myModel.py')\n        *           .then(function(run, runService) {\n        *               console.log(run.id);\n        *               runService.do('startGame');\n        *           });\n        *\n        * **Parameters**\n        * @param {string} model (Optional) The name of the model file. Required if not already passed in as `run.model` when the World Manager is created.\n        * @return {Promise}\n        */\n        getCurrentRun: function (model) {\n            var dtd = $.Deferred();\n            var session = this._auth.getCurrentUserSessionInfo();\n            var curUserId = session.userId;\n            var curGroupName = session.groupName;\n\n            function getAndRestoreLatestRun(world) {\n                if (!world) {\n                    return dtd.reject({ error: 'The user is not part of any world!' });\n                }\n\n                var currentWorldId = world.id;\n                var runOpts = $.extend(true, me.options, { model: model });\n                var strategy = buildStrategy(currentWorldId, dtd);\n                var opt = $.extend(true, {}, {\n                    strategy: strategy,\n                    run: runOpts\n                });\n                var rm = new RunManager(opt);\n                return rm.getRun()\n                    .then(function (run) {\n                        dtd.resolve(run, rm.runService, rm);\n                    });\n            }\n\n            this.getCurrentWorld(curUserId, curGroupName)\n                .then(getAndRestoreLatestRun);\n\n            return dtd.promise();\n        }\n    };\n\n    $.extend(this, api);\n};\n","/**\n * ## File API Service\n *\n * The File API Service allows you to upload and download files directly onto Epicenter, analogous to using the File Manager UI in Epicenter directly or SFTPing files in. It is based on the Epicenter File API.\n *\n *       var fa = new F.service.File({\n *          account: 'acme-simulations',\n *          project: 'supply-chain-game',\n *       });\n *       fa.create('test.txt', 'these are my filecontents');\n *\n *       // alternatively, create a new file using a file uploaded through a file input\n *       // <input id=\"fileupload\" type=\"file\">\n *       //\n *       $('#fileupload').on('change', function (e) {\n *          var file = e.target.files[0];\n *          var data = new FormData();\n *          data.append('file', file, file.name);\n *          fa.create(file.name, data);\n *       });\n */\n\n'use strict';\n\nvar ConfigService = require('./configuration-service');\nvar TransportFactory = require('../transport/http-transport-factory');\nvar SessionManager = require('../store/session-manager');\n\nmodule.exports = function (config) {\n    var defaults = {\n        /**\n         * For projects that require authentication, pass in the user access token (defaults to empty string). If the user is already logged in to Epicenter, the user access token is already set in a cookie and automatically loaded from there. (See [more background on access tokens](../../../project_access/)).\n         * @see [Authentication API Service](../auth-api-service/) for getting tokens.\n         * @type {String}\n         */\n        token: undefined,\n\n        /**\n         * The account id. In the Epicenter UI, this is the **Team ID** (for team projects) or **User ID** (for personal projects). Defaults to undefined.\n         * @type {String}\n         */\n        account: undefined,\n\n        /**\n         * The project id. Defaults to undefined.\n         * @type {String}\n         */\n        project: undefined,\n\n        /**\n         * The folder type.  One of `model` | `static` | `node`.\n         * @type {String}\n         */\n        folderType: 'static',\n\n\n        /**\n         * Options to pass on to the underlying transport layer. All jquery.ajax options at http://api.jquery.com/jQuery.ajax/ are available. Defaults to empty object.\n         * @type {Object}\n         */\n        transport: {}\n    };\n\n    this.sessionManager = new SessionManager();\n    var serviceOptions = this.sessionManager.getMergedOptions(defaults, config);\n    var urlConfig = new ConfigService(serviceOptions).get('server');\n    if (serviceOptions.account) {\n        urlConfig.accountPath = serviceOptions.account;\n    }\n    if (serviceOptions.project) {\n        urlConfig.projectPath = serviceOptions.project;\n    }\n\n    var httpOptions = $.extend(true, {}, serviceOptions.transport, {\n        url: urlConfig.getAPIPath('file')\n    });\n\n    if (serviceOptions.token) {\n        httpOptions.headers = {\n            Authorization: 'Bearer ' + serviceOptions.token\n        };\n    }\n    var http = new TransportFactory(httpOptions);\n\n    function uploadBody(fileName, contents) {\n        var boundary = '---------------------------7da24f2e50046';\n\n        return {\n            body: '--' + boundary + '\\r\\n' +\n                    'Content-Disposition: form-data; name=\"file\";' +\n                    'filename=\"' + fileName + '\"\\r\\n' +\n                    'Content-type: text/html\\r\\n\\r\\n' +\n                    contents + '\\r\\n' +\n                    '--' + boundary + '--',\n            boundary: boundary\n        };\n    }\n\n    function uploadFileOptions(filePath, contents, options) {\n        filePath = filePath.split('/');\n        var fileName = filePath.pop();\n        filePath = filePath.join('/');\n        var path = serviceOptions.folderType + '/' + filePath;\n\n        var extraParams = {};\n        if (contents instanceof FormData) {\n            extraParams = {\n                data: contents,\n                processData: false,\n                contentType: false,\n            };\n        } else {\n            var upload = uploadBody(fileName, contents);\n            extraParams = {\n                data: upload.body,\n                contentType: 'multipart/form-data; boundary=' + upload.boundary\n            };\n        }\n\n        return $.extend(true, {}, serviceOptions, options, {\n            url: urlConfig.getAPIPath('file') + path,\n        }, extraParams);\n    }\n\n    var publicAsyncAPI = {\n        /**\n         * Get a directory listing, or contents of a file.\n         * @param {String} filePath  Path to the file\n         * @param {Object} options (Optional) Overrides for configuration options.\n         * @return {Promise}\n         */\n        getContents: function (filePath, options) {\n            var path = serviceOptions.folderType + '/' + filePath;\n            var httpOptions = $.extend(true, {}, serviceOptions, options, {\n                url: urlConfig.getAPIPath('file') + path\n            });\n            return http.get('', httpOptions);\n        },\n\n        /**\n         * Replaces the file at the given file path.\n         * @param  {String} filePath Path to the file\n         * @param  {String | FormData } contents Contents to write to file\n         * @param  {Object} options  (Optional) Overrides for configuration options\n         * @return {Promise}\n         */\n        replace: function (filePath, contents, options) {\n            var httpOptions = uploadFileOptions(filePath, contents, options);\n            return http.put(httpOptions.data, httpOptions);\n        },\n\n        /**\n         * Creates a file in the given file path.\n         * @param  {String} filePath Path to the file\n         * @param  {String | FormData } contents Contents to write to file\n         * @param  {Boolean} replaceExisting Replace file if it already exists; defaults to false\n         * @param  {Object} options (Optional) Overrides for configuration options\n         * @return {Promise}\n         */\n        create: function (filePath, contents, replaceExisting, options) {\n            var httpOptions = uploadFileOptions(filePath, contents, options);\n            var prom = http.post(httpOptions.data, httpOptions);\n            var me = this;\n            if (replaceExisting === true) {\n                prom = prom.then(null, function (xhr) {\n                    var conflictStatus = 409;\n                    if (xhr.status === conflictStatus) {\n                        return me.replace(filePath, contents, options);\n                    }\n                });\n            }\n            return prom;\n        },\n\n        /**\n         * Removes the file.\n         * @param  {String} filePath Path to the file\n         * @param  {Object} options  (Optional) Overrides for configuration options\n         * @return {Promise}\n         */\n        remove: function (filePath, options) {\n            var path = serviceOptions.folderType + '/' + filePath;\n            var httpOptions = $.extend(true, {}, serviceOptions, options, {\n                url: urlConfig.getAPIPath('file') + path\n            });\n            return http.delete(null, httpOptions);\n        },\n\n        /**\n         * Renames the file.\n         * @param  {String} filePath Path to the file\n         * @param  {String} newName  New name of file\n         * @param  {Object} options  (Optional) Overrides for configuration options\n         * @return {Promise}\n         */\n        rename: function (filePath, newName, options) {\n            var path = serviceOptions.folderType + '/' + filePath;\n            var httpOptions = $.extend(true, {}, serviceOptions, options, {\n                url: urlConfig.getAPIPath('file') + path\n            });\n            return http.patch({ name: newName }, httpOptions);\n        }\n    };\n\n    $.extend(this, publicAsyncAPI);\n};\n","/**\n * ## Asset API Adapter\n *\n * The Asset API Adapter allows you to store assets -- resources or files of any kind -- used by a project with a scope that is specific to project, group, or end user.\n *\n * Assets are used with [team projects](../../../project_admin/#team). One common use case is having end users in a [group](../../../glossary/#groups) or in a [multiplayer world](../../../glossary/#world) upload data -- videos created during game play, profile pictures for customizing their experience, etc. -- as part of playing through the project.\n *\n * Resources created using the Asset Adapter are scoped:\n *\n *  * Project assets are writable only by [team members](../../../glossary/#team), that is, Epicenter authors.\n *  * Group assets are writable by anyone with access to the project that is part of that particular [group](../../../glossary/#groups). This includes all [team members](../../../glossary/#team) (Epicenter authors) and any [end users](../../../glossary/#users) who are members of the group -- both facilitators and standard end users.\n *  * User assets are writable by the specific end user, and by the facilitator of the group.\n *  * All assets are readable by anyone with the exact URI.\n *\n * To use the Asset Adapter, instantiate it and then access the methods provided. Instantiating requires the account id (**Team ID** in the Epicenter user interface) and project id (**Project ID**). The group name is required for assets with a group scope, and the group name and userId are required for assets with a user scope. If not included, they are taken from the logged in user's session information if needed.\n *\n * When creating an asset, you can pass in text (encoded data) to the `create()` call. Alternatively, you can make the `create()` call as part of an HTML form and pass in a file uploaded via the form.\n *\n *       // instantiate the Asset Adapter\n *       var aa = new F.service.Asset({\n *          account: 'acme-simulations',\n *          project: 'supply-chain-game',\n *          group: 'team1',\n *          userId: '12345'\n *       });\n *\n *       // create a new asset using encoded text\n *       aa.create('test.txt', {\n *           encoding: 'BASE_64',\n *           data: 'VGhpcyBpcyBhIHRlc3QgZmlsZS4=',\n *           contentType: 'text/plain'\n *       }, { scope: 'user' });\n *\n *       // alternatively, create a new asset using a file uploaded through a form\n *       // this sample code goes with an html form that looks like this:\n *       //\n *       // <form id=\"upload-file\">\n *       //   <input id=\"file\" type=\"file\">\n *       //   <input id=\"filename\" type=\"text\" value=\"myFile.txt\">\n *       //   <button type=\"submit\">Upload myFile</button>\n *       // </form>\n *       //\n *       $('#upload-file').on('submit', function (e) {\n *          e.preventDefault();\n *          var filename = $('#filename').val();\n *          var data = new FormData();\n *          var inputControl = $('#file')[0];\n *          data.append('file', inputControl.files[0], filename);\n *\n *          aa.create(filename, data, { scope: 'user' });\n *       });\n *\n */\n\n'use strict';\n\nvar ConfigService = require('./configuration-service');\nvar TransportFactory = require('../transport/http-transport-factory');\nvar _pick = require('../util/object-util')._pick;\nvar SessionManager = require('../store/session-manager');\n\nvar apiEndpoint = 'asset';\n\nmodule.exports = function (config) {\n    var defaults = {\n        /**\n         * For projects that require authentication, pass in the user access token (defaults to empty string). If the user is already logged in to Epicenter, the user access token is already set in a cookie and automatically loaded from there. (See [more background on access tokens](../../../project_access/)).\n         * @see [Authentication API Service](../auth-api-service/) for getting tokens.\n         * @type {String}\n         */\n        token: undefined,\n        /**\n         * The account id. In the Epicenter UI, this is the **Team ID** (for team projects). If left undefined, taken from the URL.\n         * @type {String}\n         */\n        account: undefined,\n        /**\n         * The project id. If left undefined, taken from the URL.\n         * @type {String}\n         */\n        project: undefined,\n        /**\n         * The group name. Defaults to session's `groupName`.\n         * @type {String}\n         */\n        group: undefined,\n        /**\n         * The user id. Defaults to session's `userId`.\n         * @type {String}\n         */\n        userId: undefined,\n        /**\n         * The scope for the asset. Valid values are: `user`, `group`, and `project`. See above for the required permissions to write to each scope. Defaults to `user`, meaning the current end user or a facilitator in the end user's group can edit the asset.\n         * @type {String}\n         */\n        scope: 'user',\n        /**\n         * Determines if a request to list the assets in a scope includes the complete URL for each asset (`true`), or only the file names of the assets (`false`). Defaults to `true`.\n         * @type {boolean}\n         */\n        fullUrl: true,\n        /**\n         * The transport object contains the options passed to the XHR request.\n         * @type {object}\n         */\n        transport: {\n            processData: false\n        }\n    };\n    this.sessionManager = new SessionManager();\n    var serviceOptions = this.sessionManager.getMergedOptions(defaults, config);\n    var urlConfig = new ConfigService(serviceOptions).get('server');\n\n    if (!serviceOptions.account) {\n        serviceOptions.account = urlConfig.accountPath;\n    }\n\n    if (!serviceOptions.project) {\n        serviceOptions.project = urlConfig.projectPath;\n    }\n\n    var transportOptions = $.extend(true, {}, serviceOptions.transport, {\n        url: urlConfig.getAPIPath(apiEndpoint)\n    });\n\n    if (serviceOptions.token) {\n        transportOptions.headers = {\n            Authorization: 'Bearer ' + serviceOptions.token\n        };\n    }\n\n    var http = new TransportFactory(transportOptions);\n\n    var assetApiParams = ['encoding', 'data', 'contentType'];\n    var scopeConfig = {\n        user: ['scope', 'account', 'project', 'group', 'userId'],\n        group: ['scope', 'account', 'project', 'group'],\n        project: ['scope', 'account', 'project'],\n    };\n\n    var validateFilename = function (filename) {\n        if (!filename) {\n            throw new Error('filename is needed.');\n        }\n    };\n\n    var validateUrlParams = function (options) {\n        var partKeys = scopeConfig[options.scope];\n        if (!partKeys) {\n            throw new Error('scope parameter is needed.');\n        }\n\n        $.each(partKeys, function () {\n            if (!options[this]) {\n                throw new Error(this + ' parameter is needed.');\n            }\n        });\n    };\n\n    var buildUrl = function (filename, options) {\n        validateUrlParams(options);\n        var partKeys = scopeConfig[options.scope];\n        var parts = $.map(partKeys, function (key) {\n            return options[key];\n        });\n        if (filename) {\n            // This prevents adding a trailing / in the URL as the Asset API\n            // does not work correctly with it\n            filename = '/' + filename;\n        }\n        return urlConfig.getAPIPath(apiEndpoint) + parts.join('/') + filename;\n    };\n\n    // Private function, all requests follow a more or less same approach to\n    // use the Asset API and the difference is the HTTP verb\n    //\n    // @param {string} method` (Required) HTTP verb\n    // @param {string} filename` (Required) Name of the file to delete/replace/create\n    // @param {object} params` (Optional) Body parameters to send to the Asset API\n    // @param {object} options` (Optional) Options object to override global options.\n    var upload = function (method, filename, params, options) {\n        validateFilename(filename);\n        // make sure the parameter is clean\n        method = method.toLowerCase();\n        var contentType = params instanceof FormData === true ? false : 'application/json';\n        if (contentType === 'application/json') {\n            // whitelist the fields that we actually can send to the api\n            params = _pick(params, assetApiParams);\n        } else { // else we're sending form data which goes directly in request body\n            // For multipart/form-data uploads the filename is not set in the URL,\n            // it's getting picked by the FormData field filename.\n            filename = method === 'post' || method === 'put' ? '' : filename;\n        }\n        var urlOptions = $.extend({}, serviceOptions, options);\n        var url = buildUrl(filename, urlOptions);\n        var createOptions = $.extend(true, {}, urlOptions, { url: url, contentType: contentType });\n\n        return http[method](params, createOptions);\n    };\n\n    var publicAPI = {\n        /**\n        * Creates a file in the Asset API. The server returns an error (status code `409`, conflict) if the file already exists, so\n        * check first with a `list()` or a `get()`.\n        *\n        *  **Example**\n        *\n        *       var aa = new F.service.Asset({\n        *          account: 'acme-simulations',\n        *          project: 'supply-chain-game',\n        *          group: 'team1',\n        *          userId: ''\n        *       });\n        *\n        *       // create a new asset using encoded text\n        *       aa.create('test.txt', {\n        *           encoding: 'BASE_64',\n        *           data: 'VGhpcyBpcyBhIHRlc3QgZmlsZS4=',\n        *           contentType: 'text/plain'\n        *       }, { scope: 'user' });\n        *\n        *       // alternatively, create a new asset using a file uploaded through a form\n        *       // this sample code goes with an html form that looks like this:\n        *       //\n        *       // <form id=\"upload-file\">\n        *       //   <input id=\"file\" type=\"file\">\n        *       //   <input id=\"filename\" type=\"text\" value=\"myFile.txt\">\n        *       //   <button type=\"submit\">Upload myFile</button>\n        *       // </form>\n        *       //\n        *       $('#upload-file').on('submit', function (e) {\n        *          e.preventDefault();\n        *          var filename = $('#filename').val();\n        *          var data = new FormData();\n        *          var inputControl = $('#file')[0];\n        *          data.append('file', inputControl.files[0], filename);\n        *\n        *          aa.create(filename, data, { scope: 'user' });\n        *       });\n        *\n        *\n        *  **Parameters**\n        * @param {string} filename (Required) Name of the file to create.\n        * @param {object} params (Optional) Body parameters to send to the Asset API. Required if the `options.transport.contentType` is `application/json`, otherwise ignored.\n        * @param {string} params.encoding Either `HEX` or `BASE_64`. Required if `options.transport.contentType` is `application/json`.\n        * @param {string} params.data The encoded data for the file. Required if `options.transport.contentType` is `application/json`.\n        * @param {string} params.contentType The mime type of the file. Optional.\n        * @param {object} options (Optional) Options object to override global options.\n        * @return {Promise}\n        */\n        create: function (filename, params, options) {\n            return upload('post', filename, params, options);\n        },\n\n        /**\n        * Gets a file from the Asset API, fetching the asset content. (To get a list\n        * of the assets in a scope, use `list()`.)\n        *\n        *  **Parameters**\n        * @param {string} filename (Required) Name of the file to retrieve.\n        * @param {object} options (Optional) Options object to override global options.\n        * @return {Promise}\n        */\n        get: function (filename, options) {\n            var getServiceOptions = _pick(serviceOptions, ['scope', 'account', 'project', 'group', 'userId']);\n            var urlOptions = $.extend({}, getServiceOptions, options);\n            var url = buildUrl(filename, urlOptions);\n            var getOptions = $.extend(true, {}, urlOptions, { url: url });\n\n            return http.get({}, getOptions);\n        },\n\n        /**\n        * Gets the list of the assets in a scope.\n        *\n        * **Example**\n        *\n        *       aa.list({ fullUrl: true }).then(function(fileList){\n        *           console.log('array of files = ', fileList);\n        *       });\n        *\n        *  **Parameters**\n        * @param {object} options (Optional) Options object to override global options.\n        * @param {string} options.scope (Optional) The scope (`user`, `group`, `project`).\n        * @param {boolean} options.fullUrl (Optional) Determines if the list of assets in a scope includes the complete URL for each asset (`true`), or only the file names of the assets (`false`).\n        * @return {Promise}\n        */\n        list: function (options) {\n            var dtd = $.Deferred();\n            var me = this;\n            var urlOptions = $.extend({}, serviceOptions, options);\n            var url = buildUrl('', urlOptions);\n            var getOptions = $.extend(true, {}, urlOptions, { url: url });\n            var fullUrl = getOptions.fullUrl;\n\n            if (!fullUrl) {\n                return http.get({}, getOptions);\n            }\n\n            http.get({}, getOptions)\n                .then(function (files) {\n                    var fullPathFiles = $.map(files, function (file) {\n                        return buildUrl(file, urlOptions);\n                    });\n                    dtd.resolveWith(me, [fullPathFiles]);\n                })\n                .fail(dtd.reject);\n\n            return dtd.promise();\n        },\n\n        /**\n        * Replaces an existing file in the Asset API.\n        *\n        * **Example**\n        *\n        *       // replace an asset using encoded text\n        *       aa.replace('test.txt', {\n        *           encoding: 'BASE_64',\n        *           data: 'VGhpcyBpcyBhIHNlY29uZCB0ZXN0IGZpbGUu',\n        *           contentType: 'text/plain'\n        *       }, { scope: 'user' });\n        *\n        *       // alternatively, replace an asset using a file uploaded through a form\n        *       // this sample code goes with an html form that looks like this:\n        *       //\n        *       // <form id=\"replace-file\">\n        *       //   <input id=\"file\" type=\"file\">\n        *       //   <input id=\"replace-filename\" type=\"text\" value=\"myFile.txt\">\n        *       //   <button type=\"submit\">Replace myFile</button>\n        *       // </form>\n        *       //\n        *       $('#replace-file').on('submit', function (e) {\n        *          e.preventDefault();\n        *          var filename = $('#replace-filename').val();\n        *          var data = new FormData();\n        *          var inputControl = $('#file')[0];\n        *          data.append('file', inputControl.files[0], filename);\n        *\n        *          aa.replace(filename, data, { scope: 'user' });\n        *       });\n        *\n        *  **Parameters**\n        * @param {string} filename (Required) Name of the file being replaced.\n        * @param {object} params (Optional) Body parameters to send to the Asset API. Required if the `options.transport.contentType` is `application/json`, otherwise ignored.\n        * @param {string} params.encoding Either `HEX` or `BASE_64`. Required if `options.transport.contentType` is `application/json`.\n        * @param {string} params.data The encoded data for the file. Required if `options.transport.contentType` is `application/json`.\n        * @param {string} params.contentType The mime type of the file. Optional.\n        * @param {object} options (Optional) Options object to override global options.\n        * @return {Promise}\n        */\n        replace: function (filename, params, options) {\n            return upload('put', filename, params, options);\n        },\n\n        /**\n        * Deletes a file from the Asset API.\n        *\n        * **Example**\n        *\n        *       aa.delete(sampleFileName);\n        *\n        *  **Parameters**\n        * @param {string} filename (Required) Name of the file to delete.\n        * @param {object} options (Optional) Options object to override global options.\n        * @return {Promise}\n        */\n        delete: function (filename, options) {\n            return upload('delete', filename, {}, options);\n        },\n\n        assetUrl: function (filename, options) {\n            var urlOptions = $.extend({}, serviceOptions, options);\n            return buildUrl(filename, urlOptions);\n        }\n    };\n    $.extend(this, publicAPI);\n};\n","/**\n *\n * ## Authentication API Service\n *\n * The Authentication API Service provides a method for logging in, which creates and returns a user access token.\n *\n * User access tokens are required for each call to Epicenter. (See [Project Access](../../../project_access/) for more information.)\n *\n * If you need additional functionality -- such as tracking session information, easily retrieving the user token, or getting the groups to which an end user belongs -- consider using the [Authorization Manager](../auth-manager/) instead.\n *\n *      var auth = new F.service.Auth();\n *      auth.login({ userName: 'jsmith@acmesimulations.com',\n *                  password: 'passw0rd' });\n */\n\n'use strict';\n\nvar ConfigService = require('./configuration-service');\nvar TransportFactory = require('../transport/http-transport-factory');\n\nmodule.exports = function (config) {\n    var defaults = {\n        /**\n         * Email or username to use for logging in. Defaults to empty string.\n         * @type {String}\n         */\n        userName: '',\n\n        /**\n         * Password for specified `userName`. Defaults to empty string.\n         * @type {String}\n         */\n        password: '',\n\n        /**\n         * The account id for this `userName`. In the Epicenter UI, this is the **Team ID** (for team projects) or the **User ID** (for personal projects). Required if the `userName` is for an [end user](../../../glossary/#users). Defaults to empty string.\n         * @type {String}\n         */\n        account: '',\n\n        /**\n         * Options to pass on to the underlying transport layer. All jquery.ajax options at http://api.jquery.com/jQuery.ajax/ are available. Defaults to empty object.\n         * @type {Object}\n         */\n        transport: {}\n    };\n    var serviceOptions = $.extend({}, defaults, config);\n    var urlConfig = new ConfigService(serviceOptions).get('server');\n\n    var transportOptions = $.extend(true, {}, serviceOptions.transport, {\n        url: urlConfig.getAPIPath('authentication')\n    });\n    var http = new TransportFactory(transportOptions);\n\n    var publicAPI = {\n\n        /**\n         * Logs user in, returning the user access token.\n         *\n         * If no `userName` or `password` were provided in the initial configuration options, they are required in the `options` here. If no `account` was provided in the initial configuration options and the `userName` is for an [end user](../../../glossary/#users), the `account` is required as well.\n         *\n         * **Example**\n         *\n         *      auth.login({\n         *          userName: 'jsmith',\n         *          password: 'passw0rd',\n         *          account: 'acme-simulations' })\n         *      .then(function (token) {\n         *          console.log(\"user access token is: \", token.access_token);\n         *      });\n         *\n         * **Parameters**\n         * @param {Object} options (Optional) Overrides for configuration options.\n         * @return {Promise}\n         */\n        login: function (options) {\n            var httpOptions = $.extend(true, { success: $.noop }, serviceOptions, options);\n            if (!httpOptions.userName || !httpOptions.password) {\n                var resp = { status: 401, statusMessage: 'No username or password specified.' };\n                if (options.error) {\n                    options.error.call(this, resp);\n                }\n\n                return $.Deferred().reject(resp).promise();\n            }\n\n            var postParams = {\n                userName: httpOptions.userName,\n                password: httpOptions.password,\n            };\n            if (httpOptions.account) {\n                //pass in null for account under options if you don't want it to be sent\n                postParams.account = httpOptions.account;\n            }\n\n            return http.post(postParams, httpOptions);\n        },\n\n        // (replace with /* */ comment block, to make visible in docs, once this is more than a noop)\n        //\n        // Logs user out from specified accounts.\n        //\n        // Epicenter logout is not implemented yet, so for now this is a dummy promise that gets automatically resolved.\n        //\n        // **Example**\n        //\n        //      auth.logout();\n        //\n        // **Parameters**\n        // @param {Object} `options` (Optional) Overrides for configuration options.\n        //\n        logout: function (options) {\n            var dtd = $.Deferred();\n            dtd.resolve();\n            return dtd.promise();\n        }\n    };\n\n    $.extend(this, publicAPI);\n};\n","/**\n * ## Channel Service\n *\n * The Epicenter platform provides a push channel, which allows you to publish and subscribe to messages within a [project](../../../glossary/#projects), [group](../../../glossary/#groups), or [multiplayer world](../../../glossary/#world). There are two main use cases for the channel: event notifications and chat messages.\n *\n * If you are developing with Epicenter.js, you should use the [Epicenter Channel Manager](../epicenter-channel-manager/) directly. The Epicenter Channel Manager documentation also has more [background](../epicenter-channel-manager/#background) information on channels and their use.\n *\n * The Channel Service is a building block for this functionality. It creates a publish-subscribe object, allowing you to publish messages, subscribe to messages, or unsubscribe from messages for a given 'topic' on a `$.cometd` transport instance.\n *\n * You'll need to include the `epicenter-multiplayer-dependencies.js` library in addition to the `epicenter.js` library in your project to use the Channel Service. See [Including Epicenter.js](../../#include).\n *\n * To use the Channel Service, instantiate it, then make calls to any of the methods you need.\n *\n *        var cs = new F.service.Channel();\n *        cs.publish('/acme-simulations/supply-chain-game/fall-seminar/run/variables', { price: 50 });\n *\n * If you are working through the [Epicenter Channel Manager](../epicenter-channel-manager/), when you ask to \"get\" a particular channel, you are really asking for an instance of the Channel Service with a topic already set, for example to the appropriate group or world:\n *\n *      var cm = new F.manager.ChannelManager();\n *      var gc = cm.getGroupChannel();\n *      // because we used an Epicenter Channel Manager to get the group channel,\n *      // subscribe() and publish() here default to the base topic for the group\n *      gc.subscribe('', function(data) { console.log(data); });\n *      gc.publish('', { message: 'a new message to the group' });\n *\n * The parameters for instantiating a Channel Service include:\n *\n * * `options` The options object to configure the Channel Service.\n * * `options.base` The base topic. This is added as a prefix to all further topics you publish or subscribe to while working with this Channel Service.\n * * `options.topicResolver` A function that processes all 'topics' passed into the `publish` and `subscribe` methods. This is useful if you want to implement your own serialize functions for converting custom objects to topic names. Returns a String. By default, it just echoes the topic.\n * * `options.transport` The instance of `$.cometd` to hook onto. See http://docs.cometd.org/reference/javascript.html for additional background on cometd.\n */\n\n'use strict';\nvar Channel = function (options) {\n    var defaults = {\n\n        /**\n         * The base topic. This is added as a prefix to all further topics you publish or subscribe to while working with this Channel Service.\n         * @type {string}\n         */\n        base: '',\n\n        /**\n         * A function that processes all 'topics' passed into the `publish` and `subscribe` methods. This is useful if you want to implement your own serialize functions for converting custom objects to topic names. By default, it just echoes the topic.\n         *\n         * **Parameters**\n         *\n         * * `topic` Topic to parse.\n         *\n         * **Return Value**\n         *\n         * * *String*: This function should return a string topic.\n         *\n         * @type {function}\n         * @param {String} topic topic to resolve\n         * @return {String}\n         */\n        topicResolver: function (topic) {\n            return topic;\n        },\n\n        /**\n         * The instance of `$.cometd` to hook onto.\n         * @type {object}\n         */\n        transport: null\n    };\n    this.channelOptions = $.extend(true, {}, defaults, options);\n};\n\nvar makeName = function (channelName, topic) {\n    //Replace trailing/double slashes\n    var newName = (channelName ? (channelName + '/' + topic) : topic).replace(/\\/\\//g, '/').replace(/\\/$/, '');\n    return newName;\n};\n\n\nChannel.prototype = $.extend(Channel.prototype, {\n\n    // future functionality:\n    //      // Set the context for the callback\n    //      cs.subscribe('run', function () { this.innerHTML = 'Triggered'}, document.body);\n     //\n     //      // Control the order of operations by setting the `priority`\n     //      cs.subscribe('run', cb, this, {priority: 9});\n     //\n     //      // Only execute the callback, `cb`, if the value of the `price` variable is 50\n     //      cs.subscribe('run/variables/price', cb, this, {priority: 30, value: 50});\n     //\n     //      // Only execute the callback, `cb`, if the value of the `price` variable is greater than 50\n     //      subscribe('run/variables/price', cb, this, {priority: 30, value: '>50'});\n     //\n     //      // Only execute the callback, `cb`, if the value of the `price` variable is even\n     //      subscribe('run/variables/price', cb, this, {priority: 30, value: function (val) {return val % 2 === 0}});\n\n\n    /**\n     * Subscribe to changes on a topic.\n     *\n     * The topic should include the full path of the account id (**Team ID** for team projects), project id, and group name. (In most cases, it is simpler to use the [Epicenter Channel Manager](../epicenter-channel-manager/) instead, in which case this is configured for you.)\n     *\n     *  **Examples**\n     *\n     *      var cb = function(val) { console.log(val.data); };\n     *\n     *      // Subscribe to changes on a top-level 'run' topic\n     *      cs.subscribe('/acme-simulations/supply-chain-game/fall-seminar/run', cb);\n     *\n     *      // Subscribe to changes on children of the 'run' topic. Note this will also be triggered for changes to run.x.y.z.\n     *      cs.subscribe('/acme-simulations/supply-chain-game/fall-seminar/run/*', cb);\n     *\n     *      // Subscribe to changes on both the top-level 'run' topic and its children\n     *      cs.subscribe(['/acme-simulations/supply-chain-game/fall-seminar/run',\n     *          '/acme-simulations/supply-chain-game/fall-seminar/run/*'], cb);\n     *\n     *      // Subscribe to changes on a particular variable\n     *      subscribe('/acme-simulations/supply-chain-game/fall-seminar/run/variables/price', cb);\n     *\n     *\n     * **Return Value**\n     *\n     * * *String* Returns a token you can later use to unsubscribe.\n     *\n     * **Parameters**\n     * @param  {String|Array}   topic    List of topics to listen for changes on.\n     * @param  {Function} callback Callback function to execute. Callback is called with signature `(evt, payload, metadata)`.\n     * @param  {Object}   context  Context in which the `callback` is executed.\n     * @param  {Object}   options  (Optional) Overrides for configuration options.\n     * @param  {Number}   options.priority  Used to control order of operations. Defaults to 0. Can be any +ve or -ve number.\n     * @param  {String|Number|Function}   options.value The `callback` is only triggered if this condition matches. See examples for details.\n     * @return {string} Subscription ID\n     */\n    subscribe: function (topic, callback, context, options) {\n\n        var topics = [].concat(topic);\n        var me = this;\n        var subscriptionIds = [];\n        var opts = me.channelOptions;\n\n        opts.transport.batch(function () {\n            $.each(topics, function (index, topic) {\n                topic = makeName(opts.base, opts.topicResolver(topic));\n                subscriptionIds.push(opts.transport.subscribe(topic, callback));\n            });\n        });\n        return (subscriptionIds[1] ? subscriptionIds : subscriptionIds[0]);\n    },\n\n    /**\n     * Publish data to a topic.\n     *\n     * **Examples**\n     *\n     *      // Send data to all subscribers of the 'run' topic\n     *      cs.publish('/acme-simulations/supply-chain-game/fall-seminar/run', { completed: false });\n     *\n     *      // Send data to all subscribers of the 'run/variables' topic\n     *      cs.publish('/acme-simulations/supply-chain-game/fall-seminar/run/variables', { price: 50 });\n     *\n     * **Parameters**\n     *\n     * @param  {String} topic Topic to publish to.\n     * @param  {*} data  Data to publish to topic.\n     * @return {Array | Object} Responses to published data\n     *\n     */\n    publish: function (topic, data) {\n        var topics = [].concat(topic);\n        var me = this;\n        var returnObjs = [];\n        var opts = me.channelOptions;\n\n\n        opts.transport.batch(function () {\n            $.each(topics, function (index, topic) {\n                topic = makeName(opts.base, opts.topicResolver(topic));\n                if (topic.charAt(topic.length - 1) === '*') {\n                    topic = topic.replace(/\\*+$/, '');\n                    console.warn('You can cannot publish to channels with wildcards. Publishing to ', topic, 'instead');\n                }\n                returnObjs.push(opts.transport.publish(topic, data));\n            });\n        });\n        return (returnObjs[1] ? returnObjs : returnObjs[0]);\n    },\n\n    /**\n     * Unsubscribe from changes to a topic.\n     *\n     * **Example**\n     *\n     *      cs.unsubscribe('sampleToken');\n     *\n     * **Parameters**\n     * @param  {String} token The token for topic is returned when you initially subscribe. Pass it here to unsubscribe from that topic.\n     * @return {Object} reference to current instance\n     */\n    unsubscribe: function (token) {\n        this.channelOptions.transport.unsubscribe(token);\n        return this;\n    },\n\n    /**\n     * Start listening for events on this instance. Signature is same as for jQuery Events: http://api.jquery.com/on/.\n     *\n     * Supported events are: `connect`, `disconnect`, `subscribe`, `unsubscribe`, `publish`, `error`.\n     *\n     * **Parameters**\n     *\n     * @param {string} event The event type. See more detail at jQuery Events: http://api.jquery.com/on/.\n     */\n    on: function (event) {\n        $(this).on.apply($(this), arguments);\n    },\n\n    /**\n     * Stop listening for events on this instance. Signature is same as for jQuery Events: http://api.jquery.com/off/.\n     *\n     * **Parameters**\n     *\n     * @param {string} event The event type. See more detail at jQuery Events: http://api.jquery.com/off/.\n     */\n    off: function (event) {\n        $(this).off.apply($(this), arguments);\n    },\n\n    /**\n     * Trigger events and execute handlers. Signature is same as for jQuery Events: http://api.jquery.com/trigger/.\n     *\n     * **Parameters**\n     *\n     * @param {string} event The event type. See more detail at jQuery Events: http://api.jquery.com/trigger/.\n     */\n    trigger: function (event) {\n        $(this).trigger.apply($(this), arguments);\n    }\n\n});\n\nmodule.exports = Channel;\n","/**\n * @class ConfigurationService\n *\n * All services take in a configuration settings object to configure themselves. A JS hash {} is a valid configuration object, but optionally you can use the configuration service to toggle configs based on the environment\n *\n * @example\n *     var cs = require('configuration-service')({\n *          dev: { //environment\n                port: 3000,\n                host: 'localhost',\n            },\n            prod: {\n                port: 8080,\n                host: 'api.forio.com',\n                logLevel: 'none'\n            },\n            logLevel: 'DEBUG' //global\n *     });\n *\n *      cs.get('logLevel'); //returns 'DEBUG'\n *\n *      cs.setEnv('dev');\n *      cs.get('logLevel'); //returns 'DEBUG'\n *\n *      cs.setEnv('prod');\n *      cs.get('logLevel'); //returns 'none'\n *\n */\n\n'use strict';\nvar urlService = require('./url-config-service');\n\nmodule.exports = function (config) {\n    //TODO: Environments\n    var defaults = {\n        logLevel: 'NONE'\n    };\n    var serviceOptions = $.extend({}, defaults, config);\n    serviceOptions.server = urlService(serviceOptions.server);\n\n    return {\n\n        data: serviceOptions,\n\n        /**\n         * Set the environment key to get configuration options from\n         * @param { string} env\n         */\n        setEnv: function (env) {\n\n        },\n\n        /**\n         * Get configuration.\n         * @param  { string} property optional\n         * @return {*}          Value of property if specified, the entire config object otherwise\n         */\n        get: function (property) {\n            return serviceOptions[property];\n        },\n\n        /**\n         * Set configuration.\n         * @param  { string|Object} key if a key is provided, set a key to that value. Otherwise merge object with current config\n         * @param  {*} value  value for provided key\n         */\n        set: function (key, value) {\n            serviceOptions[key] = value;\n        }\n    };\n};\n\n","/**\n * ## Data API Service\n *\n * The Data API Service allows you to create, access, and manipulate data related to any of your projects. Data are organized in collections. Each collection contains a document; each element of this top-level document is a JSON object. (See additional information on the underlying [Data API](../../../rest_apis/data_api/).)\n *\n * All API calls take in an \"options\" object as the last parameter. The options can be used to extend/override the Data API Service defaults. In particular, there are three required parameters when you instantiate the Data Service:\n *\n * * `account`: Epicenter account id (**Team ID** for team projects, **User ID** for personal projects).\n * * `project`: Epicenter project id.\n * * `root`: The the name of the collection. If you have multiple collections within each of your projects, you can also pass the collection name as an option for each call.\n *\n *       var ds = new F.service.Data({\n *          account: 'acme-simulations',\n *          project: 'supply-chain-game',\n *          root: 'survey-responses',\n *          server: { host: 'api.forio.com' }\n *       });\n *       ds.saveAs('user1',\n *          { 'question1': 2, 'question2': 10,\n *          'question3': false, 'question4': 'sometimes' } );\n *       ds.saveAs('user2',\n *          { 'question1': 3, 'question2': 8,\n *          'question3': true, 'question4': 'always' } );\n *       ds.query('',{ 'question2': { '$gt': 9} });\n *\n * Note that in addition to the `account`, `project`, and `root`, the Data Service parameters optionally include a `server` object, whose `host` field contains the URI of the Forio server. This is automatically set, but you can pass it explicitly if desired. It is most commonly used for clarity when you are [hosting an Epicenter project on your own server](../../../how_to/self_hosting/).\n */\n\n'use strict';\n\nvar ConfigService = require('./configuration-service');\nvar qutil = require('../util/query-util');\nvar TransportFactory = require('../transport/http-transport-factory');\nvar SessionManager = require('../store/session-manager');\n\nmodule.exports = function (config) {\n    var defaults = {\n        /**\n         * Name of collection. Required. Defaults to `/`, that is, the root level of your project at `forio.com/app/your-account-id/your-project-id/`, but must be set to a collection name.\n         * @type {String}\n         */\n        root: '/',\n\n        /**\n         * The account id. In the Epicenter UI, this is the **Team ID** (for team projects) or **User ID** (for personal projects). Defaults to empty string. If left undefined, taken from the URL.\n         * @type {String}\n         */\n        account: undefined,\n\n        /**\n         * The project id. Defaults to empty string. If left undefined, taken from the URL.\n         * @type {String}\n         */\n        project: undefined,\n\n        /**\n         * For operations that require authentication, pass in the user access token (defaults to empty string). If the user is already logged in to Epicenter, the user access token is already set in a cookie and automatically loaded from there. (See [more background on access tokens](../../../project_access/)).\n         * @see [Authentication API Service](../auth-api-service/) for getting tokens.\n         * @type {String}\n         */\n        token: undefined,\n\n        //Options to pass on to the underlying transport layer\n        transport: {}\n    };\n    this.sessionManager = new SessionManager();\n    var serviceOptions = this.sessionManager.getMergedOptions(defaults, config);\n\n    var urlConfig = new ConfigService(serviceOptions).get('server');\n    if (serviceOptions.account) {\n        urlConfig.accountPath = serviceOptions.account;\n    }\n    if (serviceOptions.project) {\n        urlConfig.projectPath = serviceOptions.project;\n    }\n\n    var getURL = function (key, root) {\n        if (!root) {\n            root = serviceOptions.root;\n        }\n        var url = urlConfig.getAPIPath('data') + qutil.addTrailingSlash(root);\n        if (key) {\n            url += qutil.addTrailingSlash(key);\n        }\n        return url;\n    };\n\n    var httpOptions = $.extend(true, {}, serviceOptions.transport, {\n        url: getURL\n    });\n    if (serviceOptions.token) {\n        httpOptions.headers = {\n            Authorization: 'Bearer ' + serviceOptions.token\n        };\n    }\n    var http = new TransportFactory(httpOptions);\n\n    var publicAPI = {\n\n        /**\n         * Search for data within a collection.\n         *\n         * Searching using comparison or logical operators (as opposed to exact matches) requires MongoDB syntax. See the underlying [Data API](../../../rest_apis/data_api/#searching) for additional details.\n         *\n         * **Examples**\n         *\n         *      // request all data associated with document 'user1'\n         *      ds.query('user1');\n         *\n         *      // exact matching:\n         *      // request all documents in collection where 'question2' is 9\n         *      ds.query('', { 'question2': 9});\n         *\n         *      // comparison operators:\n         *      // request all documents in collection\n         *      // where 'question2' is greater than 9\n         *      ds.query('', { 'question2': { '$gt': 9} });\n         *\n         *      // logical operators:\n         *      // request all documents in collection\n         *      // where 'question2' is less than 10, and 'question3' is false\n         *      ds.query('', { '$and': [ { 'question2': { '$lt':10} }, { 'question3': false }] });\n         *\n         *      // regular expresssions: use any Perl-compatible regular expressions\n         *      // request all documents in collection\n         *      // where 'question5' contains the string '.*day'\n         *      ds.query('', { 'question5': { '$regex': '.*day' } });\n         *\n         * **Parameters**\n         * @param {String} key The name of the document to search. Pass the empty string ('') to search the entire collection.\n         * @param {Object} query The query object. For exact matching, this object contains the field name and field value to match. For matching based on comparison, this object contains the field name and the comparison expression. For matching based on logical operators, this object contains an expression using MongoDB syntax. See the underlying [Data API](../../../rest_apis/data_api/#searching) for additional examples.\n         * @param {Object} outputModifier (Optional) Available fields include: `startrecord`, `endrecord`, `sort`, and `direction` (`asc` or `desc`).\n         * @param {Object} options (Optional) Overrides for configuration options.\n         * @return {Promise} \n         */\n        query: function (key, query, outputModifier, options) {\n            var params = $.extend(true, { q: query }, outputModifier);\n            var httpOptions = $.extend(true, {}, serviceOptions, options);\n            httpOptions.url = getURL(key, httpOptions.root);\n            return http.get(params, httpOptions);\n        },\n\n        /**\n         * Save data in an anonymous document within the collection.\n         *\n         * The `root` of the collection must be specified. By default the `root` is taken from the Data Service configuration options; you can also pass the `root` to the `save` call explicitly by overriding the options (third parameter).\n         *\n         * (Additional background: Documents are top-level elements within a collection. Collections must be unique within this account (team or personal account) and project and are set with the `root` field in the `option` parameter. See the underlying [Data API](../../../rest_apis/data_api/) for more information. The `save` method is making a `POST` request.)\n         *\n         * **Example**\n         *\n         *      // Create a new document, with one element, at the default root level\n         *      ds.save('question1', 'yes');\n         *\n         *      // Create a new document, with two elements, at the default root level\n         *      ds.save({ question1:'yes', question2: 32 });\n         *\n         *      // Create a new document, with two elements, at `/students/`\n         *      ds.save({ name:'John', className: 'CS101' }, { root: 'students' });\n         *\n         * **Parameters**\n         *\n         * @param {String|Object} key If `key` is a string, it is the id of the element to save (create) in this document. If `key` is an object, the object is the data to save (create) in this document. In both cases, the id for the document is generated automatically.\n         * @param {Object} value (Optional) The data to save. If `key` is a string, this is the value to save. If `key` is an object, the value(s) to save are already part of `key` and this argument is not required.\n         * @param {Object} options (Optional) Overrides for configuration options. If you want to override the default `root` of the collection, do so here.\n         * @return {Promise} \n         */\n        save: function (key, value, options) {\n            var attrs;\n            if (typeof key === 'object') {\n                attrs = key;\n                options = value;\n            } else {\n                (attrs = {})[key] = value;\n            }\n            var httpOptions = $.extend(true, {}, serviceOptions, options);\n            httpOptions.url = getURL('', httpOptions.root);\n\n            return http.post(attrs, httpOptions);\n        },\n\n        /**\n         * Save (create or replace) data in a named document or element within the collection. \n         * \n         * The `root` of the collection must be specified. By default the `root` is taken from the Data Service configuration options; you can also pass the `root` to the `saveAs` call explicitly by overriding the options (third parameter).\n         *\n         * Optionally, the named document or element can include path information, so that you are saving just part of the document.\n         *\n         * (Additional background: Documents are top-level elements within a collection. Collections must be unique within this account (team or personal account) and project and are set with the `root` field in the `option` parameter. See the underlying [Data API](../../../rest_apis/data_api/) for more information. The `saveAs` method is making a `PUT` request.)\n         *\n         * **Example**\n         *\n         *      // Create (or replace) the `user1` document at the default root level.\n         *      // Note that this replaces any existing content in the `user1` document.\n         *      ds.saveAs('user1',\n         *          { 'question1': 2, 'question2': 10,\n         *           'question3': false, 'question4': 'sometimes' } );\n         *\n         *      // Create (or replace) the `student1` document at the `students` root, \n         *      // that is, the data at `/students/student1/`.\n         *      // Note that this replaces any existing content in the `/students/student1/` document.\n         *      // However, this will keep existing content in other paths of this collection.\n         *      // For example, the data at `/students/student2/` is unchanged by this call.\n         *      ds.saveAs('student1',\n         *          { firstName: 'john', lastName: 'smith' },\n         *          { root: 'students' });\n         *\n         *      // Create (or replace) the `mgmt100/groupB` document at the `myclasses` root,\n         *      // that is, the data at `/myclasses/mgmt100/groupB/`.\n         *      // Note that this replaces any existing content in the `/myclasses/mgmt100/groupB/` document.\n         *      // However, this will keep existing content in other paths of this collection.\n         *      // For example, the data at `/myclasses/mgmt100/groupA/` is unchanged by this call.\n         *      ds.saveAs('mgmt100/groupB',\n         *          { scenarioYear: '2015' },\n         *          { root: 'myclasses' });\n         *\n         * **Parameters**\n         *\n         * @param {String} key Id of the document.\n         * @param {Object} value (Optional) The data to save, in key:value pairs.\n         * @param {Object} options (Optional) Overrides for configuration options. If you want to override the default `root` of the collection, do so here.\n         * @return {Promise} \n         */\n        saveAs: function (key, value, options) {\n            var httpOptions = $.extend(true, {}, serviceOptions, options);\n            httpOptions.url = getURL(key, httpOptions.root);\n\n            return http.put(value, httpOptions);\n        },\n\n        /**\n         * Get data for a specific document or field.\n         *\n         * **Example**\n         *\n         *      ds.load('user1');\n         *      ds.load('user1/question3');\n         *\n         * **Parameters**\n         * @param  {String|Object} key The id of the data to return. Can be the id of a document, or a path to data within that document.\n         * @param {Object} outputModifier (Optional) Available fields include: `startrecord`, `endrecord`, `sort`, and `direction` (`asc` or `desc`).\n         * @param {Object} options Overrides for configuration options.\n         * @return {Promise} \n         */\n        load: function (key, outputModifier, options) {\n            var httpOptions = $.extend(true, {}, serviceOptions, options);\n            httpOptions.url = getURL(key, httpOptions.root);\n            return http.get(outputModifier, httpOptions);\n        },\n\n        /**\n         * Removes data from collection. Only documents (top-level elements in each collection) can be deleted.\n         *\n         * **Example**\n         *\n         *     ds.remove('user1');\n         *\n         *\n         * **Parameters**\n         *\n         * @param {String|Array} keys The id of the document to remove from this collection, or an array of such ids.\n         * @param {Object} options (Optional) Overrides for configuration options.\n         * @return {Promise} \n         */\n        remove: function (keys, options) {\n            var httpOptions = $.extend(true, {}, serviceOptions, options);\n            var params;\n            if ($.isArray(keys)) {\n                params = { id: keys };\n            } else {\n                params = '';\n                httpOptions.url = getURL(keys, httpOptions.root);\n            }\n            return http.delete(params, httpOptions);\n        }\n\n        // Epicenter doesn't allow nuking collections\n        //     /**\n        //      * Removes collection being referenced\n        //      * @return null\n        //      */\n        //     destroy: function (options) {\n        //         return this.remove('', options);\n        //     }\n    };\n\n    $.extend(this, publicAPI);\n};\n","/**\n *\n * ## Group API Adapter\n *\n * The Group API Adapter provides methods to look up, create, change or remove information about groups in a project. It is based on query capabilities of the underlying RESTful [Group API](../../../rest_apis/user_management/group/).\n *\n * This is only needed for Authenticated projects, that is, team projects with [end users and groups](../../../groups_and_end_users/).\n *\n *      var ma = new F.service.Group({ token: 'user-or-project-access-token' });\n *      ma.getGroupsForProject({ account: 'acme', project: 'sample' });\n */\n\n'use strict';\n\nvar serviceUtils = require('./service-utils');\nvar TransportFactory = require('../transport/http-transport-factory');\nvar objectAssign = require('object-assign');\n\nvar apiEndpoint = 'group/local';\n\nvar GroupService = function (config) {\n    var defaults = {\n        /**\n         * Epicenter account name. Defaults to undefined.\n         * @type {string}\n         */\n        account: undefined,\n\n        /**\n         * Epicenter project name. Defaults to undefined.\n         * @type {string}\n         */\n        project: undefined,\n\n        /**\n         * Options to pass on to the underlying transport layer. All jquery.ajax options at http://api.jquery.com/jQuery.ajax/ are available. Defaults to empty object.\n         * @type {object}\n         */\n        transport: {}\n    };\n    var serviceOptions = serviceUtils.getDefaultOptions(defaults, config, { apiEndpoint: apiEndpoint });\n    var transportOptions = serviceOptions.transport;\n    delete serviceOptions.transport;\n    var http = new TransportFactory(transportOptions, serviceOptions);\n    var publicAPI = {\n        /**\n        * Gets information for a group or multiple groups.\n        * @param {Object} params object with query parameters\n        * @patam {string} params.q partial match for name, organization or event.\n        * @patam {string} params.account Epicenter's Team ID\n        * @patam {string} params.project Epicenter's Project ID\n        * @patam {string} params.name Epicenter's Group Name\n        * @param {Object} options (Optional) Overrides for configuration options.\n        * @return {Promise}\n        */\n        getGroups: function (params, options) {\n            //groupID is part of the URL\n            //q, account and project are part of the query string\n            var finalOpts = objectAssign({}, serviceOptions, options);\n            var finalParams;\n            if (typeof params === 'string') {\n                finalOpts.url = serviceUtils.getApiUrl(apiEndpoint + '/' + params, finalOpts);\n            } else {\n                finalParams = params;\n            }\n            return http.get(finalParams, finalOpts);\n        }\n    };\n    objectAssign(this, publicAPI);\n};\n\nmodule.exports = GroupService;\n","/**\n *\n * ## Introspection API Service\n *\n * The Introspection API Service allows you to view a list of the variables and operations in a model. Typically used in conjunction with the [Run API Service](../run-api-service/).\n *\n * The Introspection API Service is not available for Forio SimLang.\n *\n *       var intro = new F.service.Introspect({\n *               account: 'acme-simulations',\n *               project: 'supply-chain-game'\n *       });\n *       intro.byModel('supply-chain.py').then(function(data){ ... });\n *       intro.byRunID('2b4d8f71-5c34-435a-8c16-9de674ab72e6').then(function(data){ ... });\n *\n */\n\n'use strict';\n\nvar ConfigService = require('./configuration-service');\nvar TransportFactory = require('../transport/http-transport-factory');\nvar SessionManager = require('../store/session-manager');\n\nvar apiEndpoint = 'model/introspect';\n\nmodule.exports = function (config) {\n    var defaults = {\n        /**\n         * For projects that require authentication, pass in the user access token (defaults to empty string). If the user is already logged in to Epicenter, the user access token is already set in a cookie and automatically loaded from there. (See [more background on access tokens](../../../project_access/)).\n         * @see [Authentication API Service](../auth-api-service/) for getting tokens.\n         * @type {String}\n         */\n        token: undefined,\n\n        /**\n         * The account id. In the Epicenter UI, this is the **Team ID** (for team projects) or **User ID** (for personal projects). Defaults to empty string. If left undefined, taken from the URL.\n         * @type {String}\n         */\n        account: undefined,\n\n        /**\n         * The project id. Defaults to empty string. If left undefined, taken from the URL.\n         * @type {String}\n         */\n        project: undefined,\n\n    };\n\n    var sessionManager = new SessionManager();\n    var serviceOptions = sessionManager.getMergedOptions(defaults, config);\n\n    var urlConfig = new ConfigService(serviceOptions).get('server');\n    if (serviceOptions.account) {\n        urlConfig.accountPath = serviceOptions.account;\n    }\n    if (serviceOptions.project) {\n        urlConfig.projectPath = serviceOptions.project;\n    }\n\n    var transportOptions = $.extend(true, {}, serviceOptions.transport, {\n        url: urlConfig.getAPIPath(apiEndpoint)\n    });\n    if (serviceOptions.token) {\n        transportOptions.headers = {\n            Authorization: 'Bearer ' + serviceOptions.token\n        };\n    }\n    var http = new TransportFactory(transportOptions);\n\n    var publicAPI = {\n        /**\n         * Get the available variables and operations for a given model file.\n         *\n         * Note: This does not work for any model which requires additional parameters, such as `files`.\n         *\n         * **Example**\n         *\n         *      intro.byModel('abc.vmf')\n         *          .then(function(data) {\n         *              // data contains an object with available functions (used with operations API) and available variables (used with variables API)\n         *              console.log(data.functions);\n         *              console.log(data.variables);\n         *          });\n         *\n         * **Parameters**\n         * @param  {String} modelFile Name of the model file to introspect.\n         * @param  {Object} options (Optional) Overrides for configuration options.\n         * @return {Promise} \n         */\n        byModel: function (modelFile, options) {\n            var opts = $.extend(true, {}, serviceOptions, options);\n            if (!opts.account || !opts.project) {\n                throw new Error('Account and project are required when using introspect#byModel');\n            }\n            if (!modelFile) {\n                throw new Error('modelFile is required when using introspect#byModel');\n            }\n            var url = { url: urlConfig.getAPIPath(apiEndpoint) + [opts.account, opts.project, modelFile].join('/') };\n            var httpOptions = $.extend(true, {}, serviceOptions, options, url);\n            return http.get('', httpOptions);\n        },\n\n        /**\n         * Get the available variables and operations for a given model file.\n         *\n         * Note: This does not work for any model which requires additional parameters such as `files`.\n         *\n         * **Example**\n         *\n         *      intro.byRunID('2b4d8f71-5c34-435a-8c16-9de674ab72e6')\n         *          .then(function(data) {\n         *              // data contains an object with available functions (used with operations API) and available variables (used with variables API)\n         *              console.log(data.functions);\n         *              console.log(data.variables);\n         *          });\n         *\n         * **Parameters**\n         * @param  {String} runID Id of the run to introspect.\n         * @param  {Object} options (Optional) Overrides for configuration options.\n         * @return {Promise} \n         */\n        byRunID: function (runID, options) {\n            if (!runID) {\n                throw new Error('runID is required when using introspect#byModel');\n            }\n            var url = { url: urlConfig.getAPIPath(apiEndpoint) + runID };\n            var httpOptions = $.extend(true, {}, serviceOptions, options, url);\n            return http.get('', httpOptions);\n        }\n    };\n    $.extend(this, publicAPI);\n};\n","/**\n *\n * ## Member API Adapter\n *\n * The Member API Adapter provides methods to look up information about end users for your project and how they are divided across groups. It is based on query capabilities of the underlying RESTful [Member API](../../../rest_apis/user_management/member/).\n *\n * This is only needed for Authenticated projects, that is, team projects with [end users and groups](../../../groups_and_end_users/). For example, if some of your end users are facilitators, or if your end users should be treated differently based on which group they are in, use the Member API to find that information.\n *\n *      var ma = new F.service.Member({ token: 'user-or-project-access-token' });\n *      ma.getGroupsForUser({ userId: 'b6b313a3-ab84-479c-baea-206f6bff337' });\n *      ma.getGroupDetails({ groupId: '00b53308-9833-47f2-b21e-1278c07d53b8' });\n */\n\n'use strict';\n\nvar ConfigService = require('./configuration-service');\nvar TransportFactory = require('../transport/http-transport-factory');\nvar SessionManager = require('../store/session-manager');\nvar _pick = require('../util/object-util')._pick;\nvar apiEndpoint = 'member/local';\n\nmodule.exports = function (config) {\n    var defaults = {\n        /**\n         * Epicenter user id. Defaults to a blank string.\n         * @type {string}\n         */\n        userId: undefined,\n\n        /**\n         * Epicenter group id. Defaults to a blank string. Note that this is the group *id*, not the group *name*.\n         * @type {string}\n         */\n        groupId: undefined,\n\n        /**\n         * Options to pass on to the underlying transport layer. All jquery.ajax options at http://api.jquery.com/jQuery.ajax/ are available. Defaults to empty object.\n         * @type {object}\n         */\n        transport: {}\n    };\n    this.sessionManager = new SessionManager();\n    var serviceOptions = this.sessionManager.getMergedOptions(defaults, config);\n    var urlConfig = new ConfigService(serviceOptions).get('server');\n\n    var transportOptions = $.extend(true, {}, serviceOptions.transport, {\n        url: urlConfig.getAPIPath(apiEndpoint)\n    });\n\n    if (serviceOptions.token) {\n        transportOptions.headers = {\n            Authorization: 'Bearer ' + serviceOptions.token\n        };\n    }\n    var http = new TransportFactory(transportOptions, serviceOptions);\n\n    var getFinalParams = function (params) {\n        if (typeof params === 'object') {\n            return $.extend(true, serviceOptions, params);\n        }\n        return serviceOptions;\n    };\n\n    var patchUserActiveField = function (params, active, options) {\n        var httpOptions = $.extend(true, serviceOptions, options, {\n            url: urlConfig.getAPIPath(apiEndpoint) + params.groupId + '/' + params.userId\n        });\n\n        return http.patch({ active: active }, httpOptions);\n    };\n\n    var publicAPI = {\n\n        /**\n        * Retrieve details about all of the group memberships for one end user. The membership details are returned in an array, with one element (group record) for each group to which the end user belongs.\n        *\n        * In the membership array, each group record includes the group id, project id, account (team) id, and an array of members. However, only the user whose userId is included in the call is listed in the members array (regardless of whether there are other members in this group).\n        *\n        * **Example**\n        *\n        *       var ma = new F.service.Member({ token: 'user-or-project-access-token' });\n        *       ma.getGroupsForUser('42836d4b-5b61-4fe4-80eb-3136e956ee5c')\n        *           .then(function(memberships){\n        *               for (var i=0; i<memberships.length; i++) {\n        *                   console.log(memberships[i].groupId);\n        *               }\n        *           });\n        *\n        *       ma.getGroupsForUser({ userId: '42836d4b-5b61-4fe4-80eb-3136e956ee5c' });\n        *\n        * **Parameters**\n        * @param {string|object} params The user id for the end user. Alternatively, an object with field `userId` and value the user id.\n        * @param {object} options (Optional) Overrides for configuration options.\n        */\n\n        getGroupsForUser: function (params, options) {\n            options = options || {};\n            var httpOptions = $.extend(true, serviceOptions, options);\n            var isString = typeof params === 'string';\n            var objParams = getFinalParams(params);\n            if (!isString && !objParams.userId) {\n                throw new Error('No userId specified.');\n            }\n\n            var getParms = isString ? { userId: params } : _pick(objParams, 'userId');\n            return http.get(getParms, httpOptions);\n        },\n\n        /**\n        * Retrieve details about one group, including an array of all its members.\n        *\n        * **Example**\n        *\n        *       var ma = new F.service.Member({ token: 'user-or-project-access-token' });\n        *       ma.getGroupDetails('80257a25-aa10-4959-968b-fd053901f72f')\n        *           .then(function(group){\n        *               for (var i=0; i<group.members.length; i++) {\n        *                   console.log(group.members[i].userName);\n        *               }\n        *           });\n        *\n        *       ma.getGroupDetails({ groupId: '80257a25-aa10-4959-968b-fd053901f72f' });\n        *\n        * **Parameters**\n        * @param {string|object} params The group id. Alternatively, an object with field `groupId` and value the group id.\n        * @param {object} options (Optional) Overrides for configuration options.\n        * @return {Promise}\n        */\n        getGroupDetails: function (params, options) {\n            options = options || {};\n            var isString = typeof params === 'string';\n            var objParams = getFinalParams(params);\n            if (!isString && !objParams.groupId) {\n                throw new Error('No groupId specified.');\n            }\n\n            var groupId = isString ? params : objParams.groupId;\n            var httpOptions = $.extend(true, serviceOptions,\n                options,\n                { url: urlConfig.getAPIPath(apiEndpoint) + groupId }\n            );\n\n            return http.get({}, httpOptions);\n        },\n\n        /**\n        * Set a particular end user as `active`. Active end users can be assigned to [worlds](../world-manager/) in multiplayer games during automatic assignment.\n        *\n        * **Example**\n        *\n        *       var ma = new F.service.Member({ token: 'user-or-project-access-token' });\n        *       ma.makeUserActive({ userId: '42836d4b-5b61-4fe4-80eb-3136e956ee5c',\n        *                           groupId: '80257a25-aa10-4959-968b-fd053901f72f' });\n        *\n        * **Parameters**\n        * @param {object} params The end user and group information.\n        * @param {string} params.userId The id of the end user to make active.\n        * @param {string} params.groupId The id of the group to which this end user belongs, and in which the end user should become active.\n        * @param {object} options (Optional) Overrides for configuration options.\n        * @return {Promise}\n        */\n        makeUserActive: function (params, options) {\n            return patchUserActiveField(params, true, options);\n        },\n\n        /**\n        * Set a particular end user as `inactive`. Inactive end users are not assigned to [worlds](../world-manager/) in multiplayer games during automatic assignment.\n        *\n        * **Example**\n        *\n        *       var ma = new F.service.Member({ token: 'user-or-project-access-token' });\n        *       ma.makeUserInactive({ userId: '42836d4b-5b61-4fe4-80eb-3136e956ee5c',\n        *                           groupId: '80257a25-aa10-4959-968b-fd053901f72f' });\n        *\n        * **Parameters**\n        * @param {object} params The end user and group information.\n        * @param {string} params.userId The id of the end user to make inactive.\n        * @param {string} params.groupId The id of the group to which this end user belongs, and in which the end user should become inactive.\n        * @param {object} options (Optional) Overrides for configuration options.\n        * @return {Promise}\n        */\n        makeUserInactive: function (params, options) {\n            return patchUserActiveField(params, false, options);\n        }\n    };\n\n    $.extend(this, publicAPI);\n};\n","/**\n *\n * ## Presence API Service\n *\n * The Presence API Service provides methods to get and set the presence of an end user in a project, that is, to indicate whether the end user is online. This happens automatically: in projects that use [channels](../epicenter-channel-manager/), the end user's presence is published automatically on a \"presence\" channel that is specific to each group. You can also use the Presence API Service to do this explicitly: you can make a call to indicate that a particular end user is online or offline. \n *\n * The Presence API Service is only needed for Authenticated projects, that is, team projects with [end users and groups](../../../groups_and_end_users/). It is typically used only in multiplayer projects, to facilitate end users communicating with each other. It is based on the query capabilities of the underlying RESTful [Presence API](../../../rest_apis/multiplayer/presence/).\n *\n *      var pr = new F.service.Presence();\n *      pr.markOnline('example-userId');\n *      pr.markOffline('example-userId');\n *      pr.getStatus();\n */\n\nvar ConfigService = require('./configuration-service');\nvar TransportFactory = require('../transport/http-transport-factory');\nvar SessionManager = require('../store/session-manager');\nvar apiEndpoint = 'presence';\nvar ChannelManager = require('../managers/epicenter-channel-manager');\n\nmodule.exports = function (config) {\n    var defaults = {\n        /**\n         * The account id. In the Epicenter UI, this is the **Team ID** (for team projects) or **User ID** (for personal projects). Defaults to undefined. If left undefined, taken from the URL or session manager.\n         * @type {String}\n         */\n        account: undefined,\n\n        /**\n         * The project id. Defaults to undefined. If left undefined, taken from the URL or session manager.\n         * @type {String}\n         */\n        project: undefined,\n\n        /**\n         * Epicenter group name. Defaults to undefined. Note that this is the group *name*, not the group *id*. If left blank, taken from the session manager.\n         * @type {string}\n         */\n        groupName: undefined,\n\n        /**\n         * Options to pass on to the underlying transport layer. All jquery.ajax options at http://api.jquery.com/jQuery.ajax/ are available. Defaults to empty object.\n         * @type {object}\n         */\n        transport: {},\n    };\n    this.sessionManager = new SessionManager();\n    var serviceOptions = this.sessionManager.getMergedOptions(defaults, config);\n    var urlConfig = new ConfigService(serviceOptions).get('server');\n    if (serviceOptions.account) {\n        urlConfig.accountPath = serviceOptions.account;\n    }\n    if (serviceOptions.project) {\n        urlConfig.projectPath = serviceOptions.project;\n    }\n\n    var transportOptions = $.extend(true, {}, serviceOptions.transport, {\n        url: urlConfig.getAPIPath(apiEndpoint)\n    });\n\n    if (serviceOptions.token) {\n        transportOptions.headers = {\n            Authorization: 'Bearer ' + serviceOptions.token\n        };\n    }\n    var http = new TransportFactory(transportOptions, serviceOptions);\n\n\n    var getFinalParams = function (params) {\n        if (typeof params === 'object') {\n            return $.extend(true, {}, serviceOptions, params);\n        }\n        return serviceOptions;\n    };\n\n    var publicAPI = {\n        /**\n         * Marks an end user as online.\n         *\n         *\n         * **Example**\n         *\n         *     var pr = new F.service.Presence();\n         *     pr.markOnline('0000015a68d806bc09cd0a7d207f44ba5f74')\n         *          .then(function(presenceObj) {\n         *               console.log('user ', presenceObj.userId, \n         *                    ' now online, as of ', presenceObj.lastModified);\n         *          });\n         *\n         * **Return Value**\n         *\n         * Promise with presence information for user marked online.\n         *\n         * **Parameters**\n         *\n         * @param  {String} userId (optional) If not provided, taken from session cookie.\n         * @param  {Object} options Additional options to change the presence service defaults.\n         * @return {Promise} promise\n         */\n        markOnline: function (userId, options) {\n            options = options || {};\n            var isString = typeof userId === 'string';\n            var objParams = getFinalParams(userId);\n            if (!objParams.groupName && !options.groupName) {\n                throw new Error('No groupName specified.');\n            }\n            userId = isString ? userId : objParams.userId;\n            var groupName = options.groupName || objParams.groupName;\n            var httpOptions = $.extend(true, {}, serviceOptions, options,\n                { url: urlConfig.getAPIPath(apiEndpoint) + groupName + '/' + userId }\n            );\n            return http.post({ message: 'online' }, httpOptions);\n        },\n\n        /**\n         * Marks an end user as offline.\n         *\n         *\n         * **Example**\n         *\n         *     var pr = new F.service.Presence();\n         *     pr.markOffline('0000015a68d806bc09cd0a7d207f44ba5f74');\n         *\n         * **Return Value**\n         *\n         * Promise to remove presence record for end user.\n         *\n         * **Parameters**\n         *\n         * @param  {String} userId (optional) If not provided, taken from session cookie.\n         * @param  {Object} options Additional options to change the presence service defaults.\n         * @return {Promise} promise\n         */\n        markOffline: function (userId, options) {\n            options = options || {};\n            var isString = typeof userId === 'string';\n            var objParams = getFinalParams(userId);\n            if (!objParams.groupName && !options.groupName) {\n                throw new Error('No groupName specified.');\n            }\n            userId = isString ? userId : objParams.userId;\n            var groupName = options.groupName || objParams.groupName;\n            var httpOptions = $.extend(true, {}, serviceOptions, options,\n                { url: urlConfig.getAPIPath(apiEndpoint) + groupName + '/' + userId }\n            );\n            return http.delete({}, httpOptions);\n        },\n\n        /**\n         * Returns a list of all end users in this group that are currently online.\n         *\n         *\n         * **Example**\n         *\n         *     var pr = new F.service.Presence();\n         *     pr.getStatus('groupName').then(function(onlineUsers) {\n         *          for (var i=0; i < onlineUsers.length; i++) {\n         *               console.log('user ', onlineUsers[i].userId, \n         *                    ' is online as of ', onlineUsers[i].lastModified);\n         *          }\n         *     });\n         *\n         * **Return Value**\n         *\n         * Promise with response of online users\n         *\n         * **Parameters**\n         *\n         * @param  {String} groupName (optional) If not provided, taken from session cookie.\n         * @param  {Object} options Additional options to change the presence service defaults.\n         * @return {Promise} promise\n         */\n        getStatus: function (groupName, options) {\n            options = options || {};\n            var objParams = getFinalParams(groupName);\n            if (!groupName && !objParams.groupName) {\n                throw new Error('No groupName specified.');\n            }\n            groupName = groupName || objParams.groupName;\n            var httpOptions = $.extend(true, {}, serviceOptions, options,\n                { url: urlConfig.getAPIPath(apiEndpoint) + groupName }\n            );\n            return http.get({}, httpOptions);\n        },\n\n        /**\n         * End users are automatically marked online and offline in a \"presence\" channel that is specific to each group. Gets this channel (an instance of the [Channel Service](../channel-service/)) for the given group. (Note that this Channel Service instance is also available from the [Epicenter Channel Manager getPresenceChannel()](../epicenter-channel-manager/#getPresenceChannel).)\n         *\n         *\n         * **Example**\n         *\n         *     var pr = new F.service.Presence();\n         *     var cm = pr.getChannel('group1');\n         *     cm.publish('', 'a message to presence channel');\n         *\n         * **Return Value**\n         *\n         * Channel instance for Presence channel\n         *\n         * **Parameters**\n         *\n         * @param  {String} groupName (optional) If not provided, taken from session cookie.\n         * @param  {Object} options Additional options to change the presence service defaults\n         * @return {Channel} Channel instance\n         */\n        getChannel: function (groupName, options) {\n            options = options || {};\n            var isString = typeof groupName === 'string';\n            var objParams = getFinalParams(groupName);\n            if (!isString && !objParams.groupName) {\n                throw new Error('No groupName specified.');\n            }\n            groupName = isString ? groupName : objParams.groupName;\n            var cm = new ChannelManager(options);\n            return cm.getPresenceChannel(groupName);\n        }\n    };\n\n    $.extend(this, publicAPI);\n};\n","/**\n *\n * ## Run API Service\n *\n * The Run API Service allows you to perform common tasks around creating and updating runs, variables, and data.\n *\n * When building interfaces to show run one at a time (as for standard end users), typically you first instantiate a [Run Manager](../run-manager/) and then access the Run Service that is automatically part of the manager, rather than instantiating the Run Service directly. This is because the Run Manager (and associated [run strategies](../strategies/)) gives you control over run creation depending on run states.\n *\n * The Run API Service is useful for building an interface where you want to show data across multiple runs (this is easy using the `filter()` and `query()` methods). For instance, you would probably use a Run Service to build a page for a facilitator. This is because a facilitator typically wants to evaluate performance from multiple end users, each of whom have been working with their own run.\n *\n * To use the Run API Service, instantiate it by passing in:\n *\n * * `account`: Epicenter account id (**Team ID** for team projects, **User ID** for personal projects).\n * * `project`: Epicenter project id.\n *\n * If you know in advance that you would like to work with particular, existing run(s), you can optionally pass in:\n *\n * * `filter`: (Optional) Criteria by which to filter for existing runs. \n * * `id`: (Optional) The run id of an existing run. This is a convenience alias for using filter, in the case where you only want to work with one run.\n *\n * For example,\n *\n *       var rs = new F.service.Run({\n *            account: 'acme-simulations',\n *            project: 'supply-chain-game',\n *      });\n *      rs.create('supply_chain_game.py').then(function(run) {\n *             rs.do('someOperation');\n *      });\n *\n *\n * Additionally, all API calls take in an `options` object as the last parameter. The options can be used to extend/override the Run API Service defaults listed below. In particular, passing `{ id: 'a-run-id' }` in this `options` object allows you to make calls to an existing run.\n *\n * Note that in addition to the `account`, `project`, and `model`, the Run Service parameters optionally include a `server` object, whose `host` field contains the URI of the Forio server. This is automatically set, but you can pass it explicitly if desired. It is most commonly used for clarity when you are [hosting an Epicenter project on your own server](../../../how_to/self_hosting/).\n *\n *       var rm = new F.manager.RunManager({\n *           run: {\n *               account: 'acme-simulations',\n *               project: 'supply-chain-game',\n *               model: 'supply_chain_game.py',\n *               server: { host: 'api.forio.com' }\n *           }\n *       });\n *       rm.getRun()\n *           .then(function(run) {\n *               // the RunManager.run contains the instantiated Run Service,\n *               // so any Run Service method is valid here\n *               var rs = rm.run;\n *               rs.do('someOperation');\n *       })\n *\n */\n\n'use strict';\n\nvar ConfigService = require('./configuration-service');\nvar qutil = require('../util/query-util');\nvar rutil = require('../util/run-util');\nvar _pick = require('../util/object-util')._pick;\nvar TransportFactory = require('../transport/http-transport-factory');\nvar VariablesService = require('./variables-api-service');\nvar IntrospectionService = require('./introspection-api-service');\nvar SessionManager = require('../store/session-manager');\n\nmodule.exports = function (config) {\n    var defaults = {\n        /**\n         * For projects that require authentication, pass in the user access token (defaults to undefined). If the user is already logged in to Epicenter, the user access token is already set in a cookie and automatically loaded from there. (See [more background on access tokens](../../../project_access/)).\n         * @see [Authentication API Service](../auth-api-service/) for getting tokens.\n         * @type {String}\n         */\n        token: undefined,\n\n        /**\n         * The account id. In the Epicenter UI, this is the **Team ID** (for team projects) or **User ID** (for personal projects). Defaults to undefined. If left undefined, taken from the URL.\n         * @type {String}\n         */\n        account: undefined,\n\n        /**\n         * The project id. Defaults to undefined. If left undefined, taken from the URL.\n         * @type {String}\n         */\n        project: undefined,\n\n        /**\n         * Criteria by which to filter runs. Defaults to empty string.\n         * @type {String}\n         */\n        filter: '',\n\n        /**\n         * Convenience alias for filter. Pass in an existing run id to interact with a particular run.\n         * @type {String}\n         */\n        id: '',\n\n        /**\n         * Flag determines if `X-AutoRestore: true` header is sent to Epicenter, meaning runs are automatically pulled from the Epicenter backend database if not currently in memory on the Epicenter servers. Defaults to `true`.\n         * @type {boolean}\n         */\n        autoRestore: true,\n\n        /**\n         * Called when the call completes successfully. Defaults to `$.noop`.\n         * @type {function}\n         */\n        success: $.noop,\n\n        /**\n         * Called when the call fails. Defaults to `$.noop`.\n         * @type {function}\n         */\n        error: $.noop,\n\n        /**\n         * Options to pass on to the underlying transport layer. All jquery.ajax options at http://api.jquery.com/jQuery.ajax/ are available. Defaults to empty object.\n         * @type {Object}\n         */\n        transport: {}\n    };\n\n    this.sessionManager = new SessionManager();\n    var serviceOptions = this.sessionManager.getMergedOptions(defaults, config);\n    if (serviceOptions.id) {\n        serviceOptions.filter = serviceOptions.id;\n    }\n\n    function updateURLConfig(opts) {\n        var urlConfig = new ConfigService(opts).get('server');\n        if (opts.account) {\n            urlConfig.accountPath = opts.account;\n        }\n        if (opts.project) {\n            urlConfig.projectPath = opts.project;\n        }\n\n        urlConfig.filter = ';';\n        urlConfig.getFilterURL = function (filter) {\n            var url = urlConfig.getAPIPath('run');\n            var filterMatrix = qutil.toMatrixFormat(filter || opts.filter);\n\n            if (filterMatrix) {\n                url += filterMatrix + '/';\n            }\n            return url;\n        };\n\n        urlConfig.addAutoRestoreHeader = function (options) {\n            var filter = opts.filter;\n            // The semicolon separated filter is used when filter is an object\n            var isFilterRunId = filter && $.type(filter) === 'string';\n            if (opts.autoRestore && isFilterRunId) {\n                // By default autoreplay the run by sending this header to epicenter\n                // https://forio.com/epicenter/docs/public/rest_apis/aggregate_run_api/#retrieving\n                var autorestoreOpts = {\n                    headers: {\n                        'X-AutoRestore': true\n                    }\n                };\n                return $.extend(true, autorestoreOpts, options);\n            }\n\n            return options;\n        };\n        return urlConfig;\n    }\n\n    var http;\n    var httpOptions; //FIXME: Make this side-effect-less\n    function updateHTTPConfig(serviceOptions, urlConfig) {\n        httpOptions = $.extend(true, {}, serviceOptions.transport, {\n            url: urlConfig.getFilterURL\n        });\n\n        if (serviceOptions.token) {\n            httpOptions.headers = {\n                Authorization: 'Bearer ' + serviceOptions.token\n            };\n        }\n        http = new TransportFactory(httpOptions);\n        http.splitGet = rutil.splitGetFactory(httpOptions);\n    }\n\n    var urlConfig = updateURLConfig(serviceOptions); //making a function so #updateConfig can call this; change when refactored\n    updateHTTPConfig(serviceOptions, urlConfig);\n   \n\n    function setFilterOrThrowError(options) {\n        if (options.id) {\n            serviceOptions.filter = serviceOptions.id = options.id;\n        }\n        if (options.filter) {\n            serviceOptions.filter = serviceOptions.id = options.filter;\n        }\n        if (!serviceOptions.filter) {\n            throw new Error('No filter specified to apply operations against');\n        }\n    }\n\n    var publicAsyncAPI = {\n        urlConfig: urlConfig,\n\n        /**\n         * Create a new run.\n         *\n         * NOTE: Typically this is not used! Use `RunManager.getRun()` with a `strategy` of `reuse-never`, or use `RunManager.reset()`. See [Run Manager](../run-manager/) for more details.\n         *\n         *  **Example**\n         *\n         *      rs.create('hello_world.jl');\n         *\n         *  **Parameters**\n         * @param {String|Object} params If a string, the name of the primary [model file](../../../writing_your_model/). This is the one file in the project that explicitly exposes variables and methods, and it must be stored in the Model folder of your Epicenter project. If an object, may include `model`, `scope`, and `files`. (See the [Run Manager](../run_manager/) for more information on `scope` and `files`.)\n         * @param {Object} options (Optional) Overrides for configuration options.\n         * @return {Promise}\n         */\n        create: function (params, options) {\n            var createOptions = $.extend(true, {}, serviceOptions, options, { url: urlConfig.getAPIPath('run') });\n            var runApiParams = ['model', 'scope', 'files', 'ephemeral', 'cinFiles'];\n            if (typeof params === 'string') {\n                // this is just the model name\n                params = { model: params };\n            } else {\n                // whitelist the fields that we actually can send to the api\n                params = _pick(params, runApiParams);\n            }\n\n            var oldSuccess = createOptions.success;\n            createOptions.success = function (response) {\n                serviceOptions.filter = response.id; //all future chained calls to operate on this id\n                serviceOptions.id = response.id;\n                return oldSuccess.apply(this, arguments);\n            };\n\n            return http.post(params, createOptions);\n        },\n\n        /**\n         * Returns particular runs, based on conditions specified in the `qs` object.\n         *\n         * The elements of the `qs` object are ANDed together within a single call to `.query()`.\n         *\n         * **Example**\n         *\n         *      // returns runs with saved = true and variables.price > 1,\n         *      // where variables.price has been persisted (recorded)\n         *      // in the model.\n         *     rs.query({\n         *          'saved': 'true',\n         *          '.price': '>1'\n         *       },\n         *       {\n         *          startrecord: 2,\n         *          endrecord: 5\n         *       });\n         *\n         * **Parameters**\n         * @param {Object} qs Query object. Each key can be a property of the run or the name of variable that has been saved in the run (prefaced by `variables.`). Each value can be a literal value, or a comparison operator and value. (See [more on filtering](../../../rest_apis/aggregate_run_api/#filters) allowed in the underlying Run API.) Querying for variables is available for runs [in memory](../../../run_persistence/#runs-in-memory) and for runs [in the database](../../../run_persistence/#runs-in-memory) if the variables are persisted (e.g. that have been `record`ed in your model or marked for saving in your [model context file](../../../model_code/context/)).\n         * @param {Object} outputModifier (Optional) Available fields include: `startrecord`, `endrecord`, `sort`, and `direction` (`asc` or `desc`).\n         * @param {Object} options (Optional) Overrides for configuration options.\n         * @return {Promise}\n         */\n        query: function (qs, outputModifier, options) {\n            var httpOptions = $.extend(true, {}, serviceOptions, { url: urlConfig.getFilterURL(qs) }, options);\n            httpOptions = urlConfig.addAutoRestoreHeader(httpOptions);\n\n            return http.splitGet(outputModifier, httpOptions).then(function (r) {\n                return ($.isPlainObject(r) && Object.keys(r).length === 0) ? [] : r;\n            });\n        },\n\n        /**\n         * Returns particular runs, based on conditions specified in the `qs` object.\n         *\n         * Similar to `.query()`.\n         *\n         * **Parameters**\n         * @param {Object} filter Filter object. Each key can be a property of the run or the name of variable that has been saved in the run (prefaced by `variables.`). Each value can be a literal value, or a comparison operator and value. (See [more on filtering](../../../rest_apis/aggregate_run_api/#filters) allowed in the underlying Run API.) Filtering for variables is available for runs [in memory](../../../run_persistence/#runs-in-memory) and for runs [in the database](../../../run_persistence/#runs-in-memory) if the variables are persisted (e.g. that have been `record`ed in your model or marked for saving in your [model context file](../../../model_code/context/)).\n         * @param {Object} outputModifier (Optional) Available fields include: `startrecord`, `endrecord`, `sort`, and `direction` (`asc` or `desc`).\n         * @param {Object} options (Optional) Overrides for configuration options.\n         * @return {Promise}\n         */\n        filter: function (filter, outputModifier, options) {\n            if ($.isPlainObject(serviceOptions.filter)) {\n                $.extend(serviceOptions.filter, filter);\n            } else {\n                serviceOptions.filter = filter;\n            }\n            var httpOptions = $.extend(true, {}, serviceOptions, options);\n            httpOptions = urlConfig.addAutoRestoreHeader(httpOptions);\n            return http.splitGet(outputModifier, httpOptions).then(function (r) {\n                return ($.isPlainObject(r) && Object.keys(r).length === 0) ? [] : r;\n            });\n        },\n\n        /**\n         * Get data for a specific run. This includes standard run data such as the account, model, project, and created and last modified dates. To request specific model variables or run record variables, pass them as part of the `filters` parameter.\n         *\n         * Note that if the run is [in memory](../../../run_persistence/#runs-in-memory), any model variables are available; if the run is [in the database](../../../run_persistence/#runs-in-db), only model variables that have been persisted &mdash; that is, `record`ed or saved in your model &mdash; are available.\n         *\n         * **Example**\n         *\n         *     rs.load('bb589677-d476-4971-a68e-0c58d191e450', { include: ['.price', '.sales'] });\n         *\n         * **Parameters**\n         * @param {String} runID The run id.\n         * @param {Object} filters (Optional) Object containing filters and operation modifiers. Use key `include` to list model variables that you want to include in the response. Other available fields include: `startrecord`, `endrecord`, `sort`, and `direction` (`asc` or `desc`).\n         * @param {Object} options (Optional) Overrides for configuration options.\n         * @return {Promise}\n         */\n        load: function (runID, filters, options) {\n            if (runID) {\n                serviceOptions.filter = runID; //shouldn't be able to over-ride\n            }\n            var httpOptions = $.extend(true, {}, serviceOptions, options);\n            httpOptions = urlConfig.addAutoRestoreHeader(httpOptions);\n            return http.get(filters, httpOptions);\n        },\n\n        /**\n         * Removes specified runid from memory\n         *\n         * **Example**\n         *\n         *     rs.removeFromMemory('bb589677-d476-4971-a68e-0c58d191e450');\n         *\n         * See [details on run persistence](../../../run_persistence/#runs-in-memory)\n         * @param  {String} [runID]   id of run to remove\n         * @param  {Object} [filters] (Optional) Object containing filters and operation modifiers. Use key `include` to list model variables that you want to include in the response. Other available fields include: `startrecord`, `endrecord`, `sort`, and `direction` (`asc` or `desc`).\n         * @param  {Object} [options] (Optional) Overrides for configuration options.\n         * @return {Promise}\n         */\n        removeFromMemory: function (runID, filters, options) {\n            var httpOptions = $.extend(true, {}, serviceOptions, options);\n            if (runID) {\n                httpOptions.url = urlConfig.getAPIPath('run') + runID;\n            }\n            return http.delete({}, httpOptions);\n        },\n\n        /**\n         * Save attributes (data, model variables) of the run.\n         *\n         * **Examples**\n         *\n         *     // add 'completed' field to run record\n         *     rs.save({ completed: true });\n         *\n         *     // update 'saved' field of run record, and update values of model variables for this run\n         *     rs.save({ saved: true, variables: { a: 23, b: 23 } });\n         *\n         *     // update 'saved' field of run record for a particular run\n         *     rs.save({ saved: true }, { id: '0000015bf2a04995880df6b868d23eb3d229' });\n         *\n         * **Parameters**\n         * @param {Object} attributes The run data and variables to save.\n         * @param {Object} attributes.variables Model variables must be included in a `variables` field within the `attributes` object. (Otherwise they are treated as run data and added to the run record directly.)\n         * @param {Object} options (Optional) Overrides for configuration options.\n         * @return {Promise}\n         */\n        save: function (attributes, options) {\n            var httpOptions = $.extend(true, {}, serviceOptions, options);\n            setFilterOrThrowError(httpOptions);\n            return http.patch(attributes, httpOptions);\n        },\n\n        /**\n         * Call an operation from the model.\n         *\n         * Depending on the language in which you have written your model, the operation (function or method) may need to be exposed (e.g. `export` for a Julia model) in the model file in order to be called through the API. See [Writing your Model](../../../writing_your_model/)).\n         *\n         * The `params` argument is normally an array of arguments to the `operation`. In the special case where `operation` only takes one argument, you are not required to put that argument into an array.\n         *\n         * Note that you can combine the `operation` and `params` arguments into a single object if you prefer, as in the last example.\n         *\n         * **Examples**\n         *\n         *      // operation \"solve\" takes no arguments\n         *     rs.do('solve');\n         *      // operation \"echo\" takes one argument, a string\n         *     rs.do('echo', ['hello']);\n         *      // operation \"echo\" takes one argument, a string\n         *     rs.do('echo', 'hello');\n         *      // operation \"sumArray\" takes one argument, an array\n         *     rs.do('sumArray', [[4,2,1]]);\n         *      // operation \"add\" takes two arguments, both integers\n         *     rs.do({ name:'add', params:[2,4] });\n         *      // call operation \"solve\" on a different run \n         *     rs.do('solve', { id: '0000015bf2a04995880df6b868d23eb3d229' });\n         *\n         * **Parameters**\n         * @param {String} operation Name of operation.\n         * @param {Array} params (Optional) Any parameters the operation takes, passed as an array. In the special case where `operation` only takes one argument, you are not required to put that argument into an array, and can just pass it directly.\n         * @param {Object} options (Optional) Overrides for configuration options.\n         * @return {Promise}\n         */\n        do: function (operation, params, options) {\n            // console.log('do', operation, params);\n            var opsArgs;\n            var postOptions;\n            if (options) {\n                opsArgs = params;\n                postOptions = options;\n            } else if ($.isPlainObject(params)) {\n                opsArgs = null;\n                postOptions = params;\n            } else {\n                opsArgs = params;\n            }\n            var result = rutil.normalizeOperations(operation, opsArgs);\n            var httpOptions = $.extend(true, {}, serviceOptions, postOptions);\n\n            setFilterOrThrowError(httpOptions);\n\n            var prms = (result.args[0].length && (result.args[0] !== null && result.args[0] !== undefined)) ? result.args[0] : [];\n            return http.post({ arguments: prms }, $.extend(true, {}, httpOptions, {\n                url: urlConfig.getFilterURL() + 'operations/' + result.ops[0] + '/'\n            }));\n        },\n\n        /**\n         * Call several operations from the model, sequentially.\n         *\n         * Depending on the language in which you have written your model, the operation (function or method) may need to be exposed (e.g. `export` for a Julia model) in the model file in order to be called through the API. See [Writing your Model](../../../writing_your_model/)).\n         *\n         * **Examples**\n         *\n         *      // operations \"initialize\" and \"solve\" do not take any arguments\n         *     rs.serial(['initialize', 'solve']);\n         *      // operations \"init\" and \"reset\" take two arguments each\n         *     rs.serial([  { name: 'init', params: [1,2] },\n         *                  { name: 'reset', params: [2,3] }]);\n         *      // operation \"init\" takes two arguments,\n         *      // operation \"runmodel\" takes none\n         *     rs.serial([  { name: 'init', params: [1,2] },\n         *                  { name: 'runmodel', params: [] }]);\n         *\n         * **Parameters**\n         * @param {Array} operations If none of the operations take parameters, pass an array of the operation names (strings). If any of the operations do take parameters, pass an array of objects, each of which contains an operation name and its own (possibly empty) array of parameters.\n         * @param {*} params Parameters to pass to operations.\n         * @param {Object} options (Optional) Overrides for configuration options.\n         * @return {Promise} The parameter to the callback is an array. Each array element is an object containing the results of one operation.\n         */\n        serial: function (operations, params, options) {\n            var opParams = rutil.normalizeOperations(operations, params);\n            var ops = opParams.ops;\n            var args = opParams.args;\n            var me = this;\n\n            var $d = $.Deferred();\n            var postOptions = $.extend(true, {}, serviceOptions, options);\n\n            var responses = [];\n            var doSingleOp = function () {\n                var op = ops.shift();\n                var arg = args.shift();\n\n                me.do(op, arg, {\n                    success: function (result) {\n                        responses.push(result);\n                        if (ops.length) {\n                            doSingleOp();\n                        } else {\n                            $d.resolve(responses);\n                            postOptions.success(responses, me);\n                        }\n                    },\n                    error: function (err) {\n                        responses.push(err);\n                        $d.reject(responses);\n                        postOptions.error(responses, me);\n                    }\n                });\n            };\n\n            doSingleOp();\n\n            return $d.promise();\n        },\n\n        /**\n         * Call several operations from the model, executing them in parallel.\n         *\n         * Depending on the language in which you have written your model, the operation (function or method) may need to be exposed (e.g. `export` for a Julia model) in the model file in order to be called through the API. See [Writing your Model](../../../writing_your_model/)).\n         *\n         * **Example**\n         *\n         *      // operations \"solve\" and \"reset\" do not take any arguments\n         *     rs.parallel(['solve', 'reset']);\n         *      // operations \"add\" and \"subtract\" take two arguments each\n         *     rs.parallel([ { name: 'add', params: [1,2] },\n         *                   { name: 'subtract', params:[2,3] }]);\n         *      // operations \"add\" and \"subtract\" take two arguments each\n         *     rs.parallel({ add: [1,2], subtract: [2,4] });\n         *\n         * **Parameters**\n         * @param {Array|Object} operations If none of the operations take parameters, pass an array of the operation names (as strings). If any of the operations do take parameters, you have two options. You can pass an array of objects, each of which contains an operation name and its own (possibly empty) array of parameters. Alternatively, you can pass a single object with the operation name and a (possibly empty) array of parameters.\n         * @param {*} params Parameters to pass to operations.\n         * @param {Object} options (Optional) Overrides for configuration options.\n         * @return {Promise} The parameter to the callback is an array. Each array element is an object containing the results of one operation.\n         */\n        parallel: function (operations, params, options) {\n            var $d = $.Deferred();\n\n            var opParams = rutil.normalizeOperations(operations, params);\n            var ops = opParams.ops;\n            var args = opParams.args;\n            var postOptions = $.extend(true, {}, serviceOptions, options);\n\n            var queue = [];\n            for (var i = 0; i < ops.length; i++) {\n                queue.push(\n                    this.do(ops[i], args[i])\n                );\n            }\n\n            var me = this;\n            $.when.apply(this, queue)\n                .then(function () {\n                    var args = Array.prototype.slice.call(arguments);\n                    var actualResponse = args.map(function (a) {\n                        return a[0];\n                    });\n                    $d.resolve(actualResponse);\n                    postOptions.success(actualResponse, me);\n                })\n                .fail(function () {\n                    var args = Array.prototype.slice.call(arguments);\n                    var actualResponse = args.map(function (a) {\n                        return a[0];\n                    });\n                    $d.reject(actualResponse);\n                    postOptions.error(actualResponse, me);\n                });\n\n            return $d.promise();\n        },\n\n        /**\n         * Shortcut to using the [Introspection API Service](../introspection-api-service/). Allows you to view a list of the variables and operations in a model.\n         *\n         * **Example**\n         *\n         *     rs.introspect({ runID: 'cbf85437-b539-4977-a1fc-23515cf071bb' }).then(function (data) {\n         *          console.log(data.functions);\n         *          console.log(data.variables);\n         *     });\n         *\n         * **Parameters**\n         * @param  {Object} options Options can either be of the form `{ runID: <runid> }` or `{ model: <modelFileName> }`. Note that the `runID` is optional if the Run Service is already associated with a particular run (because `id` was passed in when the Run Service was initialized). If provided, the `runID` overrides the `id` currently associated with the Run Service.\n         * @param  {Object} introspectionConfig (Optional) Service options for Introspection Service\n         * @return {Promise}\n         */\n        introspect: function (options, introspectionConfig) {\n            var introspection = new IntrospectionService($.extend(true, {}, serviceOptions, introspectionConfig));\n            if (options) {\n                if (options.runID) {\n                    return introspection.byRunID(options.runID);\n                } else if (options.model) {\n                    return introspection.byModel(options.model);\n                }\n            } else if (serviceOptions.id) {\n                return introspection.byRunID(serviceOptions.id);\n            } else {\n                throw new Error('Please specify either the model or runid to introspect');\n            }\n        }\n    };\n\n    var publicSyncAPI = {\n        getCurrentConfig: function () {\n            return serviceOptions;\n        },\n        updateConfig: function (config) {\n            if (config && config.id) {\n                config.filter = config.id;\n            } else if (config && config.filter) {\n                config.id = config.filter;\n            }\n            serviceOptions = $.extend(true, {}, serviceOptions, config);\n            urlConfig = updateURLConfig(serviceOptions);\n            this.urlConfig = urlConfig;\n            updateHTTPConfig(serviceOptions, urlConfig);\n        },\n        /**\n          * Returns a Variables Service instance. Use the variables instance to load, save, and query for specific model variables. See the [Variable API Service](../variables-api-service/) for more information.\n          *\n          * **Example**\n          *\n          *      var vs = rs.variables();\n          *      vs.save({ sample_int: 4 });\n          *\n          * **Parameters**\n          * @param {Object} config (Optional) Overrides for configuration options.\n          * @return {Object} variablesService Instance\n          */\n        variables: function (config) {\n            var vs = new VariablesService($.extend(true, {}, serviceOptions, config, {\n                runService: this\n            }));\n            return vs;\n        }\n    };\n\n    $.extend(this, publicAsyncAPI);\n    $.extend(this, publicSyncAPI);\n};\n","'use strict';\n\nvar ConfigService = require('./configuration-service');\nvar SessionManager = require('../store/session-manager');\nvar objectAssign = require('object-assign');\n\nvar serviceUtils = {\n    /*\n    * Gets the default options for a api service.\n    * It will merge:\n    * - The Session options (Using the Session Manager)\n    * - The Authorization Header from the token option\n    * - The full url from the endpoint option\n    * With the supplied overrides and defaults\n    *\n    */\n    getDefaultOptions: function (defaults) {\n        var rest = Array.prototype.slice.call(arguments, 1);\n        var sessionManager = new SessionManager();\n        var serviceOptions = sessionManager.getMergedOptions.apply(sessionManager, [defaults].concat(rest));\n\n        serviceOptions.transport = objectAssign({}, serviceOptions.transport, {\n            url: this.getApiUrl(serviceOptions.apiEndpoint, serviceOptions)\n        });\n\n        if (serviceOptions.token) {\n            serviceOptions.transport.headers = {\n                Authorization: 'Bearer ' + serviceOptions.token\n            };\n        }\n        return serviceOptions;\n    },\n\n    getApiUrl: function (apiEndpoint, serviceOptions) {\n        var urlConfig = new ConfigService(serviceOptions).get('server');\n        return urlConfig.getAPIPath(apiEndpoint);\n    }\n};\n\nmodule.exports = serviceUtils;","'use strict';\n/**\n * ## State API Adapter\n *\n * The State API Adapter allows you to view the history of a run, and to replay or clone runs. \n *\n * The State API Adapter brings existing, persisted run data from the database back into memory, using the same run id (`replay`) or a new run id (`clone`). Runs must be in memory in order for you to update variables or call operations on them.\n *\n * Specifically, the State API Adapter works by \"re-running\" the run (user interactions) from the creation of the run up to the time it was last persisted in the database. This process uses the current version of the run's model. Therefore, if the model has changed since the original run was created, the retrieved run will use the new model — and may end up having different values or behavior as a result. Use with care!\n *\n * To use the State API Adapter, instantiate it and then call its methods:\n *\n *      var sa = new F.service.State();\n *      sa.replay({runId: '1842bb5c-83ad-4ba8-a955-bd13cc2fdb4f'});\n *\n * The constructor takes an optional `options` parameter in which you can specify the `account` and `project` if they are not already available in the current context.\n *\n */\n\nvar ConfigService = require('./configuration-service');\nvar TransportFactory = require('../transport/http-transport-factory');\nvar _pick = require('../util/object-util')._pick;\nvar SessionManager = require('../store/session-manager');\nvar apiEndpoint = 'model/state';\n\nmodule.exports = function (config) {\n\n    var defaults = {\n\n    };\n\n    this.sessionManager = new SessionManager();\n    var serviceOptions = this.sessionManager.getMergedOptions(defaults, config);\n    var urlConfig = new ConfigService(serviceOptions).get('server');\n\n    var transportOptions = $.extend(true, {}, serviceOptions.transport, {\n        url: urlConfig.getAPIPath(apiEndpoint)\n    });\n\n    if (serviceOptions.token) {\n        transportOptions.headers = {\n            Authorization: 'Bearer ' + serviceOptions.token\n        };\n    }\n\n    var http = new TransportFactory(transportOptions);\n    var parseRunIdOrError = function (params) {\n        if ($.isPlainObject(params) && params.runId) {\n            return params.runId;\n        } else {\n            throw new Error('Please pass in a run id');\n        }\n    };\n\n    var publicAPI = {\n\n        /**\n        * View the history of a run.\n        * \n        *  **Example**\n        *\n        *      var sa = new F.service.State();\n        *      sa.load('0000015a06bb58613b28b57365677ec89ec5').then(function(history) {\n        *            console.log('history = ', history);\n        *      });\n        *\n        *  **Parameters**\n        * @param {string} runId The id of the run.\n        * @param {object} options (Optional) Overrides for configuration options.\n        * @return {Promise}\n        */\n        load: function (runId, options) {\n            var httpParams = $.extend(true, {},\n                serviceOptions,\n                options,\n                { url: urlConfig.getAPIPath(apiEndpoint) + runId }\n            );\n            return http.get('', httpParams);\n        },\n\n        /**\n        * Replay a run. After this call, the run, with its original run id, is now available [in memory](../../../run_persistence/#runs-in-memory). (It continues to be persisted into the Epicenter database at regular intervals.)\n        *\n        *  **Example**\n        *\n        *      var sa = new F.service.State();\n        *      sa.replay({runId: '1842bb5c-83ad-4ba8-a955-bd13cc2fdb4f', stopBefore: 'calculateScore'});\n        *\n        *  **Parameters**\n        * @param {object} params Parameters object.\n        * @param {string} params.runId The id of the run to bring back to memory.\n        * @param {string} params.stopBefore (Optional) The run is advanced only up to the first occurrence of this method.\n        * @param {array} params.exclude (Optional) Array of methods to exclude when advancing the run.\n        * @param {object} options (Optional) Overrides for configuration options.\n        * @return {Promise}\n        */\n        replay: function (params, options) {\n            var runId = parseRunIdOrError(params);\n\n            var replayOptions = $.extend(true, {},\n                serviceOptions,\n                options,\n                { url: urlConfig.getAPIPath(apiEndpoint) + runId }\n            );\n\n            params = $.extend(true, { action: 'replay' }, _pick(params, ['stopBefore', 'exclude']));\n\n            return http.post(params, replayOptions);\n        },\n\n        /**\n        * Clone a given run and return a new run in the same state as the given run.\n        *\n        * The new run id is now available [in memory](../../../run_persistence/#runs-in-memory). The new run includes a copy of all of the data from the original run, EXCEPT:\n        *\n        * * The `saved` field in the new run record is not copied from the original run record. It defaults to `false`.\n        * * The `initialized` field in the new run record is not copied from the original run record. It defaults to `false` but may change to `true` as the new run is advanced. For example, if there has been a call to the `step` function (for Vensim models), the `initialized` field is set to `true`.\n        * * The `created` field in the new run record is the date and time at which the clone was created (not the time that the original run was created.)\n        *\n        * The original run remains only [in the database](../../../run_persistence/#runs-in-db).\n        *\n        *  **Example**\n        *\n        *      var sa = new F.service.State();\n        *      sa.clone({runId: '1842bb5c-83ad-4ba8-a955-bd13cc2fdb4f', stopBefore: 'calculateScore', exclude: ['interimCalculation'] });\n        *\n        *  **Parameters**\n        * @param {object} params Parameters object.\n        * @param {string} params.runId The id of the run to clone from memory.\n        * @param {string} params.stopBefore (Optional) The newly cloned run is advanced only up to the first occurrence of this method.\n        * @param {array} params.exclude (Optional) Array of methods to exclude when advancing the newly cloned run.\n        * @param {object} options (Optional) Overrides for configuration options.\n        * @return {Promise}\n        */\n        clone: function (params, options) {\n            var runId = parseRunIdOrError(params);\n\n            var replayOptions = $.extend(true, {},\n                serviceOptions,\n                options,\n                { url: urlConfig.getAPIPath(apiEndpoint) + runId }\n            );\n\n            params = $.extend(true, { action: 'clone' }, _pick(params, ['stopBefore', 'exclude']));\n\n            return http.post(params, replayOptions);\n        }\n    };\n\n    $.extend(this, publicAPI);\n};\n","'use strict';\n\nvar epiVersion = require('../api-version.json');\n\n//TODO: urlutils to get host, since no window on node\nvar defaults = {\n    host: window.location.host,\n    pathname: window.location.pathname\n};\n\nfunction getLocalHost(existingFn, host) {\n    var localHostFn;\n    if (existingFn !== undefined) {\n        if (!$.isFunction(existingFn)) {\n            localHostFn = function () { return existingFn; };\n        } else {\n            localHostFn = existingFn;\n        }\n    } else {\n        localHostFn = function () {\n            var isLocal = !host || //phantomjs\n                host === '127.0.0.1' || \n                host.indexOf('local.') === 0 || \n                host.indexOf('localhost') === 0;\n            return isLocal;\n        };\n    }\n    return localHostFn;\n}\n\nvar UrlConfigService = function (config) {\n    var envConf = UrlConfigService.defaults;\n\n    if (!config) {\n        config = {};\n    }\n    // console.log(this.defaults);\n    var overrides = $.extend({}, envConf, config);\n    var options = $.extend({}, defaults, overrides);\n\n    overrides.isLocalhost = options.isLocalhost = getLocalHost(options.isLocalhost, options.host);\n    \n    // console.log(isLocalhost(), '___________');\n    var actingHost = config && config.host;\n    if (!actingHost && options.isLocalhost()) {\n        actingHost = 'forio.com';\n    } else {\n        actingHost = options.host;\n    }\n\n    var API_PROTOCOL = 'https';\n    var HOST_API_MAPPING = {\n        'forio.com': 'api.forio.com',\n        'foriodev.com': 'api.epicenter.foriodev.com'\n    };\n\n    var publicExports = {\n        protocol: API_PROTOCOL,\n\n        api: '',\n\n        //TODO: this should really be called 'apihost', but can't because that would break too many things\n        host: (function () {\n            var apiHost = (HOST_API_MAPPING[actingHost]) ? HOST_API_MAPPING[actingHost] : actingHost;\n            // console.log(actingHost, config, apiHost);\n            return apiHost;\n        }()),\n\n        isCustomDomain: (function () {\n            var path = options.pathname.split('\\/');\n            var pathHasApp = path && path[1] === 'app';\n            return (!options.isLocalhost() && !pathHasApp);\n        }()),\n\n        appPath: (function () {\n            var path = options.pathname.split('\\/');\n\n            return path && path[1] || '';\n        }()),\n\n        accountPath: (function () {\n            var accnt = '';\n            var path = options.pathname.split('\\/');\n            if (path && path[1] === 'app') {\n                accnt = path[2];\n            }\n            return accnt;\n        }()),\n\n        projectPath: (function () {\n            var prj = '';\n            var path = options.pathname.split('\\/');\n            if (path && path[1] === 'app') {\n                prj = path[3]; //eslint-disable-line no-magic-numbers\n            }\n            return prj;\n        }()),\n\n        versionPath: (function () {\n            var version = epiVersion.version ? epiVersion.version + '/' : '';\n            return version;\n        }()),\n\n        getAPIPath: function (api) {\n            var PROJECT_APIS = ['run', 'data', 'file', 'presence'];\n            var apiMapping = {\n                channel: 'channel/subscribe'\n            };\n            var apiEndpoint = apiMapping[api] || api;\n            \n            if (apiEndpoint === 'config') {\n                var actualProtocol = window.location.protocol.replace(':', '');\n                var configProtocol = (options.isLocalhost()) ? this.protocol : actualProtocol;\n                return configProtocol + '://' + actingHost + '/epicenter/' + this.versionPath + 'config';\n            }\n            var apiPath = this.protocol + '://' + this.host + '/' + this.versionPath + apiEndpoint + '/';\n\n            if ($.inArray(apiEndpoint, PROJECT_APIS) !== -1) {\n                apiPath += this.accountPath + '/' + this.projectPath + '/';\n            }\n            return apiPath;\n        }\n    };\n\n\n    $.extend(publicExports, overrides);\n    return publicExports;\n};\n// This data can be set by external scripts, for loading from an env server for eg;\nUrlConfigService.defaults = {};\n\nmodule.exports = UrlConfigService;\n","'use strict';\n/**\n* ## User API Adapter\n*\n* The User API Adapter allows you to retrieve details about end users in your team (account). It is based on the querying capabilities of the underlying RESTful [User API](../../../rest_apis/user_management/user/).\n*\n* To use the User API Adapter, instantiate it and then call its methods.\n*\n*       var ua = new F.service.User({\n*           account: 'acme-simulations',\n*           token: 'user-or-project-access-token'\n*       });\n*       ua.getById('42836d4b-5b61-4fe4-80eb-3136e956ee5c');\n*       ua.get({ userName: 'jsmith' });\n*       ua.get({ id: ['42836d4b-5b61-4fe4-80eb-3136e956ee5c',\n*                   '4ea75631-4c8d-4872-9d80-b4600146478e'] });\n*\n* The constructor takes an optional `options` parameter in which you can specify the `account` and `token` if they are not already available in the current context.\n*/\n\nvar ConfigService = require('./configuration-service');\nvar TransportFactory = require('../transport/http-transport-factory');\nvar SessionManager = require('../store/session-manager');\nvar qutil = require('../util/query-util');\n\nmodule.exports = function (config) {\n    var defaults = {\n\n        /**\n         * The account id. In the Epicenter UI, this is the **Team ID** (for team projects) or **User ID** (for personal projects). Defaults to empty string.\n         * @type {String}\n         */\n        account: undefined,\n\n        /**\n         * The access token to use when searching for end users. (See [more background on access tokens](../../../project_access/)).\n         * @type {String}\n         */\n        token: undefined,\n\n        /**\n         * Options to pass on to the underlying transport layer. All jquery.ajax options at http://api.jquery.com/jQuery.ajax/ are available. Defaults to empty object.\n         * @type {Object}\n         */\n        transport: {}\n    };\n\n    this.sessionManager = new SessionManager();\n    var serviceOptions = this.sessionManager.getMergedOptions(defaults, config);\n    var urlConfig = new ConfigService(serviceOptions).get('server');\n    var transportOptions = $.extend(true, {}, serviceOptions.transport, {\n        url: urlConfig.getAPIPath('user')\n    });\n\n    if (serviceOptions.token) {\n        transportOptions.headers = {\n            Authorization: 'Bearer ' + serviceOptions.token\n        };\n    }\n\n    var http = new TransportFactory(transportOptions);\n\n    var publicAPI = {\n\n        /**\n        * Retrieve details about particular end users in your team, based on user name or user id.\n        *\n        * **Example**\n        *\n        *       var ua = new F.service.User({\n        *           account: 'acme-simulations',\n        *           token: 'user-or-project-access-token'\n        *       });\n        *       ua.get({ userName: 'jsmith' });\n        *       ua.get({ id: ['42836d4b-5b61-4fe4-80eb-3136e956ee5c',\n        *                   '4ea75631-4c8d-4872-9d80-b4600146478e'] });\n        *\n        * **Parameters**\n        * @param {object} filter Object with field `userName` and value of the username. Alternatively, object with field `id` and value of an array of user ids.\n        * @param {object} options (Optional) Overrides for configuration options.\n        * @return {Promise}\n        */\n\n        get: function (filter, options) {\n            options = options || {};\n            filter = filter || {};\n\n            var getOptions = $.extend(true, {},\n                serviceOptions,\n                options\n            );\n\n            var toQFilter = function (filter) {\n                var res = {};\n\n                // API only supports filtering by username for now\n                if (filter.userName) {\n                    res.q = filter.userName;\n                }\n\n                return res;\n            };\n\n            var toIdFilters = function (id) {\n                if (!id) {\n                    return '';\n                }\n\n                id = $.isArray(id) ? id : [id];\n                return 'id=' + id.join('&id=');\n            };\n\n            var getFilters = [\n                'account=' + getOptions.account,\n                toIdFilters(filter.id),\n                qutil.toQueryFormat(toQFilter(filter))\n            ].join('&');\n\n            // special case for queries with large number of ids\n            // make it as a post with GET semantics\n            var threshold = 30;\n            if (filter.id && $.isArray(filter.id) && filter.id.length >= threshold) {\n                getOptions.url = urlConfig.getAPIPath('user') + '?_method=GET';\n                return http.post({ id: filter.id }, getOptions);\n            } else {\n                return http.get(getFilters, getOptions);\n            }\n        },\n\n        /**\n        * Retrieve details about a single end user in your team, based on user id.\n        *\n        * **Example**\n        *\n        *       var ua = new F.service.User({\n        *           account: 'acme-simulations',\n        *           token: 'user-or-project-access-token'\n        *       });\n        *       ua.getById('42836d4b-5b61-4fe4-80eb-3136e956ee5c');\n        *\n        * **Parameters**\n        * @param {string} userId The user id for the end user in your team.\n        * @param {object} options (Optional) Overrides for configuration options.\n        * @return {Promise}\n        */\n\n        getById: function (userId, options) {\n            return publicAPI.get({ id: userId }, options);\n        }\n    };\n\n    $.extend(this, publicAPI);\n};\n\n\n","/**\n *\n * ## Variables API Service\n *\n * Used in conjunction with the [Run API Service](../run-api-service/) to read, write, and search for specific model variables.\n *\n *     var rm = new F.manager.RunManager({\n *           run: {\n *               account: 'acme-simulations',\n *               project: 'supply-chain-game',\n *               model: 'supply-chain-model.jl'\n *           }\n *      });\n *     rm.getRun()\n *       .then(function() {\n *          var vs = rm.run.variables();\n *          vs.save({sample_int: 4});\n *        });\n *\n */\n\n\n 'use strict';\n\n var TransportFactory = require('../transport/http-transport-factory');\n var rutil = require('../util/run-util');\n\n module.exports = function (config) {\n     var defaults = {\n        /**\n         * The runs object to which the variable filters apply. Defaults to null.\n         * @type {runService}\n         */\n         runService: null\n     };\n     var serviceOptions = $.extend({}, defaults, config);\n\n     var getURL = function () {\n        //TODO: Replace with getCurrentconfig instead?\n         return serviceOptions.runService.urlConfig.getFilterURL() + 'variables/';\n     };\n\n     var addAutoRestoreHeader = function (options) {\n         return serviceOptions.runService.urlConfig.addAutoRestoreHeader(options);\n     };\n\n     var httpOptions = {\n         url: getURL\n     };\n     if (serviceOptions.token) {\n         httpOptions.headers = {\n             Authorization: 'Bearer ' + serviceOptions.token\n         };\n     }\n     var http = new TransportFactory(httpOptions);\n     http.splitGet = rutil.splitGetFactory(httpOptions);\n\n     var publicAPI = {\n\n        /**\n         * Get values for a variable.\n         *\n         * **Example**\n         *\n         *      vs.load('sample_int')\n         *          .then(function(val){\n         *              // val contains the value of sample_int\n         *          });\n         *\n         * **Parameters**\n         * @param {String} variable Name of variable to load.\n         * @param {Object} outputModifier (Optional) Available fields include: `startrecord`, `endrecord`, `sort`, and `direction` (`asc` or `desc`).\n         * @param {Object} options (Optional) Overrides for configuration options.\n         * @return {Promise}\n         */\n         load: function (variable, outputModifier, options) {\n             var httpOptions = $.extend(true, {}, serviceOptions, options);\n             httpOptions = addAutoRestoreHeader(httpOptions);\n             return http.get(outputModifier, $.extend({}, httpOptions, {\n                 url: getURL() + variable + '/'\n             }));\n         },\n\n        /**\n         * Returns particular variables, based on conditions specified in the `query` object.\n         *\n         * **Example**\n         *\n         *      vs.query(['price', 'sales'])\n         *          .then(function(val) {\n         *              // val is an object with the values of the requested variables: val.price, val.sales\n         *          });\n         *\n         *      vs.query({ include:['price', 'sales'] });\n         *\n         * **Parameters**\n         * @param {Object|Array} query The names of the variables requested.\n         * @param {Object} outputModifier (Optional) Available fields include: `startrecord`, `endrecord`, `sort`, and `direction` (`asc` or `desc`).\n         * @param {Object} options (Optional) Overrides for configuration options.\n         * @return {Promise}\n         */\n         query: function (query, outputModifier, options) {\n            //Query and outputModifier are both querystrings in the url; only calling them out separately here to be consistent with the other calls\n             var httpOptions = $.extend(true, {}, serviceOptions, options);\n             httpOptions = addAutoRestoreHeader(httpOptions);\n\n             if ($.isArray(query)) {\n                 query = { include: query };\n             }\n             $.extend(query, outputModifier);\n             return http.splitGet(query, httpOptions);\n         },\n\n        /**\n         * Save values to model variables. Overwrites existing values. Note that you can only update model variables if the run is [in memory](../../../run_persistence/#runs-in-memory). (An alternate way to update model variables is to call a method from the model and make sure that the method persists the variables. See `do`, `serial`, and `parallel` in the [Run API Service](../run-api-service/) for calling methods from the model.)\n         *\n         * **Example**\n         *\n         *      vs.save('price', 4);\n         *      vs.save({ price: 4, quantity: 5, products: [2,3,4] });\n         *\n         * **Parameters**\n         * @param {Object|String} variable An object composed of the model variables and the values to save. Alternatively, a string with the name of the variable.\n         * @param {Object} val (Optional) If passing a string for `variable`, use this argument for the value to save.\n         * @param {Object} options (Optional) Overrides for configuration options.\n         * @return {Promise}\n         */\n         save: function (variable, val, options) {\n             var attrs;\n             if (typeof variable === 'object') {\n                 attrs = variable;\n                 options = val;\n             } else {\n                 (attrs = {})[variable] = val;\n             }\n             var httpOptions = $.extend(true, {}, serviceOptions, options);\n\n             return http.patch.call(this, attrs, httpOptions);\n         }\n\n        // Not Available until underlying API supports PUT. Otherwise save would be PUT and merge would be PATCH\n        // *\n        //  * Save values to the api. Merges arrays, but otherwise same as save\n        //  * @param {Object|String} variable Object with attributes, or string key\n        //  * @param {Object} val Optional if prev parameter was a string, set value here\n        //  * @param {Object} options Overrides for configuration options\n        //  *\n        //  * @example\n        //  *     vs.merge({ price: 4, quantity: 5, products: [2,3,4] })\n        //  *     vs.merge('price', 4);\n\n        // merge: function (variable, val, options) {\n        //     var attrs;\n        //     if (typeof variable === 'object') {\n        //       attrs = variable;\n        //       options = val;\n        //     } else {\n        //       (attrs = {})[variable] = val;\n        //     }\n        //     var httpOptions = $.extend(true, {}, serviceOptions, options);\n\n        //     return http.patch.call(this, attrs, httpOptions);\n        // }\n     };\n     $.extend(this, publicAPI);\n };\n","/**\n * ## World API Adapter\n *\n * A [run](../../../glossary/#run) is a collection of end user interactions with a project and its model -- including setting variables, making decisions, and calling operations. For building multiplayer simulations you typically want multiple end users to share the same set of interactions, and work within a common state. Epicenter allows you to create \"worlds\" to handle such cases. Only [team projects](../../../glossary/#team) can be multiplayer.\n *\n * The World API Adapter allows you to create, access, and manipulate multiplayer worlds within your Epicenter project. You can use this to add and remove end users from the world, and to create, access, and remove their runs. Because of this, typically the World Adapter is used for facilitator pages in your project. (The related [World Manager](../world-manager/) provides an easy way to access runs and worlds for particular end users, so is typically used in pages that end users will interact with.)\n *\n * As with all the other [API Adapters](../../), all methods take in an \"options\" object as the last parameter. The options can be used to extend/override the World API Service defaults.\n *\n * To use the World Adapter, instantiate it and then access the methods provided. Instantiating requires the account id (**Team ID** in the Epicenter user interface), project id (**Project ID**), and group (**Group Name**).\n *\n *       var wa = new F.service.World({\n *          account: 'acme-simulations',\n *          project: 'supply-chain-game',\n *          group: 'team1' });\n *       wa.create()\n *          .then(function(world) {\n *              // call methods, e.g. wa.addUsers()\n *          });\n */\n\n'use strict';\n\nvar ConfigService = require('./configuration-service');\n// var qutil = require('../util/query-util');\nvar TransportFactory = require('../transport/http-transport-factory');\nvar SessionManager = require('../store/session-manager');\nvar _pick = require('../util/object-util')._pick;\n\nvar apiBase = 'multiplayer/';\nvar assignmentEndpoint = apiBase + 'assign';\nvar apiEndpoint = apiBase + 'world';\nvar projectEndpoint = apiBase + 'project';\n\nmodule.exports = function (config) {\n    var defaults = {\n        /**\n         * For projects that require authentication, pass in the user access token (defaults to empty string). If the user is already logged in to Epicenter, the user access token is already set in a cookie and automatically loaded from there. (See [more background on access tokens](../../../project_access/)).\n         * @see [Authentication API Service](../auth-api-service/) for getting tokens.\n         * @type {String}\n         */\n        token: undefined,\n\n        /**\n         * The project id. If left undefined, taken from the URL.\n         * @type {String}\n         */\n        project: undefined,\n\n        /**\n         * The account id. In the Epicenter UI, this is the **Team ID** (for team projects). If left undefined, taken from the URL.\n         * @type {String}\n         */\n        account: undefined,\n\n        /**\n         * The group name. Defaults to undefined.\n         * @type {String}\n         */\n        group: undefined,\n\n        /**\n         * The model file to use to create runs in this world. Defaults to undefined.\n         * @type {String}\n         */\n        model: undefined,\n\n        /**\n         * Criteria by which to filter world. Currently only supports world-ids as filters.\n         * @type {String}\n         */\n        filter: '',\n\n        /**\n         * Convenience alias for filter\n         * @type {String}\n         */\n        id: '',\n\n        /**\n         * Options to pass on to the underlying transport layer. All jquery.ajax options at http://api.jquery.com/jQuery.ajax/ are available. Defaults to empty object.\n         * @type {Object}\n         */\n        transport: {},\n\n        /**\n         * Called when the call completes successfully. Defaults to `$.noop`.\n         * @type {function}\n         */\n        success: $.noop,\n\n        /**\n         * Called when the call fails. Defaults to `$.noop`.\n         * @type {function}\n         */\n        error: $.noop\n    };\n\n    this.sessionManager = new SessionManager();\n    var serviceOptions = this.sessionManager.getMergedOptions(defaults, config);\n    if (serviceOptions.id) {\n        serviceOptions.filter = serviceOptions.id;\n    }\n\n    var urlConfig = new ConfigService(serviceOptions).get('server');\n\n    if (!serviceOptions.account) {\n        serviceOptions.account = urlConfig.accountPath;\n    }\n\n    if (!serviceOptions.project) {\n        serviceOptions.project = urlConfig.projectPath;\n    }\n\n    var transportOptions = $.extend(true, {}, serviceOptions.transport, {\n        url: urlConfig.getAPIPath(apiEndpoint)\n    });\n\n    if (serviceOptions.token) {\n        transportOptions.headers = {\n            Authorization: 'Bearer ' + serviceOptions.token\n        };\n    }\n\n    var http = new TransportFactory(transportOptions);\n\n    var setIdFilterOrThrowError = function (options) {\n        if (options.id) {\n            serviceOptions.filter = options.id;\n        }\n        if (options.filter) {\n            serviceOptions.filter = options.filter;\n        }\n        if (!serviceOptions.filter) {\n            throw new Error('No world id specified to apply operations against. This could happen if the user is not assigned to a world and is trying to work with runs from that world.');\n        }\n    };\n\n    var validateModelOrThrowError = function (options) {\n        if (!options.model) {\n            throw new Error('No model specified to get the current run');\n        }\n    };\n\n    var publicAPI = {\n\n        /**\n        * Creates a new World.\n        *\n        * Using this method is rare. It is more common to create worlds automatically while you `autoAssign()` end users to worlds. (In this case, configuration data for the world, such as the roles, are read from the project-level world configuration information, for example by `getProjectSettings()`.)\n        *\n        *  **Example**\n        *\n        *      var wa = new F.service.World({\n        *           account: 'acme-simulations',\n        *           project: 'supply-chain-game',\n        *           group: 'team1' });\n        *      wa.create({\n        *           roles: ['VP Marketing', 'VP Sales', 'VP Engineering']\n        *       });\n        *\n        *  **Parameters**\n        * @param {object} params Parameters to create the world.\n        * @param {string} params.group (Optional) The **Group Name** to create this world under. Only end users in this group are eligible to join the world. Optional here; required when instantiating the service (`new F.service.World()`).\n        * @param {object} params.roles (Optional) The list of roles (strings) for this world. Some worlds have specific roles that **must** be filled by end users. Listing the roles allows you to autoassign users to worlds and ensure that all roles are filled in each world.\n        * @param {object} params.optionalRoles (Optional) The list of optional roles (strings) for this world. Some worlds have specific roles that **may** be filled by end users. Listing the optional roles as part of the world object allows you to autoassign users to worlds and ensure that all roles are filled in each world.\n        * @param {integer} params.minUsers (Optional) The minimum number of users for the world. Including this number allows you to autoassign end users to worlds and ensure that the correct number of users are in each world.\n        * @param {object} options (Optional) Options object to override global options.\n        * @return {Promise}\n        */\n        create: function (params, options) {\n            var createOptions = $.extend(true, {}, serviceOptions, options, { url: urlConfig.getAPIPath(apiEndpoint) });\n            var worldApiParams = ['scope', 'files', 'roles', 'optionalRoles', 'minUsers', 'group', 'name'];\n            var validParams = _pick(serviceOptions, ['account', 'project', 'group']);\n            // whitelist the fields that we actually can send to the api\n            params = _pick(params, worldApiParams);\n\n            // account and project go in the body, not in the url\n            params = $.extend({}, validParams, params);\n\n            var oldSuccess = createOptions.success;\n            createOptions.success = function (response) {\n                serviceOptions.filter = response.id; //all future chained calls to operate on this id\n                return oldSuccess.apply(this, arguments);\n            };\n\n            return http.post(params, createOptions);\n        },\n\n        /**\n        * Updates a World, for example to replace the roles in the world.\n        *\n        * Typically, you complete world configuration at the project level, rather than at the world level. For example, each world in your project probably has the same roles for end users. And your project is probably either configured so that all end users share the same world (and run), or smaller sets of end users share worlds — but not both. However, this method is available if you need to update the configuration of a particular world.\n        *\n        *  **Example**\n        *\n        *      var wa = new F.service.World({\n        *           account: 'acme-simulations',\n        *           project: 'supply-chain-game',\n        *           group: 'team1' });\n        *      wa.create()\n        *           .then(function(world) {\n        *               wa.update({ roles: ['VP Marketing', 'VP Sales', 'VP Engineering'] });\n        *           });\n        *\n        *  **Parameters**\n        * @param {object} params Parameters to update the world.\n        * @param {string} params.name A string identifier for the linked end users, for example, \"name\": \"Our Team\".\n        * @param {object} params.roles (Optional) The list of roles (strings) for this world. Some worlds have specific roles that **must** be filled by end users. Listing the roles allows you to autoassign users to worlds and ensure that all roles are filled in each world.\n        * @param {object} params.optionalRoles (Optional) The list of optional roles (strings) for this world. Some worlds have specific roles that **may** be filled by end users. Listing the optional roles as part of the world object allows you to autoassign users to worlds and ensure that all roles are filled in each world.\n        * @param {integer} params.minUsers (Optional) The minimum number of users for the world. Including this number allows you to autoassign end users to worlds and ensure that the correct number of users are in each world.\n        * @param {object} options (Optional) Options object to override global options.\n        * @return {Promise}\n        */\n        update: function (params, options) {\n            var whitelist = ['roles', 'optionalRoles', 'minUsers'];\n            options = options || {};\n            setIdFilterOrThrowError(options);\n\n            var updateOptions = $.extend(true, {},\n                serviceOptions,\n                options,\n                { url: urlConfig.getAPIPath(apiEndpoint) + serviceOptions.filter }\n            );\n\n            params = _pick(params || {}, whitelist);\n\n            return http.patch(params, updateOptions);\n        },\n\n        /**\n        * Deletes an existing world.\n        *\n        * This function optionally takes one argument. If the argument is a string, it is the id of the world to delete. If the argument is an object, it is the override for global options.\n        *\n        *  **Example**\n        *\n        *      var wa = new F.service.World({\n        *           account: 'acme-simulations',\n        *           project: 'supply-chain-game',\n        *           group: 'team1' });\n        *      wa.create()\n        *           .then(function(world) {\n        *               wa.delete();\n        *           });\n        *\n        *  **Parameters**\n        * @param {String|Object} options (Optional) The id of the world to delete, or options object to override global options.\n        * @return {Promise}\n        */\n        delete: function (options) {\n            options = (options && (typeof options === 'string')) ? { filter: options } : {};\n            setIdFilterOrThrowError(options);\n\n            var deleteOptions = $.extend(true, {},\n                serviceOptions,\n                options,\n                { url: urlConfig.getAPIPath(apiEndpoint) + serviceOptions.filter }\n            );\n\n            return http.delete(null, deleteOptions);\n        },\n\n        /**\n        * Updates the configuration for the current instance of the World API Adapter (including all subsequent function calls, until the configuration is updated again).\n        *\n        * **Example**\n        *\n        *      var wa = new F.service.World({...}).updateConfig({ filter: '123' }).addUser({ userId: '123' });\n        *\n        * **Parameters**\n        * @param {object} config The configuration object to use in updating existing configuration.\n        * @return {Object} reference to current instance\n        */\n        updateConfig: function (config) {\n            $.extend(serviceOptions, config);\n\n            return this;\n        },\n\n        /**\n        * Lists all worlds for a given account, project, and group. All three are required, and if not specified as parameters, are read from the service.\n        *\n        *  **Example**\n        *\n        *      var wa = new F.service.World({\n        *           account: 'acme-simulations',\n        *           project: 'supply-chain-game',\n        *           group: 'team1' });\n        *      wa.create()\n        *           .then(function(world) {\n        *               // lists all worlds in group \"team1\"\n        *               wa.list();\n        *\n        *               // lists all worlds in group \"other-group-name\"\n        *               wa.list({ group: 'other-group-name' });\n        *           });\n        *\n        *  **Parameters**\n        * @param {object} options (Optional) Options object to override global options.\n        * @return {Promise}\n        */\n        list: function (options) {\n            options = options || {};\n\n            var getOptions = $.extend(true, {},\n                serviceOptions,\n                options,\n                { url: urlConfig.getAPIPath(apiEndpoint) }\n            );\n\n            var filters = _pick(getOptions, ['account', 'project', 'group']);\n\n            return http.get(filters, getOptions);\n        },\n\n        /**\n        * Gets all worlds that an end user belongs to for a given account (team), project, and group.\n        *\n        *  **Example**\n        *\n        *      var wa = new F.service.World({\n        *           account: 'acme-simulations',\n        *           project: 'supply-chain-game',\n        *           group: 'team1' });\n        *      wa.create()\n        *           .then(function(world) {\n        *               wa.getWorldsForUser('b1c19dda-2d2e-4777-ad5d-3929f17e86d3')\n        *           });\n        *\n        * ** Parameters **\n        * @param {string} userId The `userId` of the user whose worlds are being retrieved.\n        * @param {object} options (Optional) Options object to override global options.\n        * @return {Promise}\n        */\n        getWorldsForUser: function (userId, options) {\n            options = options || {};\n\n            var getOptions = $.extend(true, {},\n                serviceOptions,\n                options,\n                { url: urlConfig.getAPIPath(apiEndpoint) }\n            );\n\n            var filters = $.extend(\n                _pick(getOptions, ['account', 'project', 'group']),\n                { userId: userId }\n            );\n\n            return http.get(filters, getOptions);\n        },\n\n        /**\n         * Load information for a specific world. All further calls to the world service will use the id provided.\n         *\n         * **Parameters**\n         * @param {String} worldId The id of the world to load.\n         * @param {Object} options (Optional) Options object to override global options.\n         * @return {Promise}\n         */\n        load: function (worldId, options) {\n            if (worldId) {\n                serviceOptions.filter = worldId;\n            }\n            if (!serviceOptions.filter) {\n                throw new Error('Please provide a worldid to load');\n            }\n            var httpOptions = $.extend(true, {}, serviceOptions, options, { url: urlConfig.getAPIPath(apiEndpoint) + serviceOptions.filter + '/' });\n            return http.get('', httpOptions);\n        },\n\n        /**\n        * Adds an end user or list of end users to a given world. The end user must be a member of the `group` that is associated with this world.\n        *\n        *  **Example**\n        *\n        *      var wa = new F.service.World({\n        *           account: 'acme-simulations',\n        *           project: 'supply-chain-game',\n        *           group: 'team1' });\n        *      wa.create()\n        *           .then(function(world) {\n        *               // add one user\n        *               wa.addUsers('b1c19dda-2d2e-4777-ad5d-3929f17e86d3');\n        *               wa.addUsers(['b1c19dda-2d2e-4777-ad5d-3929f17e86d3']);\n        *               wa.addUsers({ userId: 'b1c19dda-2d2e-4777-ad5d-3929f17e86d3', role: 'VP Sales' });\n        *\n        *               // add several users\n        *               wa.addUsers([\n        *                   { userId: 'a6fe0c1e-f4b8-4f01-9f5f-01ccf4c2ed44',\n        *                     role: 'VP Marketing' },\n        *                   { userId: '8f2604cf-96cd-449f-82fa-e331530734ee',\n        *                     role: 'VP Engineering' }\n        *               ]);\n        *\n        *               // add one user to a specific world\n        *               wa.addUsers('b1c19dda-2d2e-4777-ad5d-3929f17e86d3', world.id);\n        *               wa.addUsers('b1c19dda-2d2e-4777-ad5d-3929f17e86d3', { filter: world.id });\n        *           });\n        *\n        * ** Parameters **\n        * @param {string|object|array} users User id, array of user ids, object, or array of objects of the users to add to this world.\n        * @param {string} users.role The `role` the user should have in the world. It is up to the caller to ensure, if needed, that the `role` passed in is one of the `roles` or `optionalRoles` of this world.\n        * @param {string} worldId The world to which the users should be added. If not specified, the filter parameter of the `options` object is used.\n        * @param {object} options (Optional) Options object to override global options.\n        * @return {Promise}\n        */\n        addUsers: function (users, worldId, options) {\n\n            if (!users) {\n                throw new Error('Please provide a list of users to add to the world');\n            }\n\n            // normalize the list of users to an array of user objects\n            users = $.map([].concat(users), function (u) {\n                var isObject = $.isPlainObject(u);\n\n                if (typeof u !== 'string' && !isObject) {\n                    throw new Error('Some of the users in the list are not in the valid format: ' + u);\n                }\n\n                return isObject ? u : { userId: u };\n            });\n\n            // check if options were passed as the second parameter\n            if ($.isPlainObject(worldId) && !options) {\n                options = worldId;\n                worldId = null;\n            }\n\n            options = options || {};\n\n            // we must have options by now\n            if (typeof worldId === 'string') {\n                options.filter = worldId;\n            }\n\n            setIdFilterOrThrowError(options);\n\n            var updateOptions = $.extend(true, {},\n                serviceOptions,\n                options,\n                { url: urlConfig.getAPIPath(apiEndpoint) + serviceOptions.filter + '/users' }\n            );\n\n            return http.post(users, updateOptions);\n        },\n\n        /**\n        * Updates the role of an end user in a given world. (You can only update one end user at a time.)\n        *\n        * **Example**\n        *\n        *      var wa = new F.service.World({\n        *           account: 'acme-simulations',\n        *           project: 'supply-chain-game',\n        *           group: 'team1' });\n        *\n        *      wa.create().then(function(world) {\n        *           wa.addUsers('b1c19dda-2d2e-4777-ad5d-3929f17e86d3');\n        *           wa.updateUser({ userId: 'b1c19dda-2d2e-4777-ad5d-3929f17e86d3', role: 'leader' });\n        *      });\n        *\n        * **Parameters**\n        * @param {object} user User object with `userId` and the new `role`.\n        * @param {object} options (Optional) Options object to override global options.\n        * @return {Promise}\n        */\n        updateUser: function (user, options) {\n            options = options || {};\n\n            if (!user || !user.userId) {\n                throw new Error('You need to pass a userId to update from the world');\n            }\n\n            setIdFilterOrThrowError(options);\n\n            var patchOptions = $.extend(true, {},\n                serviceOptions,\n                options,\n                { url: urlConfig.getAPIPath(apiEndpoint) + serviceOptions.filter + '/users/' + user.userId }\n            );\n\n            return http.patch(_pick(user, 'role'), patchOptions);\n        },\n\n        /**\n        * Removes an end user from a given world.\n        *\n        *  **Example**\n        *\n        *      var wa = new F.service.World({\n        *           account: 'acme-simulations',\n        *           project: 'supply-chain-game',\n        *           group: 'team1' });\n        *      wa.create()\n        *           .then(function(world) {\n        *               wa.addUsers(['a6fe0c1e-f4b8-4f01-9f5f-01ccf4c2ed44', '8f2604cf-96cd-449f-82fa-e331530734ee']);\n        *               wa.removeUser('a6fe0c1e-f4b8-4f01-9f5f-01ccf4c2ed44');\n        *               wa.removeUser({ userId: '8f2604cf-96cd-449f-82fa-e331530734ee' });\n        *           });\n        *\n        * ** Parameters **\n        * @param {object|string} user The `userId` of the user to remove from the world, or an object containing the `userId` field.\n        * @param {object} options (Optional) Options object to override global options.\n        * @return {Promise}\n        */\n        removeUser: function (user, options) {\n            options = options || {};\n\n            if (typeof user === 'string') {\n                user = { userId: user };\n            }\n\n            if (!user.userId) {\n                throw new Error('You need to pass a userId to remove from the world');\n            }\n\n            setIdFilterOrThrowError(options);\n\n            var getOptions = $.extend(true, {},\n                serviceOptions,\n                options,\n                { url: urlConfig.getAPIPath(apiEndpoint) + serviceOptions.filter + '/users/' + user.userId }\n            );\n\n            return http.delete(null, getOptions);\n        },\n\n        /**\n        * Gets the run id of current run for the given world. If the world does not have a run, creates a new one and returns the run id.\n        *\n        * Remember that a [run](../../glossary/#run) is a collection of interactions with a project and its model. In the case of multiplayer projects, the run is shared by all end users in the world.\n        *\n        *  **Example**\n        *\n        *      var wa = new F.service.World({\n        *           account: 'acme-simulations',\n        *           project: 'supply-chain-game',\n        *           group: 'team1' });\n        *      wa.create()\n        *           .then(function(world) {\n        *               wa.getCurrentRunId({ model: 'model.py' });\n        *           });\n        *\n        * ** Parameters **\n        * @param {object} options (Optional) Options object to override global options.\n        * @param {object} options.model The model file to use to create a run if needed.\n        * @return {Promise}\n        */\n        getCurrentRunId: function (options) {\n            options = options || {};\n\n            setIdFilterOrThrowError(options);\n\n            var getOptions = $.extend(true, {},\n                serviceOptions,\n                options,\n                { url: urlConfig.getAPIPath(apiEndpoint) + serviceOptions.filter + '/run' }\n            );\n\n            validateModelOrThrowError(getOptions);\n            return http.post(_pick(getOptions, 'model'), getOptions);\n        },\n\n        /**\n        * Gets the current (most recent) world for the given end user in the given group. Brings this most recent world into memory if needed.\n        *\n        *  **Example**\n        *\n        *      var wa = new F.service.World({\n        *           account: 'acme-simulations',\n        *           project: 'supply-chain-game',\n        *           group: 'team1' });\n        *      wa.getCurrentWorldForUser('8f2604cf-96cd-449f-82fa-e331530734ee')\n        *           .then(function(world) {\n        *               // use data from world\n        *           });\n        *\n        * ** Parameters **\n        * @param {string} userId The `userId` of the user whose current (most recent) world is being retrieved.\n        * @param {string} groupName (Optional) The name of the group. If not provided, defaults to the group used to create the service.\n        * @return {Promise}\n        */\n        getCurrentWorldForUser: function (userId, groupName) {\n            var dtd = $.Deferred();\n            var me = this;\n            this.getWorldsForUser(userId, { group: groupName })\n                .then(function (worlds) {\n                    // assume the most recent world as the 'active' world\n                    worlds.sort(function (a, b) { return new Date(b.lastModified) - new Date(a.lastModified); });\n                    var currentWorld = worlds[0];\n\n                    if (currentWorld) {\n                        serviceOptions.filter = currentWorld.id;\n                    }\n\n                    dtd.resolveWith(me, [currentWorld]);\n                })\n                .fail(dtd.reject);\n\n            return dtd.promise();\n        },\n\n        /**\n        * Deletes the current run from the world.\n        *\n        * (Note that the world id remains part of the run record, indicating that the run was formerly an active run for the world.)\n        *\n        *  **Example**\n        *\n        *      var wa = new F.service.World({\n        *           account: 'acme-simulations',\n        *           project: 'supply-chain-game',\n        *           group: 'team1' });\n        *\n        *      wa.deleteRun('sample-world-id');\n        *\n        *  **Parameters**\n        * @param {string} worldId The `worldId` of the world from which the current run is being deleted.\n        * @param {object} options (Optional) Options object to override global options.\n        * @return {Promise}\n        */\n        deleteRun: function (worldId, options) {\n            options = options || {};\n\n            if (worldId) {\n                options.filter = worldId;\n            }\n\n            setIdFilterOrThrowError(options);\n\n            var deleteOptions = $.extend(true, {},\n                serviceOptions,\n                options,\n                { url: urlConfig.getAPIPath(apiEndpoint) + serviceOptions.filter + '/run' }\n            );\n\n            return http.delete(null, deleteOptions);\n        },\n\n        /**\n        * Creates a new run for the world.\n        *\n        *  **Example**\n        *\n        *      var wa = new F.service.World({\n        *           account: 'acme-simulations',\n        *           project: 'supply-chain-game',\n        *           group: 'team1' });\n        *\n        *      wa.getCurrentWorldForUser('8f2604cf-96cd-449f-82fa-e331530734ee')\n        *           .then(function (world) {\n        *                   wa.newRunForWorld(world.id);\n        *           });\n        *\n        *  **Parameters**\n        * @param {string} worldId worldId in which we create the new run.\n        * @param {object} options (Optional) Options object to override global options.\n        * @param {object} options.model The model file to use to create a run if needed.\n        * @return {Promise}\n        */\n        newRunForWorld: function (worldId, options) {\n            var currentRunOptions = $.extend(true, {},\n                serviceOptions,\n                options,\n                { filter: worldId || serviceOptions.filter }\n            );\n            var me = this;\n\n            validateModelOrThrowError(currentRunOptions);\n\n            return this.deleteRun(worldId, options)\n                .then(function () {\n                    return me.getCurrentRunId(currentRunOptions);\n                });\n        },\n\n        /**\n        * Assigns end users to worlds, creating new worlds as appropriate, automatically. Assigns all end users in the group, and creates new worlds as needed based on the project-level world configuration (roles, optional roles, and minimum end users per world).\n        *\n        * **Example**\n        *\n        *      var wa = new F.service.World({\n        *           account: 'acme-simulations',\n        *           project: 'supply-chain-game',\n        *           group: 'team1' });\n        *\n        *      wa.autoAssign();\n        *\n        * **Parameters**\n        * @param {object} options (Optional) Options object to override global options.\n        * @param {number} options.maxUsers Sets the maximum number of users in a world.\n        * @param {string[]} options.userIds A list of users to be assigned be assigned instead of all end users in the group.\n        * @return {Promise}\n        *\n        */\n        autoAssign: function (options) {\n            options = options || {};\n\n            var opt = $.extend(true, {},\n                serviceOptions,\n                options,\n                { url: urlConfig.getAPIPath(assignmentEndpoint) }\n            );\n\n            var params = {\n                account: opt.account,\n                project: opt.project,\n                group: opt.group\n            };\n\n            if (opt.maxUsers) {\n                params.maxUsers = opt.maxUsers;\n            }\n\n            if (opt.userIds) {\n                params.userIds = opt.userIds;\n            }\n\n            return http.post(params, opt);\n        },\n\n        /**\n        * Gets the project's world configuration.\n        *\n        * Typically, every interaction with your project uses the same configuration of each world. For example, each world in your project probably has the same roles for end users. And your project is probably either configured so that all end users share the same world (and run), or smaller sets of end users share worlds — but not both.\n        *\n        * (The [Multiplayer Project REST API](../../../rest_apis/multiplayer/multiplayer_project/) allows you to set these project-level world configurations. The World Adapter simply retrieves them, for example so they can be used in auto-assignment of end users to worlds.)\n        *\n        * **Example**\n        *\n        *      var wa = new F.service.World({\n        *           account: 'acme-simulations',\n        *           project: 'supply-chain-game',\n        *           group: 'team1' });\n        *\n        *      wa.getProjectSettings()\n        *           .then(function(settings) {\n        *               console.log(settings.roles);\n        *               console.log(settings.optionalRoles);\n        *           });\n        *\n        * **Parameters**\n        * @param {object} options (Optional) Options object to override global options.\n        * @return {Promise}\n        */\n        getProjectSettings: function (options) {\n            options = options || {};\n\n            var opt = $.extend(true, {},\n                serviceOptions,\n                options,\n                { url: urlConfig.getAPIPath(projectEndpoint) }\n            );\n\n            opt.url += [opt.account, opt.project].join('/');\n            return http.get(null, opt);\n        }\n\n    };\n\n    $.extend(this, publicAPI);\n};\n","/**\n * @class Cookie Storage Service\n *\n * @example\n *      var people = require('cookie-store')({ root: 'people' });\n        people\n            .save({lastName: 'smith' })\n\n */\n\n\n'use strict';\n\n// Thin document.cookie wrapper to allow unit testing\nvar Cookie = function () {\n    this.get = function () {\n        return document.cookie;\n    };\n\n    this.set = function (newCookie) {\n        document.cookie = newCookie;\n    };\n};\n\nmodule.exports = function (config) {\n    var host = window.location.hostname;\n    var validHost = host.split('.').length > 1;\n    var domain = validHost ? '.' + host : null;\n\n    var defaults = {\n        /**\n         * Name of collection\n         * @type { string}\n         */\n        root: '/',\n\n        domain: domain,\n        cookie: new Cookie()\n    };\n    this.serviceOptions = $.extend({}, defaults, config);\n\n    var publicAPI = {\n        // * TBD\n        //  * Query collection; uses MongoDB syntax\n        //  * @see  <TBD: Data API URL>\n        //  *\n        //  * @param { string} qs Query Filter\n        //  * @param { string} limiters @see <TBD: url for limits, paging etc>\n        //  *\n        //  * @example\n        //  *     cs.query(\n        //  *      { name: 'John', className: 'CSC101' },\n        //  *      {limit: 10}\n        //  *     )\n\n        // query: function (qs, limiters) {\n\n        // },\n\n        /**\n         * Save cookie value\n         * @param  { string|Object} key   If given a key save values under it, if given an object directly, save to top-level api\n         * @param  {Object} value (Optional)\n         * @param {Object} options Overrides for service options\n         *\n         * @return {*} The saved value\n         *\n         * @example\n         *     cs.set('person', { firstName: 'john', lastName: 'smith' });\n         *     cs.set({ name:'smith', age:'32' });\n         */\n        set: function (key, value, options) {\n            var setOptions = $.extend(true, {}, this.serviceOptions, options);\n\n            var domain = setOptions.domain;\n            var path = setOptions.root;\n            var cookie = setOptions.cookie;\n\n            cookie.set(encodeURIComponent(key) + '=' +\n                                encodeURIComponent(value) +\n                                (domain ? '; domain=' + domain : '') +\n                                (path ? '; path=' + path : '')\n            );\n\n            return value;\n        },\n\n        /**\n         * Load cookie value\n         * @param  { string|Object} key   If given a key save values under it, if given an object directly, save to top-level api\n         * @return {*} The value stored\n         *\n         * @example\n         *     cs.get('person');\n         */\n        get: function (key) {\n            var cookie = this.serviceOptions.cookie;\n            var cookieReg = new RegExp('(?:^|;)\\\\s*' + encodeURIComponent(key).replace(/[\\-\\.\\+\\*]/g, '\\\\$&') + '\\\\s*\\\\=\\\\s*([^;]*).*$');\n            var res = cookieReg.exec(cookie.get());\n            var val = res ? decodeURIComponent(res[1]) : null;\n            return val;\n        },\n\n        /**\n         * Removes key from collection\n         * @param { string} key key to remove\n         * @param {object} options (optional) overrides for service options\n         * @return { string} key The key removed\n         *\n         * @example\n         *     cs.remove('person');\n         */\n        remove: function (key, options) {\n            var remOptions = $.extend(true, {}, this.serviceOptions, options);\n\n            var domain = remOptions.domain;\n            var path = remOptions.root;\n            var cookie = remOptions.cookie;\n\n            cookie.set(encodeURIComponent(key) +\n                            '=; expires=Thu, 01 Jan 1970 00:00:00 GMT' +\n                            (domain ? '; domain=' + domain : '') +\n                            (path ? '; path=' + path : '')\n            );\n            return key;\n        },\n\n        /**\n         * Removes collection being referenced\n         * @return { array} keys All the keys removed\n         */\n        destroy: function () {\n            var cookie = this.serviceOptions.cookie;\n            var aKeys = cookie.get().replace(/((?:^|\\s*;)[^\\=]+)(?=;|$)|^\\s*|\\s*(?:\\=[^;]*)?(?:\\1|$)/g, '').split(/\\s*(?:\\=[^;]*)?;\\s*/);\n            for (var nIdx = 0; nIdx < aKeys.length; nIdx++) {\n                var cookieKey = decodeURIComponent(aKeys[nIdx]);\n                this.remove(cookieKey);\n            }\n            return aKeys;\n        }\n    };\n\n    $.extend(this, publicAPI);\n};\n","'use strict';\n\nvar keyNames = require('../managers/key-names');\nvar StorageFactory = require('./store-factory');\nvar optionUtils = require('../util/option-utils');\n\nvar EPI_SESSION_KEY = keyNames.EPI_SESSION_KEY;\nvar EPI_MANAGER_KEY = 'epicenter.token'; //can't be under key-names, or logout will clear this too\n\nvar defaults = {\n    /**\n     * Where to store user access tokens for temporary access. Defaults to storing in a cookie in the browser.\n     * @type {string}\n     */\n    store: { synchronous: true }\n};\n\nvar SessionManager = function (managerOptions) {\n    managerOptions = managerOptions || {};\n    function getBaseOptions(overrides) {\n        overrides = overrides || {};\n        var libOptions = optionUtils.getOptions();\n        var finalOptions = $.extend(true, {}, defaults, libOptions, managerOptions, overrides);\n        return finalOptions;\n    }\n\n    function getStore(overrides) {\n        var baseOptions = getBaseOptions(overrides);\n        var storeOpts = baseOptions.store || {};\n        var isEpicenterDomain = !baseOptions.isLocal && !baseOptions.isCustomDomain;\n        if (storeOpts.root === undefined && baseOptions.account && baseOptions.project && isEpicenterDomain) {\n            storeOpts.root = '/app/' + baseOptions.account + '/' + baseOptions.project;\n        }\n        return new StorageFactory(storeOpts);\n    }\n\n    var publicAPI = {\n        saveSession: function (userInfo, options) {\n            var serialized = JSON.stringify(userInfo);\n            getStore(options).set(EPI_SESSION_KEY, serialized);\n        },\n        getSession: function (options) {\n            var store = getStore(options);\n            var finalOpts = store.serviceOptions;\n            var serialized = store.get(EPI_SESSION_KEY) || '{}';\n            var session = JSON.parse(serialized);\n            // If the url contains the project and account\n            // validate the account and project in the session\n            // and override project, groupName, groupId and isFac\n            // Otherwise (i.e. localhost) use the saved session values\n            var account = finalOpts.account;\n            var project = finalOpts.project;\n            if (account && session.account !== account) {\n                // This means that the token was not used to login to the same account\n                return {};\n            }\n            if (session.groups && account && project) {\n                var group = session.groups[project] || { groupId: '', groupName: '', isFac: false };\n                $.extend(session, { project: project }, group);\n            }\n            return session;\n        },\n        removeSession: function (options) {\n            var store = getStore(options);\n            Object.keys(keyNames).forEach(function (cookieKey) {\n                var cookieName = keyNames[cookieKey];\n                store.remove(cookieName);\n            });\n            return true;\n        },\n        getStore: function (options) {\n            return getStore(options);\n        },\n\n        getMergedOptions: function () {\n            var args = Array.prototype.slice.call(arguments);\n            var overrides = $.extend.apply($, [true, {}].concat(args));\n            var baseOptions = getBaseOptions(overrides);\n            var session = this.getSession(overrides);\n\n            var token = session.auth_token;\n            if (!token) {\n                var factory = new StorageFactory();\n                token = factory.get(EPI_MANAGER_KEY);\n            }\n\n            var sessionDefaults = {\n                /**\n                 * For projects that require authentication, pass in the user access token (defaults to empty string). If the user is already logged in to Epicenter, the user access token is already set in a cookie and automatically loaded from there. (See [more background on access tokens](../../../project_access/)).\n                 * @see [Authentication API Service](../auth-api-service/) for getting tokens.\n                 * @type {String}\n                 */\n                token: token,\n\n                /**\n                 * The account. If left undefined, taken from the cookie session.\n                 * @type {String}\n                 */\n                account: session.account,\n\n                /**\n                 * The project. If left undefined, taken from the cookie session.\n                 * @type {String}\n                 */\n                project: session.project,\n\n\n                /**\n                 * The group name. If left undefined, taken from the cookie session.\n                 * @type {String}\n                 */\n                group: session.groupName,\n                /**\n                 * Alias for group. \n                 * @type {String}\n                 */\n                groupName: session.groupName, //It's a little weird that it's called groupName in the cookie, but 'group' in all the service options, so normalize for both\n                /**\n                 * The group id. If left undefined, taken from the cookie session.\n                 * @type {String}\n                 */\n                groupId: session.groupId,\n                userId: session.userId,\n                userName: session.userName,\n            };\n            return $.extend(true, sessionDefaults, baseOptions);\n        }\n    };\n    $.extend(this, publicAPI);\n};\n\nmodule.exports = SessionManager;","/**\n    Decides type of store to provide\n*/\n\n'use strict';\n// var isNode = false; FIXME: Browserify/minifyify has issues with the next link\n// var store = (isNode) ? require('./session-store') : require('./cookie-store');\nvar store = require('./cookie-store');\n\nmodule.exports = store;\n","'use strict';\n\nvar qutils = require('../util/query-util');\n\nmodule.exports = function (config) {\n\n    var defaults = {\n        url: '',\n\n        contentType: 'application/json',\n        headers: {},\n        statusCode: {\n            404: $.noop\n        },\n\n        /**\n         * ONLY for strings in the url. All GET & DELETE params are run through this\n         * @type {[type] }\n         */\n        parameterParser: qutils.toQueryFormat,\n\n        // To allow epicenter.token and other session cookies to be passed\n        // with the requests\n        xhrFields: {\n            withCredentials: true\n        }\n    };\n\n    var transportOptions = $.extend({}, defaults, config);\n\n    var result = function (d) {\n        return ($.isFunction(d)) ? d() : d;\n    };\n\n    var connect = function (method, params, connectOptions) {\n        params = result(params);\n        params = ($.isPlainObject(params) || $.isArray(params)) ? JSON.stringify(params) : params;\n\n        var options = $.extend(true, {}, transportOptions, connectOptions, {\n            type: method,\n            data: params\n        });\n        var ALLOWED_TO_BE_FUNCTIONS = ['data', 'url'];\n        $.each(options, function (key, value) {\n            if ($.isFunction(value) && $.inArray(key, ALLOWED_TO_BE_FUNCTIONS) !== -1) {\n                options[key] = value();\n            }\n        });\n\n        if (options.logLevel && options.logLevel === 'DEBUG') {\n            console.log(options.url);\n            var oldSuccessFn = options.success || $.noop;\n            options.success = function (response, ajaxStatus, ajaxReq) {\n                console.log(response);\n                oldSuccessFn.apply(this, arguments);\n            };\n        }\n\n        var beforeSend = options.beforeSend;\n        options.beforeSend = function (xhr, settings) {\n            xhr.requestUrl = (connectOptions || {}).url;\n            if (beforeSend) {\n                beforeSend.apply(this, arguments);\n            }\n        };\n\n        return $.ajax(options);\n    };\n\n    var publicAPI = {\n        get: function (params, ajaxOptions) {\n            var options = $.extend({}, transportOptions, ajaxOptions);\n            params = options.parameterParser(result(params));\n            return connect.call(this, 'GET', params, options);\n        },\n        splitGet: function () {\n\n        },\n        post: function () {\n            return connect.apply(this, ['post'].concat([].slice.call(arguments)));\n        },\n        patch: function () {\n            return connect.apply(this, ['patch'].concat([].slice.call(arguments)));\n        },\n        put: function () {\n            return connect.apply(this, ['put'].concat([].slice.call(arguments)));\n        },\n        delete: function (params, ajaxOptions) {\n            //DELETE doesn't support body params, but jQuery thinks it does.\n            var options = $.extend({}, transportOptions, ajaxOptions);\n            params = options.parameterParser(result(params));\n            if ($.trim(params)) {\n                var delimiter = (result(options.url).indexOf('?') === -1) ? '?' : '&';\n                options.url = result(options.url) + delimiter + params;\n            }\n            return connect.call(this, 'DELETE', null, options);\n        },\n        head: function () {\n            return connect.apply(this, ['head'].concat([].slice.call(arguments)));\n        },\n        options: function () {\n            return connect.apply(this, ['options'].concat([].slice.call(arguments)));\n        }\n    };\n\n    return $.extend(this, publicAPI);\n};\n","'use strict';\n\n// var isNode = false; FIXME: Browserify/minifyify has issues with the next link\n// var transport = (isNode) ? require('./node-http-transport') : require('./ajax-http-transport');\nvar transport = require('./ajax-http-transport');\nmodule.exports = transport;\n","/**\n/* Inherit from a class (using prototype borrowing)\n*/\n'use strict';\n\nfunction inherit(C, P) {\n    var F = function () {};\n    F.prototype = P.prototype;\n    C.prototype = new F();\n    C.__super = P.prototype;\n    C.prototype.constructor = C;\n}\n\n/**\n* Shallow copy of an object\n* @param {Object} dest object to extend\n* @return {Object} extended object\n*/\nvar extend = function (dest /*, var_args*/) {\n    var obj = Array.prototype.slice.call(arguments, 1);\n    var current;\n    for (var j = 0; j < obj.length; j++) {\n        if (!(current = obj[j])) { //eslint-disable-line\n            continue;\n        }\n\n        // do not wrap inner in dest.hasOwnProperty or bad things will happen\n        for (var key in current) { //eslint-disable-line\n            dest[key] = current[key];\n        }\n    }\n\n    return dest;\n};\n\nmodule.exports = function (base, props, staticProps) {\n    var parent = base;\n    var child;\n\n    child = props && props.hasOwnProperty('constructor') ? props.constructor : function () { return parent.apply(this, arguments); };\n\n    // add static properties to the child constructor function\n    extend(child, parent, staticProps);\n\n    // associate prototype chain\n    inherit(child, parent);\n\n    // add instance properties\n    if (props) {\n        extend(child.prototype, props);\n    }\n\n    // done\n    return child;\n};\n","'use strict';\n\nmodule.exports = {\n    _pick: function (obj, props) {\n        var res = {};\n        for (var p in obj) {\n            if (props.indexOf(p) !== -1) {\n                res[p] = obj[p];\n            }\n        }\n\n        return res;\n    },\n    isEmpty: function isEmpty(value) {\n        return (!value || ($.isPlainObject(value) && Object.keys(value).length === 0));\n    }\n};\n","'use strict';\n\nvar ConfigService = require('../service/configuration-service');\n\nvar urlConfig = new ConfigService().get('server');\nvar customDefaults = {};\nvar libDefaults = {\n    /**\n     * The account id. In the Epicenter UI, this is the **Team ID** (for team projects) or **User ID** (for personal projects). Defaults to empty string. If left undefined, taken from the URL.\n     * @type {String}\n     */\n    account: urlConfig.accountPath || undefined,\n    /**\n     * The account id. In the Epicenter UI, this is the **Team ID** (for team projects) or **User ID** (for personal projects). Defaults to empty string. If left undefined, taken from the URL.\n     * @type {String}\n     */\n    project: urlConfig.projectPath || undefined,\n    isLocal: urlConfig.isLocalhost(),\n    isCustomDomain: urlConfig.isCustomDomain,\n    store: {}\n};\n\nvar optionUtils = {\n    /**\n     * Gets the final options by overriding the global options set with\n     * optionUtils#setDefaults() and the lib defaults.\n     * @param {object} options The final options object.\n     * @return {object} Extended object\n     */\n    getOptions: function (options) {\n        return $.extend(true, {}, libDefaults, customDefaults, options);\n    },\n    /**\n     * Sets the global defaults for the optionUtils#getOptions() method.\n     * @param {object} defaults The defaults object.\n     */\n    setDefaults: function (defaults) {\n        customDefaults = defaults;\n    }\n};\nmodule.exports = optionUtils;\n","/**\n * Utilities for working with query strings\n*/\n'use strict';\n\nmodule.exports = (function () {\n\n    return {\n        /**\n         * Converts to matrix format\n         * @param  {Object} qs Object to convert to query string\n         * @return { string}    Matrix-format query parameters\n         */\n        toMatrixFormat: function (qs) {\n            if (qs === null || qs === undefined || qs === '') {\n                return ';';\n            }\n            if (typeof qs === 'string' || qs instanceof String) {\n                return qs;\n            }\n\n            var returnArray = [];\n            var OPERATORS = ['<', '>', '!'];\n            $.each(qs, function (key, value) {\n                if (typeof value !== 'string' || $.inArray($.trim(value).charAt(0), OPERATORS) === -1) {\n                    value = '=' + value;\n                }\n                returnArray.push(key + value);\n            });\n\n            var mtrx = ';' + returnArray.join(';');\n            return mtrx;\n        },\n\n        /**\n         * Converts strings/arrays/objects to type 'a=b&b=c'\n         * @param  { string|Array|Object} qs\n         * @return { string}\n         */\n        toQueryFormat: function (qs) {\n            if (qs === null || qs === undefined) {\n                return '';\n            }\n            if (typeof qs === 'string' || qs instanceof String) {\n                return qs;\n            }\n\n            var returnArray = [];\n            $.each(qs, function (key, value) {\n                if ($.isArray(value)) {\n                    value = value.join(',');\n                }\n                if ($.isPlainObject(value)) {\n                    //Mostly for data api\n                    value = JSON.stringify(value);\n                }\n                returnArray.push(key + '=' + value);\n            });\n\n            var result = returnArray.join('&');\n            return result;\n        },\n\n        /**\n         * Converts strings of type 'a=b&b=c' to { a:b, b:c}\n         * @param  { string} qs\n         * @return {object}\n         */\n        qsToObject: function (qs) {\n            if (qs === null || qs === undefined || qs === '') {\n                return {};\n            }\n\n            var qsArray = qs.split('&');\n            var returnObj = {};\n            $.each(qsArray, function (index, value) {\n                var qKey = value.split('=')[0];\n                var qVal = value.split('=')[1];\n\n                if (qVal.indexOf(',') !== -1) {\n                    qVal = qVal.split(',');\n                }\n\n                returnObj[qKey] = qVal;\n            });\n\n            return returnObj;\n        },\n\n        /**\n         * Normalizes and merges strings of type 'a=b', { b:c} to { a:b, b:c}\n         * @param  { string|Array|Object} qs1\n         * @param  { string|Array|Object} qs2\n         * @return {Object}\n         */\n        mergeQS: function (qs1, qs2) {\n            var obj1 = this.qsToObject(this.toQueryFormat(qs1));\n            var obj2 = this.qsToObject(this.toQueryFormat(qs2));\n            return $.extend(true, {}, obj1, obj2);\n        },\n\n        addTrailingSlash: function (url) {\n            if (!url) {\n                return '';\n            }\n            return (url.charAt(url.length - 1) === '/') ? url : (url + '/');\n        }\n    };\n}());\n\n\n","/**\n * Utilities for working with the run service\n*/\n'use strict';\nvar qutil = require('./query-util');\nvar MAX_URL_LENGTH = 2048;\n\nmodule.exports = (function () {\n    return {\n        /**\n         * normalizes different types of operation inputs\n         * @param  {Object|Array|String} operations operations to perform\n         * @param  {Array} args arguments for operation\n         * @return {String} operations of the form `{ ops: [], args: [] }`\n         */\n        normalizeOperations: function (operations, args) {\n            if (!args) {\n                args = [];\n            }\n            var returnList = {\n                ops: [],\n                args: []\n            };\n\n            var _concat = function (arr) {\n                return (arr !== null && arr !== undefined) ? [].concat(arr) : [];\n            };\n\n            //{ add: [1,2], subtract: [2,4] }\n            var _normalizePlainObjects = function (operations, returnList) {\n                if (!returnList) {\n                    returnList = { ops: [], args: [] };\n                }\n                $.each(operations, function (opn, arg) {\n                    returnList.ops.push(opn);\n                    returnList.args.push(_concat(arg));\n                });\n                return returnList;\n            };\n            //{ name: 'add', params: [1] }\n            var _normalizeStructuredObjects = function (operation, returnList) {\n                if (!returnList) {\n                    returnList = { ops: [], args: [] };\n                }\n                returnList.ops.push(operation.name);\n                returnList.args.push(_concat(operation.params));\n                return returnList;\n            };\n\n            var _normalizeObject = function (operation, returnList) {\n                return ((operation.name) ? _normalizeStructuredObjects : _normalizePlainObjects)(operation, returnList);\n            };\n\n            var _normalizeLiterals = function (operation, args, returnList) {\n                if (!returnList) {\n                    returnList = { ops: [], args: [] };\n                }\n                returnList.ops.push(operation);\n                returnList.args.push(_concat(args));\n                return returnList;\n            };\n\n\n            var _normalizeArrays = function (operations, arg, returnList) {\n                if (!returnList) {\n                    returnList = { ops: [], args: [] };\n                }\n                $.each(operations, function (index, opn) {\n                    if ($.isPlainObject(opn)) {\n                        _normalizeObject(opn, returnList);\n                    } else {\n                        _normalizeLiterals(opn, args[index], returnList);\n                    }\n                });\n                return returnList;\n            };\n\n            if ($.isPlainObject(operations)) {\n                _normalizeObject(operations, returnList);\n            } else if ($.isArray(operations)) {\n                _normalizeArrays(operations, args, returnList);\n            } else {\n                _normalizeLiterals(operations, args, returnList);\n            }\n\n            return returnList;\n        },\n\n        splitGetFactory: function (httpOptions) {\n            return function (params, options) {\n                var http = this; //eslint-disable-line\n                var getValue = function (name) {\n                    var value = options[name] || httpOptions[name];\n                    if (typeof value === 'function') {\n                        value = value();\n                    }\n                    return value;\n                };\n                var getFinalUrl = function (params) {\n                    var url = getValue('url', options);\n                    var data = params;\n                    // There is easy (or known) way to get the final URL jquery is going to send so\n                    // we're replicating it. The process might change at some point but it probably will not.\n                    // 1. Remove hash\n                    url = url.replace(/#.*$/, '');\n                    // 1. Append query string\n                    var queryParams = qutil.toQueryFormat(data);\n                    var questionIdx = url.indexOf('?');\n                    if (queryParams && questionIdx > -1) {\n                        return url + '&' + queryParams;\n                    } else if (queryParams) {\n                        return url + '?' + queryParams;\n                    }\n                    return url;\n                };\n                var url = getFinalUrl(params);\n                // We must split the GET in multiple short URL's\n                // The only property allowed to be split is \"include\"\n                if (params && params.include && encodeURI(url).length > MAX_URL_LENGTH) {\n                    var dtd = $.Deferred();\n                    var paramsCopy = $.extend(true, {}, params);\n                    delete paramsCopy.include;\n                    var urlNoIncludes = getFinalUrl(paramsCopy);\n                    var diff = MAX_URL_LENGTH - urlNoIncludes.length;\n                    var oldSuccess = options.success || httpOptions.success || $.noop;\n                    var oldError = options.error || httpOptions.error || $.noop;\n                    // remove the original success and error callbacks\n                    options.success = $.noop;\n                    options.error = $.noop;\n\n                    var include = params.include;\n                    var currIncludes = [];\n                    var includeOpts = [currIncludes];\n                    var currLength = encodeURIComponent('?include=').length;\n                    var variable = include.pop();\n                    while (variable) {\n                        var varLenght = encodeURIComponent(variable).length;\n                        // Use a greedy approach for now, can be optimized to be solved in a more\n                        // efficient way\n                        // + 1 is the comma\n                        if (currLength + varLenght + 1 < diff) {\n                            currIncludes.push(variable);\n                            currLength += varLenght + 1;\n                        } else {\n                            currIncludes = [variable];\n                            includeOpts.push(currIncludes);\n                            currLength = '?include='.length + varLenght;\n                        }\n                        variable = include.pop();\n                    }\n                    var reqs = $.map(includeOpts, function (include) {\n                        var reqParams = $.extend({}, params, { include: include });\n                        return http.get(reqParams, options);\n                    });\n                    $.when.apply($, reqs).then(function () {\n                        // Each argument are arrays of the arguments of each done request\n                        // So the first argument of the first array of arguments is the data\n                        var isValid = arguments[0] && arguments[0][0];\n                        if (!isValid) {\n                            // Should never happen...\n                            oldError();\n                            return dtd.reject();\n                        }\n                        var firstResponse = arguments[0][0];\n                        var isObject = $.isPlainObject(firstResponse);\n                        var isRunAPI = (isObject && $.isPlainObject(firstResponse.variables)) || !isObject;\n                        if (isRunAPI) {\n                            if (isObject) {\n                                // aggregate the variables property only\n                                var aggregateRun = arguments[0][0];\n                                $.each(arguments, function (idx, args) {\n                                    var run = args[0];\n                                    $.extend(true, aggregateRun.variables, run.variables);\n                                });\n                                oldSuccess(aggregateRun, arguments[0][1], arguments[0][2]);\n                                dtd.resolve(aggregateRun, arguments[0][1], arguments[0][2]);\n                            } else {\n                                // array of runs\n                                // Agregate variables in each run\n                                var aggregatedRuns = {};\n                                $.each(arguments, function (idx, args) {\n                                    var runs = args[0];\n                                    if (!$.isArray(runs)) {\n                                        return;\n                                    }\n                                    $.each(runs, function (idxRun, run) {\n                                        if (run.id && !aggregatedRuns[run.id]) {\n                                            run.variables = run.variables || {};\n                                            aggregatedRuns[run.id] = run;\n                                        } else if (run.id) {\n                                            $.extend(true, aggregatedRuns[run.id].variables, run.variables);\n                                        }\n                                    });\n                                });\n                                // turn it into an array\n                                aggregatedRuns = $.map(aggregatedRuns, function (run) { return run; });\n                                oldSuccess(aggregatedRuns, arguments[0][1], arguments[0][2]);\n                                dtd.resolve(aggregatedRuns, arguments[0][1], arguments[0][2]);\n                            }\n                        } else {\n                            // is variables API\n                            // aggregate the response\n                            var aggregatedVariables = {};\n                            $.each(arguments, function (idx, args) {\n                                var vars = args[0];\n                                $.extend(true, aggregatedVariables, vars);\n                            });\n                            oldSuccess(aggregatedVariables, arguments[0][1], arguments[0][2]);\n                            dtd.resolve(aggregatedVariables, arguments[0][1], arguments[0][2]);\n                        }\n                    }, function () {\n                        oldError.apply(http, arguments);\n                        dtd.reject.apply(dtd, arguments);\n                    });\n                    return dtd.promise();\n                } else {\n                    return http.get(params, options);\n                }\n            };\n        }\n    };\n}());\n"]} diff --git a/dist/epicenter.min.js b/dist/epicenter.min.js index f25368b3..636b8daf 100644 --- a/dist/epicenter.min.js +++ b/dist/epicenter.min.js @@ -34,9 +34,9 @@ var list={"conditional-creation":require("./conditional-creation-strategy"),"new },{"../../service/world-api-adapter":44,"../../util/inherit":50,"./none-strategy":16}],16:[function(require,module,exports){ "use strict";var classFrom=require("../../util/inherit");var Base={};module.exports=classFrom(Base,{constructor:function(options){},reset:function(){return $.Deferred().resolve().promise()},getRun:function(runService){return $.Deferred().resolve(runService).promise()}}); },{"../../util/inherit":50}],17:[function(require,module,exports){ -"use strict";var classFrom=require("../../util/inherit");var IdentityStrategy=require("./none-strategy");var injectFiltersFromSession=require("../strategy-utils").injectFiltersFromSession;var injectScopeFromSession=require("../strategy-utils").injectScopeFromSession;var defaults={filter:{}};var Strategy=classFrom(IdentityStrategy,{constructor:function(options){var strategyOptions=options?options.strategyOptions:{};this.options=$.extend(!0,{},defaults,strategyOptions)},reset:function(runService,userSession,options){var opt=injectScopeFromSession(runService.getCurrentConfig(),userSession);return runService.create(opt,options).then(function(run){return run.freshlyCreated=!0,run})},getRun:function(runService,userSession,runSession,options){var filter=injectFiltersFromSession(this.options.filter,userSession);var me=this;return runService.query(filter,{sort:"created",direction:"desc"}).then(function(runs){return runs.length?runs[0]:me.reset(runService,userSession,options)})}});module.exports=Strategy; +"use strict";var classFrom=require("../../util/inherit");var IdentityStrategy=require("./none-strategy");var injectFiltersFromSession=require("../strategy-utils").injectFiltersFromSession;var injectScopeFromSession=require("../strategy-utils").injectScopeFromSession;var defaults={filter:{}};var Strategy=classFrom(IdentityStrategy,{constructor:function(options){var strategyOptions=options?options.strategyOptions:{};this.options=$.extend(!0,{},defaults,strategyOptions)},reset:function(runService,userSession,options){var opt=injectScopeFromSession(runService.getCurrentConfig(),userSession);return runService.create(opt,options).then(function(run){return run.freshlyCreated=!0,run})},getRun:function(runService,userSession,runSession,options){var filter=injectFiltersFromSession(this.options.filter,userSession);var me=this;return runService.query(filter,{startrecord:0,endrecord:0,sort:"created",direction:"desc"}).then(function(runs){return runs.length?runs[0]:me.reset(runService,userSession,options)})}});module.exports=Strategy; },{"../../util/inherit":50,"../strategy-utils":26,"./none-strategy":16}],18:[function(require,module,exports){ -"use strict";var classFrom=require("../../util/inherit");var injectFiltersFromSession=require("../strategy-utils").injectFiltersFromSession;var injectScopeFromSession=require("../strategy-utils").injectScopeFromSession;var Base={};var defaults={initOperation:[],flag:null};module.exports=classFrom(Base,{constructor:function(options){var strategyOptions=options?options.strategyOptions:{};if(this.options=$.extend(!0,{},defaults,strategyOptions),!this.options.initOperation||!this.options.initOperation.length)throw new Error("Specifying an init function is required for this strategy");this.options.flag||(this.options.flag={isInitComplete:!0})},reset:function(runService,userSession,options){var opt=injectScopeFromSession(runService.getCurrentConfig(),userSession);var me=this;return runService.create(opt,options).then(function(createResponse){return runService.serial([].concat(me.options.initOperation)).then(function(){return createResponse})}).then(function(createResponse){return runService.save(me.options.flag).then(function(patchResponse){return $.extend(!0,{},createResponse,patchResponse)})})},getRun:function(runService,userSession,runSession,options){var sessionFilter=injectFiltersFromSession(this.options.flag,userSession);var runopts=runService.getCurrentConfig();var filter=$.extend(!0,{trashed:!1},sessionFilter,{model:runopts.model});var me=this;return runService.query(filter,{sort:"created",direction:"desc"}).then(function(runs){return runs.length?runs[0]:me.reset(runService,userSession,options)})}}); +"use strict";var classFrom=require("../../util/inherit");var injectFiltersFromSession=require("../strategy-utils").injectFiltersFromSession;var injectScopeFromSession=require("../strategy-utils").injectScopeFromSession;var Base={};var defaults={initOperation:[],flag:null};module.exports=classFrom(Base,{constructor:function(options){var strategyOptions=options?options.strategyOptions:{};if(this.options=$.extend(!0,{},defaults,strategyOptions),!this.options.initOperation||!this.options.initOperation.length)throw new Error("Specifying an init function is required for this strategy");this.options.flag||(this.options.flag={isInitComplete:!0})},reset:function(runService,userSession,options){var opt=injectScopeFromSession(runService.getCurrentConfig(),userSession);var me=this;return runService.create(opt,options).then(function(createResponse){return runService.serial([].concat(me.options.initOperation)).then(function(){return createResponse})}).then(function(createResponse){return runService.save(me.options.flag).then(function(patchResponse){return $.extend(!0,{},createResponse,patchResponse)})})},getRun:function(runService,userSession,runSession,options){var sessionFilter=injectFiltersFromSession(this.options.flag,userSession);var runopts=runService.getCurrentConfig();var filter=$.extend(!0,{trashed:!1},sessionFilter,{model:runopts.model});var me=this;return runService.query(filter,{startrecord:0,endrecord:0,sort:"created",direction:"desc"}).then(function(runs){return runs.length?runs[0]:me.reset(runService,userSession,options)})}}); },{"../../util/inherit":50,"../strategy-utils":26}],19:[function(require,module,exports){ "use strict";var classFrom=require("../../util/inherit");var ConditionalStrategy=require("./conditional-creation-strategy");var __super=ConditionalStrategy.prototype;var Strategy=classFrom(ConditionalStrategy,{constructor:function(options){__super.constructor.call(this,this.createIf,options)},createIf:function(run,headers){return!0}});module.exports=Strategy; },{"../../util/inherit":50,"./conditional-creation-strategy":11}],20:[function(require,module,exports){ @@ -48,7 +48,7 @@ var list={"conditional-creation":require("./conditional-creation-strategy"),"new },{"../service/run-api-service":38,"../service/state-api-adapter":40,"../util/run-util":54,"./run-manager":10,"./run-strategies/none-strategy":16,"./saved-runs-manager":21,"./scenario-strategies/baseline-strategy":23,"./scenario-strategies/reuse-last-unsaved":24,"./strategy-utils":26}],23:[function(require,module,exports){ "use strict";var ReuseinitStrategy=require("../run-strategies/reuse-last-initialized");module.exports=function(options){var defaults={baselineName:"Baseline",initOperation:[{stepTo:"end"}]};var strategyOptions=options?options.strategyOptions:{};var opts=$.extend({},defaults,strategyOptions);return new ReuseinitStrategy({strategyOptions:{initOperation:opts.initOperation,flag:{saved:!0,trashed:!1,name:opts.baselineName}}})}; },{"../run-strategies/reuse-last-initialized":18}],24:[function(require,module,exports){ -"use strict";var classFrom=require("../../util/inherit");var injectFiltersFromSession=require("../strategy-utils").injectFiltersFromSession;var injectScopeFromSession=require("../strategy-utils").injectScopeFromSession;var Base={};module.exports=classFrom(Base,{constructor:function(options){var strategyOptions=options?options.strategyOptions:{};this.options=strategyOptions},reset:function(runService,userSession,options){var opt=injectScopeFromSession(runService.getCurrentConfig(),userSession);return runService.create(opt,options).then(function(createResponse){return $.extend(!0,{},createResponse,{freshlyCreated:!0})})},getRun:function(runService,userSession,opts){var runopts=runService.getCurrentConfig();var filter=injectFiltersFromSession({saved:!1,trashed:!1,model:runopts.model},userSession);var me=this;var outputModifiers={sort:"created",direction:"desc"};return runService.query(filter,outputModifiers).then(function(runs){return runs.length?runs[0]:me.reset(runService,userSession)})}},{requiresAuth:!1}); +"use strict";var classFrom=require("../../util/inherit");var injectFiltersFromSession=require("../strategy-utils").injectFiltersFromSession;var injectScopeFromSession=require("../strategy-utils").injectScopeFromSession;var Base={};module.exports=classFrom(Base,{constructor:function(options){var strategyOptions=options?options.strategyOptions:{};this.options=strategyOptions},reset:function(runService,userSession,options){var opt=injectScopeFromSession(runService.getCurrentConfig(),userSession);return runService.create(opt,options).then(function(createResponse){return $.extend(!0,{},createResponse,{freshlyCreated:!0})})},getRun:function(runService,userSession,opts){var runopts=runService.getCurrentConfig();var filter=injectFiltersFromSession({saved:!1,trashed:!1,model:runopts.model},userSession);var me=this;var outputModifiers={startrecord:0,endrecord:0,sort:"created",direction:"desc"};return runService.query(filter,outputModifiers).then(function(runs){return runs.length?runs[0]:me.reset(runService,userSession)})}},{requiresAuth:!1}); },{"../../util/inherit":50,"../strategy-utils":26}],25:[function(require,module,exports){ "use strict";module.exports={reset:function(params,options,manager){return manager.reset(options)}}; },{}],26:[function(require,module,exports){ diff --git a/dist/epicenter.min.js.map b/dist/epicenter.min.js.map index e4915c77..2646cd59 100644 --- a/dist/epicenter.min.js.map +++ b/dist/epicenter.min.js.map @@ -1 +1 @@ -{"version":3,"sources":["node_modules/browser-pack/_prelude.js","src/api-version.json","node_modules/Base64/base64.js","node_modules/object-assign/index.js","src/app.js","src/env-load.js","src/managers/auth-manager.js","src/managers/channel-manager.js","src/managers/epicenter-channel-manager.js","src/managers/key-names.js","src/managers/run-manager.js","src/managers/run-strategies/conditional-creation-strategy.js","src/managers/run-strategies/deprecated/new-if-initialized-strategy.js","src/managers/run-strategies/deprecated/new-if-persisted-strategy.js","src/managers/run-strategies/index.js","src/managers/run-strategies/multiplayer-strategy.js","src/managers/run-strategies/none-strategy.js","src/managers/run-strategies/reuse-across-sessions.js","src/managers/run-strategies/reuse-last-initialized.js","src/managers/run-strategies/reuse-never.js","src/managers/run-strategies/reuse-per-session.js","src/managers/saved-runs-manager.js","src/managers/scenario-manager.js","src/managers/scenario-strategies/baseline-strategy.js","src/managers/scenario-strategies/reuse-last-unsaved.js","src/managers/special-operations.js","src/managers/strategy-utils.js","src/managers/world-manager.js","src/service/admin-file-service.js","src/service/asset-api-adapter.js","src/service/auth-api-service.js","src/service/channel-service.js","src/service/configuration-service.js","src/service/data-api-service.js","src/service/group-api-service.js","src/service/introspection-api-service.js","src/service/member-api-adapter.js","src/service/presence-api-service.js","src/service/run-api-service.js","src/service/service-utils.js","src/service/state-api-adapter.js","src/service/url-config-service.js","src/service/user-api-adapter.js","src/service/variables-api-service.js","src/service/world-api-adapter.js","src/store/cookie-store.js","src/store/session-manager.js","src/store/store-factory.js","src/transport/ajax-http-transport.js","src/transport/http-transport-factory.js","src/util/inherit.js","src/util/object-util.js","src/util/option-utils.js","src/util/query-util.js","src/util/run-util.js"],"names":["InvalidCharacterError","message","this","object","exports","self","chars","prototype","Error","name","btoa","input","str","String","block","charCode","idx","map","output","charAt","charCodeAt","atob","replace","length","bs","buffer","bc","fromCharCode","indexOf","toObject","val","undefined","TypeError","Object","shouldUseNative","assign","test1","getOwnPropertyNames","test2","i","order2","n","join","test3","split","forEach","letter","keys","e","hasOwnProperty","propIsEnumerable","propertyIsEnumerable","module","target","source","from","to","symbols","s","arguments","key","call","getOwnPropertySymbols","F","_private","util","factory","transport","store","service","manager","strategy","load","require","global","SKIP_ENV_LOAD","query","run","classFrom","strategyutils","Transport","Ajax","URL","Config","Run","File","Variables","Data","Auth","World","State","User","Member","Asset","Group","Introspect","Presence","Cookie","Store","ScenarioManager","RunManager","AuthManager","WorldManager","SavedRunsManager","strategies","list","ChannelManager","Channel","version","api","constants","URLConfigService","envLoad","callback","urlService","infoUrl","getAPIPath","envPromise","$","ajax","url","async","then","res","overrides","defaults","extend","fail","options","sessionManager","SessionManager","getMergedOptions","authAdapter","AuthAdapter","MemberAdapter","GroupService","_pick","objectAssign","window","requiresGroup","_findUserInGroup","members","id","j","userId","login","me","$d","Deferred","adapterOptions","success","noop","error","outSuccess","outError","groupId","decodeToken","token","encoded","JSON","parse","handleGroupError","statusCode","data","type","logout","statusText","status","reject","handleSuccess","response","access_token","userInfo","oldGroups","getSession","groups","userGroupOpts","auth","user","project","isTeamMember","parent_account_id","userName","user_name","sessionInfo","auth_token","account","user_id","saveSession","apply","resolve","handleGroupList","groupList","userGroups","group","filteredGroups","grep","resGroup","isFac","role","groupData","groupName","sessionInfoWithGroup","opts","groupService","getGroups","getUserGroups","promise","removeCookieFn","removeSession","getToken","httpOptions","session","params","memberInfo","memberAdapter","server","getGroupsForUser","isLoggedIn","getCurrentUserSessionInfo","addGroups","isArray","Array","each","index","extendedGroup","validProps","cometd","logLevel","websocketEnabled","ackEnabled","shareConnection","channel","handshake","defaultCometOptions","currentSubscriptions","_cometd","CometD","isConnected","connectionBroken","trigger","connectionSucceeded","configure","addListener","wasConnected","successful","bind","batch","subs","resubscribe","getChannel","isPlainObject","base","subscribe","subid","concat","unsubs","unsubscribe","removed","splice","on","event","off","ConfigService","validTypes","world","general","chat","getFromSessionOrError","value","sessionKeyName","settings","isPresenceData","payload","__super","EpicenterChannelManager","constructor","urlConfig","get","userProp","ext","authorization","channelOpts","allowAllChannels","baseParts","channelType","getGroupChannel","baseTopic","oldsubs","topic","context","callbackWithoutPresenceData","getWorldChannel","worldid","getUserChannel","userid","getPresenceChannel","callbackWithOnlyPresenceData","getDataChannel","collection","callbackWithCleanData","meta","path","subType","date","dataPath","actualData","EPI_SESSION_KEY","STRATEGY_SESSION_KEY","patchRunService","patched","orig","do","operation","reservedOps","specialOperations","sessionKeyFromOptions","runService","config","getCurrentConfig","sessionKey","isFunction","setRunInSession","variables","getStore","set","stringify","RunService","isEmpty","getBestStrategy","keyNames","baseKey","reduce","accum","getRun","sessionStore","sessionContents","runSession","runId","authSession","requiresAuth","console","updateConfig","filter","results","catch","err","reset","Base","Strategy","condition","userSession","opt","scope","create","freshlyCreated","loadAndCheck","shouldCreate","msg","headers","ConditionalStrategy","createIf","warn","getResponseHeader","initialized","conditional-creation","new-if-initialized","new-if-persisted","none","multiplayer","reuse-never","reuse-per-session","reuse-across-sessions","reuse-last-initialized","byName","strategyName","strategyOptions","initOperation","StrategyCtor","strategyInstance","register","IdentityStrategy","WorldApiAdapter","worldApi","curUserId","curGroupName","getCurrentWorldForUser","newRunForWorld","runid","model","dtd","loadRunFromWorld","getCurrentRunId","serverError","injectFiltersFromSession","injectScopeFromSession","sort","direction","runs","flag","isInitComplete","createResponse","serial","save","patchResponse","sessionFilter","runopts","trashed","scopeByGroup","scopeByUser","otherFields","param","saved","mark","remove","toMark","rs","existingOptions","proms","r","when","autoRestore","getRuns","modifiers","scopedFilter","opModifiers","include","cookieNameFromOptions","prefix","advanceOperation","BaselineStrategyToUse","includeBaseLine","BaselineStrategy","NoneStrategy","baseline","strategyUtils","mergeRunOptions","baselineName","runName","savedRuns","origGetRuns","args","current","LastUnsavedStrategy","saveAndAdvance","metadata","clone","sa","StateService","advanceOpns","rutil","normalizeOperations","exclude","ops","markSaved","savedResponse","advance","ReuseinitStrategy","stepTo","outputModifiers","currentFilter","currentParams","buildStrategy","worldId","resolveWith","WorldApi","_auth","getCurrentWorld","getCurrentRun","getAndRestoreLatestRun","currentWorldId","runOpts","rm","TransportFactory","uploadBody","fileName","contents","boundary","body","uploadFileOptions","filePath","pop","serviceOptions","folderType","extraParams","FormData","processData","contentType","upload","accountPath","projectPath","Authorization","http","publicAsyncAPI","getContents","put","replaceExisting","prom","post","xhr","conflictStatus","delete","rename","newName","patch","apiEndpoint","fullUrl","transportOptions","assetApiParams","scopeConfig","validateFilename","filename","validateUrlParams","partKeys","buildUrl","parts","method","toLowerCase","urlOptions","createOptions","publicAPI","getServiceOptions","getOptions","files","fullPathFiles","file","assetUrl","password","resp","statusMessage","postParams","topicResolver","channelOptions","makeName","channelName","topics","subscriptionIds","push","publish","returnObjs","setEnv","env","property","qutil","root","getURL","addTrailingSlash","outputModifier","q","attrs","saveAs","serviceUtils","getDefaultOptions","finalOpts","finalParams","getApiUrl","byModel","modelFile","byRunID","runID","getFinalParams","patchUserActiveField","active","isString","objParams","getParms","getGroupDetails","makeUserActive","makeUserInactive","markOnline","markOffline","getStatus","cm","VariablesService","IntrospectionService","updateURLConfig","getFilterURL","filterMatrix","toMatrixFormat","addAutoRestoreHeader","isFilterRunId","autorestoreOpts","X-AutoRestore","updateHTTPConfig","splitGet","splitGetFactory","setFilterOrThrowError","runApiParams","oldSuccess","qs","filters","removeFromMemory","attributes","opsArgs","postOptions","result","prms","operations","opParams","responses","doSingleOp","op","shift","arg","parallel","queue","slice","actualResponse","a","introspect","introspectionConfig","introspection","publicSyncAPI","vs","rest","parseRunIdOrError","httpParams","replay","replayOptions","action","getLocalHost","existingFn","host","localHostFn","isLocal","epiVersion","location","pathname","UrlConfigService","envConf","isLocalhost","actingHost","API_PROTOCOL","HOST_API_MAPPING","forio.com","foriodev.com","publicExports","protocol","apiHost","isCustomDomain","pathHasApp","appPath","accnt","prj","versionPath","PROJECT_APIS","apiMapping","actualProtocol","configProtocol","apiPath","inArray","toQFilter","toIdFilters","getFilters","toQueryFormat","threshold","getById","variable","apiBase","assignmentEndpoint","projectEndpoint","setIdFilterOrThrowError","validateModelOrThrowError","worldApiParams","validParams","update","whitelist","updateOptions","deleteOptions","getWorldsForUser","addUsers","users","u","isObject","updateUser","patchOptions","removeUser","worlds","b","Date","lastModified","currentWorld","deleteRun","currentRunOptions","autoAssign","maxUsers","userIds","getProjectSettings","document","cookie","newCookie","hostname","validHost","domain","setOptions","encodeURIComponent","cookieReg","RegExp","exec","decodeURIComponent","remOptions","destroy","aKeys","nIdx","cookieKey","StorageFactory","optionUtils","EPI_MANAGER_KEY","synchronous","managerOptions","getBaseOptions","libOptions","finalOptions","baseOptions","storeOpts","isEpicenterDomain","serialized","cookieName","sessionDefaults","qutils","404","parameterParser","xhrFields","withCredentials","d","connect","connectOptions","ALLOWED_TO_BE_FUNCTIONS","log","oldSuccessFn","ajaxStatus","ajaxReq","beforeSend","requestUrl","ajaxOptions","trim","delimiter","head","inherit","C","P","dest","obj","props","staticProps","parent","child","p","customDefaults","libDefaults","setDefaults","returnArray","OPERATORS","mtrx","qsToObject","qsArray","returnObj","qKey","qVal","mergeQS","qs1","qs2","obj1","obj2","MAX_URL_LENGTH","returnList","_concat","arr","_normalizePlainObjects","opn","_normalizeStructuredObjects","_normalizeObject","_normalizeLiterals","_normalizeArrays","getValue","getFinalUrl","queryParams","questionIdx","encodeURI","paramsCopy","urlNoIncludes","diff","oldError","currIncludes","includeOpts","currLength","varLenght","reqs","reqParams","isValid","firstResponse","isRunAPI","aggregateRun","aggregatedRuns","idxRun","aggregatedVariables","vars"],"mappings":"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","file":"bundle.js","sourcesContent":["(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require==\"function\"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error(\"Cannot find module '\"+o+\"'\");throw f.code=\"MODULE_NOT_FOUND\",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require==\"function\"&&require;for(var o=0;o> 8 - idx % 1 * 8)\n ) {\n charCode = str.charCodeAt(idx += 3/4);\n if (charCode > 0xFF) {\n throw new InvalidCharacterError(\"'btoa' failed: The string to be encoded contains characters outside of the Latin1 range.\");\n }\n block = block << 8 | charCode;\n }\n return output;\n });\n\n // decoder\n // [https://gist.github.com/1020396] by [https://github.com/atk]\n object.atob || (\n object.atob = function (input) {\n var str = String(input).replace(/=+$/, '');\n if (str.length % 4 == 1) {\n throw new InvalidCharacterError(\"'atob' failed: The string to be decoded is not correctly encoded.\");\n }\n for (\n // initialize result and counters\n var bc = 0, bs, buffer, idx = 0, output = '';\n // get next character\n buffer = str.charAt(idx++);\n // character found in table? initialize bit storage and add its ascii value;\n ~buffer && (bs = bc % 4 ? bs * 64 + buffer : buffer,\n // and if not first of each 4 characters,\n // convert the first 8 bits to one ascii character\n bc++ % 4) ? output += String.fromCharCode(255 & bs >> (-2 * bc & 6)) : 0\n ) {\n // try to find character in table (0-63, not found => -1)\n buffer = chars.indexOf(buffer);\n }\n return output;\n });\n\n}());\n","'use strict';\n/* eslint-disable no-unused-vars */\nvar hasOwnProperty = Object.prototype.hasOwnProperty;\nvar propIsEnumerable = Object.prototype.propertyIsEnumerable;\n\nfunction toObject(val) {\n\tif (val === null || val === undefined) {\n\t\tthrow new TypeError('Object.assign cannot be called with null or undefined');\n\t}\n\n\treturn Object(val);\n}\n\nfunction shouldUseNative() {\n\ttry {\n\t\tif (!Object.assign) {\n\t\t\treturn false;\n\t\t}\n\n\t\t// Detect buggy property enumeration order in older V8 versions.\n\n\t\t// https://bugs.chromium.org/p/v8/issues/detail?id=4118\n\t\tvar test1 = new String('abc'); // eslint-disable-line\n\t\ttest1[5] = 'de';\n\t\tif (Object.getOwnPropertyNames(test1)[0] === '5') {\n\t\t\treturn false;\n\t\t}\n\n\t\t// https://bugs.chromium.org/p/v8/issues/detail?id=3056\n\t\tvar test2 = {};\n\t\tfor (var i = 0; i < 10; i++) {\n\t\t\ttest2['_' + String.fromCharCode(i)] = i;\n\t\t}\n\t\tvar order2 = Object.getOwnPropertyNames(test2).map(function (n) {\n\t\t\treturn test2[n];\n\t\t});\n\t\tif (order2.join('') !== '0123456789') {\n\t\t\treturn false;\n\t\t}\n\n\t\t// https://bugs.chromium.org/p/v8/issues/detail?id=3056\n\t\tvar test3 = {};\n\t\t'abcdefghijklmnopqrst'.split('').forEach(function (letter) {\n\t\t\ttest3[letter] = letter;\n\t\t});\n\t\tif (Object.keys(Object.assign({}, test3)).join('') !==\n\t\t\t\t'abcdefghijklmnopqrst') {\n\t\t\treturn false;\n\t\t}\n\n\t\treturn true;\n\t} catch (e) {\n\t\t// We don't expect any of the above to throw, but better to be safe.\n\t\treturn false;\n\t}\n}\n\nmodule.exports = shouldUseNative() ? Object.assign : function (target, source) {\n\tvar from;\n\tvar to = toObject(target);\n\tvar symbols;\n\n\tfor (var s = 1; s < arguments.length; s++) {\n\t\tfrom = Object(arguments[s]);\n\n\t\tfor (var key in from) {\n\t\t\tif (hasOwnProperty.call(from, key)) {\n\t\t\t\tto[key] = from[key];\n\t\t\t}\n\t\t}\n\n\t\tif (Object.getOwnPropertySymbols) {\n\t\t\tsymbols = Object.getOwnPropertySymbols(from);\n\t\t\tfor (var i = 0; i < symbols.length; i++) {\n\t\t\t\tif (propIsEnumerable.call(from, symbols[i])) {\n\t\t\t\t\tto[symbols[i]] = from[symbols[i]];\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\t}\n\n\treturn to;\n};\n","/**\n * Epicenter Javascript libraries\n * v<%= version %>\n * https://github.com/forio/epicenter-js-libs\n */\n\nvar F = {\n _private: {}, //need this hook now because tests expect everything to be global. Delete once tests are browserified\n util: {},\n factory: {},\n transport: {},\n store: {},\n service: {},\n manager: {\n strategy: {}\n },\n\n};\n\nF.load = require('./env-load');\n\nif (!global.SKIP_ENV_LOAD) {\n F.load();\n}\n\nF.util.query = require('./util/query-util');\nF.util.run = require('./util/run-util');\nF.util.classFrom = require('./util/inherit');\nF._private.strategyutils = require('./managers/strategy-utils');\n\nF.factory.Transport = require('./transport/http-transport-factory');\nF.transport.Ajax = require('./transport/ajax-http-transport');\n\nF.service.URL = require('./service/url-config-service');\nF.service.Config = require('./service/configuration-service');\nF.service.Run = require('./service/run-api-service');\nF.service.File = require('./service/admin-file-service');\nF.service.Variables = require('./service/variables-api-service');\nF.service.Data = require('./service/data-api-service');\nF.service.Auth = require('./service/auth-api-service');\nF.service.World = require('./service/world-api-adapter');\nF.service.State = require('./service/state-api-adapter');\nF.service.User = require('./service/user-api-adapter');\nF.service.Member = require('./service/member-api-adapter');\nF.service.Asset = require('./service/asset-api-adapter');\nF.service.Group = require('./service/group-api-service');\nF.service.Introspect = require('./service/introspection-api-service');\nF.service.Presence = require('./service/presence-api-service');\n\nF.store.Cookie = require('./store/cookie-store');\nF.factory.Store = require('./store/store-factory');\n\nF.manager.ScenarioManager = require('./managers/scenario-manager');\nF.manager.RunManager = require('./managers/run-manager');\nF.manager.AuthManager = require('./managers/auth-manager');\nF.manager.WorldManager = require('./managers/world-manager');\nF.manager.SavedRunsManager = require('./managers/saved-runs-manager');\n\nvar strategies = require('./managers/run-strategies');\nF.manager.strategy = strategies.list; //TODO: this is not really a manager so namespace this better\n\nF.manager.ChannelManager = require('./managers/epicenter-channel-manager');\nF.service.Channel = require('./service/channel-service');\n\nF.version = '<%= version %>';\nF.api = require('./api-version.json');\n\nF.constants = require('./managers/key-names');\n\nglobal.F = F;\nmodule.exports = F;\n","'use strict';\n\nvar URLConfigService = require('./service/url-config-service');\n\nvar envLoad = function (callback) {\n var urlService = new URLConfigService();\n var infoUrl = urlService.getAPIPath('config');\n var envPromise = $.ajax({ url: infoUrl, async: false });\n envPromise = envPromise.then(function (res) {\n var overrides = res.api;\n URLConfigService.defaults = $.extend(URLConfigService.defaults, overrides);\n });\n return envPromise.then(callback).fail(callback);\n};\n\nmodule.exports = envLoad;\n","/**\n* ## Authorization Manager\n*\n* The Authorization Manager provides an easy way to manage user authentication (logging in and out) and authorization (keeping track of tokens, sessions, and groups) for projects.\n*\n* The Authorization Manager is most useful for [team projects](../../../glossary/#team) with an access level of [Authenticated](../../../glossary/#access). These projects are accessed by [end users](../../../glossary/#users) who are members of one or more [groups](../../../glossary/#groups).\n*\n* #### Using the Authorization Manager\n*\n* To use the Authorization Manager, instantiate it. Then, make calls to any of the methods you need:\n*\n* var authMgr = new F.manager.AuthManager({\n* account: 'acme-simulations',\n* userName: 'enduser1',\n* password: 'passw0rd'\n* });\n* authMgr.login().then(function () {\n* authMgr.getCurrentUserSessionInfo();\n* });\n*\n*\n* The `options` object passed to the `F.manager.AuthManager()` call can include:\n*\n* * `account`: The account id for this `userName`. In the Epicenter UI, this is the **Team ID** (for team projects) or the **User ID** (for personal projects).\n* * `userName`: Email or username to use for logging in.\n* * `password`: Password for specified `userName`.\n* * `project`: The **Project ID** for the project to log this user into. Optional.\n* * `groupId`: Id of the group to which `userName` belongs. Required for end users if the `project` is specified.\n*\n* If you prefer starting from a template, the Epicenter JS Libs [Login Component](../../#components) uses the Authorization Manager as well. This sample HTML page (and associated CSS and JS files) provides a login form for team members and end users of your project. It also includes a group selector for end users that are members of multiple groups.\n*/\n\n'use strict';\nvar AuthAdapter = require('../service/auth-api-service');\nvar MemberAdapter = require('../service/member-api-adapter');\nvar GroupService = require('../service/group-api-service');\nvar SessionManager = require('../store/session-manager');\nvar _pick = require('../util/object-util')._pick;\nvar objectAssign = require('object-assign');\n\nvar atob = window.atob || require('Base64').atob;\n\nvar defaults = {\n requiresGroup: true\n};\n\nfunction AuthManager(options) {\n options = $.extend(true, {}, defaults, options);\n this.sessionManager = new SessionManager(options);\n this.options = this.sessionManager.getMergedOptions();\n\n this.authAdapter = new AuthAdapter(this.options);\n}\n\nvar _findUserInGroup = function (members, id) {\n for (var j = 0; j < members.length; j++) {\n if (members[j].userId === id) {\n return members[j];\n }\n }\n return null;\n};\n\nAuthManager.prototype = $.extend(AuthManager.prototype, {\n\n /**\n * Logs user in.\n *\n * **Example**\n *\n * authMgr.login({\n * account: 'acme-simulations',\n * project: 'supply-chain-game',\n * userName: 'enduser1',\n * password: 'passw0rd'\n * })\n * .then(function(statusObj) {\n * // if enduser1 belongs to exactly one group\n * // (or if the login() call is modified to include the group id)\n * // continue here\n * })\n * .fail(function(statusObj) {\n * // if enduser1 belongs to multiple groups,\n * // the login() call fails\n * // and returns all groups of which the user is a member\n * for (var i=0; i < statusObj.userGroups.length; i++) {\n * console.log(statusObj.userGroups[i].name, statusObj.userGroups[i].groupId);\n * }\n * });\n *\n * **Parameters**\n *\n * @param {Object} options (Optional) Overrides for configuration options. If not passed in when creating an instance of the manager (`F.manager.AuthManager()`), these options should include:\n * @param {string} options.account The account id for this `userName`. In the Epicenter UI, this is the **Team ID** (for team projects) or the **User ID** (for personal projects).\n * @param {string} options.userName Email or username to use for logging in.\n * @param {string} options.password Password for specified `userName`.\n * @param {string} options.project (Optional) The **Project ID** for the project to log this user into.\n * @param {string} options.groupId The id of the group to which `userName` belongs. Required for [end users](../../../glossary/#users) if the `project` is specified and if the end users are members of multiple [groups](../../../glossary/#groups), otherwise optional.\n * @return {Promise}\n */\n login: function (options) {\n var me = this;\n var $d = $.Deferred();\n var sessionManager = this.sessionManager;\n var adapterOptions = sessionManager.getMergedOptions({ success: $.noop, error: $.noop }, options);\n var outSuccess = adapterOptions.success;\n var outError = adapterOptions.error;\n var groupId = adapterOptions.groupId;\n\n var decodeToken = function (token) {\n var encoded = token.split('.')[1];\n while (encoded.length % 4 !== 0) { //eslint-disable-line\n encoded += '=';\n }\n return JSON.parse(atob(encoded));\n };\n\n var handleGroupError = function (message, statusCode, data, type) {\n // logout the user since it's in an invalid state with no group selected\n me.logout().then(function () {\n var error = $.extend(true, {}, data, { statusText: message, status: statusCode, type: type });\n $d.reject(error);\n });\n };\n\n var handleSuccess = function (response) {\n var token = response.access_token;\n var userInfo = decodeToken(token);\n var oldGroups = sessionManager.getSession(adapterOptions).groups || {};\n var userGroupOpts = $.extend(true, {}, adapterOptions, { success: $.noop });\n var data = { auth: response, user: userInfo };\n var project = adapterOptions.project;\n var isTeamMember = userInfo.parent_account_id === null;\n var requiresGroup = adapterOptions.requiresGroup && project;\n\n var userName = (userInfo.user_name || '').split('/')[0]; //of form /\n var sessionInfo = {\n auth_token: token,\n account: adapterOptions.account,\n project: project,\n userId: userInfo.user_id,\n groups: oldGroups,\n isTeamMember: isTeamMember,\n userName: userName,\n };\n // The group is not required if the user is not logging into a project\n if (!requiresGroup) {\n sessionManager.saveSession(sessionInfo);\n outSuccess.apply(this, [data]);\n $d.resolve(data);\n return;\n }\n\n var handleGroupList = function (groupList) {\n data.userGroups = groupList;\n\n var group = null;\n if (groupList.length === 0) {\n handleGroupError('The user has no groups associated in this account', 403, data, 'NO_GROUPS');\n return;\n } else if (groupList.length === 1) {\n // Select the only group\n group = groupList[0];\n } else if (groupList.length > 1) {\n if (groupId) {\n var filteredGroups = $.grep(groupList, function (resGroup) {\n return resGroup.groupId === groupId;\n });\n group = filteredGroups.length === 1 ? filteredGroups[0] : null;\n }\n }\n\n if (group) {\n // A team member does not get the group members because is calling the Group API\n // but it's automatically a fac user\n var isFac = isTeamMember ? true : _findUserInGroup(group.members, userInfo.user_id).role === 'facilitator';\n var groupData = {\n groupId: group.groupId,\n groupName: group.name,\n isFac: isFac\n };\n var sessionInfoWithGroup = objectAssign({}, sessionInfo, groupData);\n sessionInfo.groups[project] = groupData;\n me.sessionManager.saveSession(sessionInfoWithGroup, adapterOptions);\n outSuccess.apply(this, [data]);\n $d.resolve(data);\n } else {\n handleGroupError('This user is associated with more than one group. Please specify a group id to log into and try again', 403, data, 'MULTIPLE_GROUPS');\n }\n };\n\n if (!isTeamMember) {\n me.getUserGroups({ userId: userInfo.user_id, token: token }, userGroupOpts)\n .then(handleGroupList, $d.reject);\n } else {\n var opts = objectAssign({}, userGroupOpts, { token: token });\n var groupService = new GroupService(opts);\n groupService.getGroups({ account: adapterOptions.account, project: project })\n .then(function (groups) {\n // Group API returns id instead of groupId\n groups.forEach(function (group) {\n group.groupId = group.id;\n });\n\n if (groups.length) {\n handleGroupList(groups);\n } else {\n //either it's a private project or there are no groups\n sessionManager.saveSession(sessionInfo);\n outSuccess.apply(this, [data]);\n $d.resolve(data);\n return;\n }\n }, $d.reject);\n }\n };\n\n adapterOptions.success = handleSuccess;\n adapterOptions.error = function (response) {\n if (adapterOptions.account) {\n // Try to login as a system user\n adapterOptions.account = null;\n adapterOptions.error = function () {\n outError.apply(this, arguments);\n $d.reject(response);\n };\n\n me.authAdapter.login(adapterOptions);\n return;\n }\n\n outError.apply(this, arguments);\n $d.reject(response);\n };\n\n this.authAdapter.login(adapterOptions);\n return $d.promise();\n },\n\n /**\n * Logs user out by clearing all session information.\n *\n * **Example**\n *\n * authMgr.logout();\n *\n * **Parameters**\n *\n * @param {Object} options (Optional) Overrides for configuration options.\n * @return {Promise}\n */\n logout: function (options) {\n var me = this;\n var adapterOptions = this.sessionManager.getMergedOptions(options);\n\n var removeCookieFn = function (response) {\n me.sessionManager.removeSession();\n };\n\n return this.authAdapter.logout(adapterOptions).then(removeCookieFn);\n },\n\n /**\n * Returns the existing user access token if the user is already logged in. Otherwise, logs the user in, creating a new user access token, and returns the new token. (See [more background on access tokens](../../../project_access/)).\n *\n * **Example**\n *\n * authMgr.getToken()\n * .then(function (token) {\n * console.log('My token is ', token);\n * });\n *\n * **Parameters**\n * @param {Object} options (Optional) Overrides for configuration options.\n * @return {Promise}\n */\n getToken: function (options) {\n var httpOptions = this.sessionManager.getMergedOptions(options);\n\n var session = this.sessionManager.getSession(httpOptions);\n var $d = $.Deferred();\n if (session.auth_token) {\n $d.resolve(session.auth_token);\n } else {\n this.login(httpOptions).then($d.resolve);\n }\n return $d.promise();\n },\n\n /**\n * Returns an array of group records, one for each group of which the current user is a member. Each group record includes the group `name`, `account`, `project`, and `groupId`.\n *\n * If some end users in your project are members of multiple groups, this is a useful method to call on your project's login page. When the user attempts to log in, you can use this to display the groups of which the user is member, and have the user select the correct group to log in to for this session.\n *\n * **Example**\n *\n * // get groups for current user\n * var sessionObj = authMgr.getCurrentUserSessionInfo();\n * authMgr.getUserGroups({ userId: sessionObj.userId, token: sessionObj.auth_token })\n * .then(function (groups) {\n * for (var i=0; i < groups.length; i++)\n * { console.log(groups[i].name); }\n * });\n *\n * // get groups for particular user\n * authMgr.getUserGroups({userId: 'b1c19dda-2d2e-4777-ad5d-3929f17e86d3', token: savedProjAccessToken });\n *\n * **Parameters**\n * @param {Object} params Object with a userId and token properties.\n * @param {String} params.userId The userId. If looking up groups for the currently logged in user, this is in the session information. Otherwise, pass a string.\n * @param {String} params.token The authorization credentials (access token) to use for checking the groups for this user. If looking up groups for the currently logged in user, this is in the session information. A team member's token or a project access token can access all the groups for all end users in the team or project.\n * @param {Object} options (Optional) Overrides for configuration options.\n * @return {Promise}\n */\n getUserGroups: function (params, options) {\n var adapterOptions = this.sessionManager.getMergedOptions({ success: $.noop }, options);\n var $d = $.Deferred();\n var outSuccess = adapterOptions.success;\n\n adapterOptions.success = function (memberInfo) {\n // The member API is at the account scope, we filter by project\n if (adapterOptions.project) {\n memberInfo = $.grep(memberInfo, function (group) {\n return group.project === adapterOptions.project;\n });\n }\n\n outSuccess.apply(this, [memberInfo]);\n $d.resolve(memberInfo);\n };\n\n var memberAdapter = new MemberAdapter({ token: params.token, server: adapterOptions.server });\n memberAdapter.getGroupsForUser(params, adapterOptions).fail($d.reject);\n return $d.promise();\n },\n\n /**\n * Helper method to check if you're currently logged in\n * @return {Boolean} true if you're logged in\n */\n isLoggedIn: function () {\n var session = this.getCurrentUserSessionInfo();\n return !!(session && session.userId);\n },\n\n /**\n * Returns session information for the current user, including the `userId`, `account`, `project`, `groupId`, `groupName`, `isFac` (whether the end user is a facilitator of this group), and `auth_token` (user access token).\n *\n * *Important*: This method is synchronous. The session information is returned immediately in an object; no callbacks or promises are needed.\n *\n * Session information is stored in a cookie in the browser.\n *\n * **Example**\n *\n * var sessionObj = authMgr.getCurrentUserSessionInfo();\n *\n * **Parameters**\n * @param {Object} options (Optional) Overrides for configuration options.\n * @return {Object} session information\n */\n getCurrentUserSessionInfo: function (options) {\n var adapterOptions = this.sessionManager.getMergedOptions({ success: $.noop }, options);\n return this.sessionManager.getSession(adapterOptions);\n },\n\n /*\n * Adds one or more groups to the current session. \n *\n * This method assumes that the project and group exist and the user specified in the session is part of this project and group.\n *\n * Returns the new session object.\n *\n * **Example**\n *\n * authMgr.addGroups({ project: 'hello-world', groupName: 'groupName', groupId: 'groupId' });\n * authMgr.addGroups([{ project: 'hello-world', groupName: 'groupName', groupId: 'groupId' }, { project: 'hello-world', groupName: '...' }]);\n *\n * **Parameters**\n * @param {object|array} groups (Required) The group object must contain the `project` (**Project ID**) and `groupName` properties. If passing an array of such objects, all of the objects must contain *different* `project` (**Project ID**) values: although end users may be logged in to multiple projects at once, they may only be logged in to one group per project at a time.\n * @param {string} group.isFac (optional) Defaults to `false`. Set to `true` if the user in the session should be a facilitator in this group.\n * @param {string} group.groupId (optional) Defaults to undefined. Needed mostly for the Members API.\n * @return {Object} session information\n */\n addGroups: function (groups) {\n var session = this.getCurrentUserSessionInfo();\n var isArray = Array.isArray(groups);\n groups = isArray ? groups : [groups];\n\n $.each(groups, function (index, group) {\n var extendedGroup = $.extend({}, { isFac: false }, group);\n var project = extendedGroup.project;\n var validProps = ['groupName', 'groupId', 'isFac'];\n if (!project || !extendedGroup.groupName) {\n throw new Error('No project or groupName specified.');\n }\n // filter object\n extendedGroup = _pick(extendedGroup, validProps);\n session.groups[project] = extendedGroup;\n });\n this.sessionManager.saveSession(session);\n return session;\n }\n});\n\nmodule.exports = AuthManager;\n","'use strict';\n\n/**\n * ## Channel Manager\n *\n * There are two main use cases for the channel: event notifications and chat messages.\n *\n * If you are developing with Epicenter.js, you should use the [Epicenter Channel Manager](../epicenter-channel-manager/) rather than this more generic Channel Manager. (The Epicenter Channel Manager is a wrapper that instantiates a Channel Manager with Epicenter-specific defaults.) The Epicenter Channel Manager documentation also has more [background](../epicenter-channel-manager/#background) information on channels and their use. \n *\n * However, you can work directly with the Channel Manager if you like. (This might be useful if you are working through Node.js, for example, `require('manager/channel-manager')`.)\n *\n * The Channel Manager is a wrapper around the default [cometd JavaScript library](http://docs.cometd.org/2/reference/javascript.html), `$.cometd`. It provides a few nice features that `$.cometd` doesn't, including:\n *\n * * Automatic re-subscription to channels if you lose your connection\n * * Online / Offline notifications\n * * 'Events' for cometd notifications (instead of having to listen on specific meta channels)\n *\n * You'll need to include the `epicenter-multiplayer-dependencies.js` library in addition to the `epicenter.js` library in your project to use the Channel Manager. (See [Including Epicenter.js](../../#include).)\n *\n * To use the Channel Manager in client-side JavaScript, instantiate the [Epicenter Channel Manager](../epicenter-channel-manager/), get a particular channel -- that is, an instance of a [Channel Service](../channel-service/) -- then use the channel's `subscribe()` and `publish()` methods to subscribe to topics or publish data to topics.\n *\n * var cm = new F.manager.ChannelManager();\n * var gc = cm.getGroupChannel();\n * // because we used an Epicenter Channel Manager to get the group channel,\n * // subscribe() and publish() here default to the base topic for the group;\n * gc.subscribe('', function(data) { console.log(data); });\n * gc.publish('', { message: 'a new message to the group' });\n *\n * The parameters for instantiating a Channel Manager include:\n *\n * * `options` The options object to configure the Channel Manager. Besides the common options listed here, see http://docs.cometd.org/reference/javascript.html for other supported options.\n * * `options.url` The Cometd endpoint URL.\n * * `options.websocketEnabled` Whether websocket support is active (boolean).\n * * `options.channel` Other defaults to pass on to instances of the underlying Channel Service. See [Channel Service](../channel-service/) for details.\n *\n */\n\nvar Channel = require('../service/channel-service');\nvar SessionManager = require('../store/session-manager');\n\nvar ChannelManager = function (options) {\n if (!$.cometd) {\n throw new Error('Cometd library not found. Please include epicenter-multiplayer-dependencies.js');\n }\n if (!options || !options.url) {\n throw new Error('Please provide an url for the cometd server');\n }\n\n var defaults = {\n /**\n * The Cometd endpoint URL.\n * @type {string}\n */\n url: '',\n\n /**\n * The log level for the channel (logs to console).\n * @type {string}\n */\n logLevel: 'info',\n\n /**\n * Whether websocket support is active. Defaults to `true`.\n * @type {boolean}\n */\n websocketEnabled: true,\n\n /**\n * Whether the ACK extension is enabled. Defaults to `true`. See [https://docs.cometd.org/current/reference/#_extensions_acknowledge](https://docs.cometd.org/current/reference/#_extensions_acknowledge) for more info.\n * @type {boolean}\n */\n ackEnabled: true,\n\n /**\n * If false each instance of Channel will have a separate cometd connection to server, which could be noisy. Set to true to re-use the same connection across instances.\n * @type {boolean}\n */\n shareConnection: false,\n\n /**\n * Other defaults to pass on to instances of the underlying [Channel Service](../channel-service/), which are created through `getChannel()`.\n * @type {object}\n */\n channel: {\n\n },\n\n /**\n * Options to pass to the channel handshake.\n *\n * For example, the [Epicenter Channel Manager](../epicenter-channel-manager/) passes `ext` and authorization information. More information on possible options is in the details of the underlying [Push Channel API](../../../rest_apis/multiplayer/channel/).\n *\n * @type {object}\n */\n handshake: undefined\n };\n this.sessionManager = new SessionManager();\n var defaultCometOptions = this.sessionManager.getMergedOptions(defaults, options);\n this.currentSubscriptions = [];\n this.options = defaultCometOptions;\n\n if (defaultCometOptions.shareConnection && ChannelManager.prototype._cometd) {\n this.cometd = ChannelManager.prototype._cometd;\n return this;\n }\n var cometd = new $.CometD();\n ChannelManager.prototype._cometd = cometd;\n\n cometd.websocketEnabled = defaultCometOptions.websocketEnabled;\n cometd.ackEnabled = defaultCometOptions.ackEnabled;\n\n this.isConnected = false;\n var connectionBroken = function (message) {\n $(this).trigger('disconnect', message);\n };\n var connectionSucceeded = function (message) {\n $(this).trigger('connect', message);\n };\n var me = this;\n\n cometd.configure(defaultCometOptions);\n\n cometd.addListener('/meta/connect', function (message) {\n var wasConnected = this.isConnected;\n this.isConnected = (message.successful === true);\n if (!wasConnected && this.isConnected) { //Connecting for the first time\n connectionSucceeded.call(this, message);\n } else if (wasConnected && !this.isConnected) { //Only throw disconnected message fro the first disconnect, not once per try\n connectionBroken.call(this, message);\n }\n }.bind(this));\n\n cometd.addListener('/meta/disconnect', connectionBroken);\n\n cometd.addListener('/meta/handshake', function (message) {\n if (message.successful) {\n //http://docs.cometd.org/reference/javascript_subscribe.html#javascript_subscribe_meta_channels\n // ^ \"dynamic subscriptions are cleared (like any other subscription) and the application needs to figure out which dynamic subscription must be performed again\"\n cometd.batch(function () {\n $(me.currentSubscriptions).each(function (index, subs) {\n cometd.resubscribe(subs);\n });\n });\n }\n });\n\n //Other interesting events for reference\n cometd.addListener('/meta/subscribe', function (message) {\n $(me).trigger('subscribe', message);\n });\n cometd.addListener('/meta/unsubscribe', function (message) {\n $(me).trigger('unsubscribe', message);\n });\n cometd.addListener('/meta/publish', function (message) {\n $(me).trigger('publish', message);\n });\n cometd.addListener('/meta/unsuccessful', function (message) {\n $(me).trigger('error', message);\n });\n\n cometd.handshake(defaultCometOptions.handshake);\n\n this.cometd = cometd;\n};\n\n\nChannelManager.prototype = $.extend(ChannelManager.prototype, {\n\n /**\n * Creates and returns a channel, that is, an instance of a [Channel Service](../channel-service/).\n *\n * **Example**\n *\n * var cm = new F.manager.ChannelManager();\n * var channel = cm.getChannel();\n *\n * channel.subscribe('topic', callback);\n * channel.publish('topic', { myData: 100 });\n *\n * **Parameters**\n * @param {Object|String} options (Optional) If string, assumed to be the base channel url. If object, assumed to be configuration options for the constructor.\n * @return {Channel} Channel instance\n */\n getChannel: function (options) {\n //If you just want to pass in a string\n if (options && !$.isPlainObject(options)) {\n options = {\n base: options\n };\n }\n var defaults = {\n transport: this.cometd\n };\n var channel = new Channel($.extend(true, {}, this.options.channel, defaults, options));\n\n\n //Wrap subs and unsubs so we can use it to re-attach handlers after being disconnected\n var subs = channel.subscribe;\n channel.subscribe = function () {\n var subid = subs.apply(channel, arguments);\n this.currentSubscriptions = this.currentSubscriptions.concat(subid);\n return subid;\n }.bind(this);\n\n\n var unsubs = channel.unsubscribe;\n channel.unsubscribe = function () {\n var removed = unsubs.apply(channel, arguments);\n for (var i = 0; i < this.currentSubscriptions.length; i++) {\n if (this.currentSubscriptions[i].id === removed.id) {\n this.currentSubscriptions.splice(i, 1);\n }\n }\n return removed;\n }.bind(this);\n\n return channel;\n },\n\n /**\n * Start listening for events on this instance. Signature is same as for jQuery Events: http://api.jquery.com/on/.\n *\n * Supported events are: `connect`, `disconnect`, `subscribe`, `unsubscribe`, `publish`, `error`.\n *\n * **Parameters**\n *\n * @param {string} event The event type. See more detail at jQuery Events: http://api.jquery.com/on/.\n */\n on: function (event) {\n $(this).on.apply($(this), arguments);\n },\n\n /**\n * Stop listening for events on this instance. Signature is same as for jQuery Events: http://api.jquery.com/off/.\n *\n * **Parameters**\n *\n * @param {string} event The event type. See more detail at jQuery Events: http://api.jquery.com/off/.\n */\n off: function (event) {\n $(this).off.apply($(this), arguments);\n },\n\n /**\n * Trigger events and execute handlers. Signature is same as for jQuery Events: http://api.jquery.com/trigger/.\n *\n * **Parameters**\n *\n * @param {string} event The event type. See more detail at jQuery Events: http://api.jquery.com/trigger/.\n */\n trigger: function (event) {\n $(this).trigger.apply($(this), arguments);\n }\n});\n\nmodule.exports = ChannelManager;\n","'use strict';\n\n/**\n * ## Epicenter Channel Manager\n *\n * The Epicenter platform provides a push channel, which allows you to publish and subscribe to messages within a [project](../../../glossary/#projects), [group](../../../glossary/#groups), or [multiplayer world](../../../glossary/#world).\n *\n * \n * ### Channel Background\n *\n * Channel notifications are only available for [team projects](../../../glossary/#team). There are two main use cases for the push channel: event notifications and chat messages.\n *\n * #### Event Notifications\n *\n * Within a [multiplayer simulation or world](../../../glossary/#world), it is often useful for your project's [model](../../../writing_your_model/) to alert the [user interface (browser)](../../../creating_your_interface/) that something new has happened.\n *\n * Usually, this \"something new\" is an event within the project, group, or world, such as:\n *\n * * An end user comes online (logs in) or goes offline. (This is especially interesting in a multiplayer world; only available if you have [enabled authorization](../../../updating_your_settings/#general-settings) for the channel.)\n * * An end user is assigned to a world.\n * * An end user updates a variable / makes a decision.\n * * An end user creates or updates data stored in the [Data API](../data-api-service/).\n * * An operation (method) is called. (This is especially interesting if the model is advanced, for instance, the Vensim `step` operation is called.)\n *\n * When these events occur, you often want to have the user interface for one or more end users automatically update with new information.\n *\n * #### Chat Messages\n *\n * Another reason to use the push channel is to allow players (end users) to send chat messages to other players, and to have those messages appear immediately.\n *\n * #### Getting Started\n *\n * For both the event notification and chat message use cases:\n *\n * * First, enable channel notifications for your project.\n * * Channel notifications are only available for [team projects](../../../glossary/#team). To enable notifications for your project, [update your project settings](../../../updating_your_settings/#general-settings) to turn on the **Push Channel** setting, and optionally require authorization for the channel.\n * * Then, instantiate an Epicenter Channel Manager.\n * * Next, get the channel with the scope you want (user, world, group, data).\n * * Finally, use the channel's `subscribe()` and `publish()` methods to subscribe to topics or publish data to topics.\n *\n * Here's an example of those last three steps (instantiate, get channel, subscribe):\n *\n * var cm = new F.manager.ChannelManager();\n * var gc = cm.getGroupChannel();\n * gc.subscribe('', function(data) { console.log(data); });\n * gc.publish('', { message: 'a new message to the group' });\n *\n * For a more detailed example, see a [complete publish and subscribe example](../../../rest_apis/multiplayer/channel/#epijs-example).\n *\n * For details on what data is published automatically to which channels, see [Automatic Publishing of Events](../../../rest_apis/multiplayer/channel/#publish-message-auto).\n *\n * #### Creating an Epicenter Channel Manager\n *\n * The Epicenter Channel Manager is a wrapper around the (more generic) [Channel Manager](../channel-manager/), to instantiate it with Epicenter-specific defaults. If you are interested in including a notification or chat feature in your project, using an Epicenter Channel Manager is the easiest way to get started.\n *\n * You'll need to include the `epicenter-multiplayer-dependencies.js` library in addition to the `epicenter.js` library in your project to use the Epicenter Channel Manager. See [Including Epicenter.js](../../#include).\n *\n * The parameters for instantiating an Epicenter Channel Manager include:\n *\n * * `options` Object with details about the Epicenter project for this Epicenter Channel Manager instance.\n * * `options.account` The Epicenter account id (**Team ID** for team projects, **User ID** for personal projects).\n * * `options.project` Epicenter project id.\n * * `options.userName` Epicenter userName used for authentication.\n * * `options.userId` Epicenter user id used for authentication. Optional; `options.userName` is preferred.\n * * `options.token` Epicenter token used for authentication. (You can retrieve this using `authManager.getToken()` from the [Authorization Manager](../auth-manager/).)\n * * `options.allowAllChannels` If not included or if set to `false`, all channel paths are validated; if your project requires [Push Channel Authorization](../../../updating_your_settings/), you should use this option. If you want to allow other channel paths, set to `true`; this is not common.\n */\n\nvar ChannelManager = require('./channel-manager');\nvar ConfigService = require('../service/configuration-service');\nvar classFrom = require('../util/inherit');\nvar SessionManager = require('../store/session-manager');\n\nvar validTypes = {\n project: true,\n group: true,\n world: true,\n user: true,\n data: true,\n general: true,\n chat: true\n};\nvar getFromSessionOrError = function (value, sessionKeyName, settings) {\n if (!value) {\n if (settings && settings[sessionKeyName]) {\n value = settings[sessionKeyName];\n } else {\n throw new Error(sessionKeyName + ' not found. Please log-in again, or specify ' + sessionKeyName + ' explicitly');\n }\n }\n return value;\n};\n\nvar isPresenceData = function (payload) {\n return payload.data && payload.data.type === 'user' && payload.data.user;\n};\n\nvar __super = ChannelManager.prototype;\nvar EpicenterChannelManager = classFrom(ChannelManager, {\n constructor: function (options) {\n this.sessionManager = new SessionManager(options);\n var defaultCometOptions = this.sessionManager.getMergedOptions(options);\n\n var urlConfig = new ConfigService(defaultCometOptions).get('server');\n if (!defaultCometOptions.url) {\n defaultCometOptions.url = urlConfig.getAPIPath('channel');\n }\n\n if (defaultCometOptions.handshake === undefined) {\n var userName = defaultCometOptions.userName;\n var userId = defaultCometOptions.userId;\n var token = defaultCometOptions.token;\n if ((userName || userId) && token) {\n var userProp = userName ? 'userName' : 'userId';\n var ext = {\n authorization: 'Bearer ' + token\n };\n ext[userProp] = userName ? userName : userId;\n\n defaultCometOptions.handshake = {\n ext: ext\n };\n }\n }\n\n this.options = defaultCometOptions;\n return __super.constructor.call(this, defaultCometOptions);\n },\n\n /**\n * Creates and returns a channel, that is, an instance of a [Channel Service](../channel-service/).\n *\n * This method enforces Epicenter-specific channel naming: all channels requested must be in the form `/{type}/{account id}/{project id}/{...}`, where `type` is one of `run`, `data`, `user`, `world`, or `chat`.\n *\n * **Example**\n *\n * var cm = new F.manager.EpicenterChannelManager();\n * var channel = cm.getChannel('/group/acme/supply-chain-game/');\n *\n * channel.subscribe('topic', callback);\n * channel.publish('topic', { myData: 100 });\n *\n * **Parameters**\n * @param {Object|String} options (Optional) If string, assumed to be the base channel url. If object, assumed to be configuration options for the constructor.\n * @return {Channel} Channel instance\n */\n getChannel: function (options) {\n if (options && typeof options !== 'object') {\n options = {\n base: options\n };\n }\n var channelOpts = $.extend({}, this.options, options);\n var base = channelOpts.base;\n if (!base) {\n throw new Error('No base topic was provided');\n }\n\n if (!channelOpts.allowAllChannels) {\n var baseParts = base.split('/');\n var channelType = baseParts[1];\n if (baseParts.length < 4) { //eslint-disable-line\n throw new Error('Invalid channel base name, it must be in the form /{type}/{account id}/{project id}/{...}');\n }\n if (!validTypes[channelType]) {\n throw new Error('Invalid channel type');\n }\n }\n return __super.getChannel.apply(this, arguments);\n },\n\n /**\n * Create and return a publish/subscribe channel (from the underlying [Channel Manager](../channel-manager/)) for the given [group](../../../glossary/#groups). The group must exist in the account (team) and project provided.\n *\n * There are no notifications from Epicenter on this channel; all messages are user-originated.\n *\n * **Example**\n *\n * var cm = new F.manager.ChannelManager();\n * var gc = cm.getGroupChannel();\n * gc.subscribe('broadcasts', callback);\n *\n * **Return Value**\n *\n * * *Channel* Returns the channel (an instance of the [Channel Service](../channel-service/)).\n *\n * **Parameters**\n *\n * @param {String} groupName (Optional) Group to broadcast to. If not provided, picks up group from current session if end user is logged in.\n * @return {Channel} Channel instance\n */\n getGroupChannel: function (groupName) {\n var session = this.sessionManager.getMergedOptions(this.options);\n groupName = getFromSessionOrError(groupName, 'groupName', session);\n var account = getFromSessionOrError('', 'account', session);\n var project = getFromSessionOrError('', 'project', session);\n\n var baseTopic = ['/group', account, project, groupName].join('/');\n var channel = __super.getChannel.call(this, { base: baseTopic });\n var oldsubs = channel.subscribe;\n channel.subscribe = function (topic, callback, context, options) {\n var callbackWithoutPresenceData = function (payload) {\n if (!isPresenceData(payload)) {\n callback.call(context, payload);\n }\n };\n return oldsubs.call(channel, topic, callbackWithoutPresenceData, context, options);\n };\n return channel;\n },\n\n /**\n * Create and return a publish/subscribe channel (from the underlying [Channel Manager](../channel-manager/)) for the given [world](../../../glossary/#world).\n *\n * This is typically used together with the [World Manager](../world-manager).\n *\n * **Example**\n *\n * var cm = new F.manager.ChannelManager();\n * var worldManager = new F.manager.WorldManager({\n * account: 'acme-simulations',\n * project: 'supply-chain-game',\n * group: 'team1',\n * run: { model: 'model.eqn' }\n * });\n * worldManager.getCurrentWorld().then(function (worldObject, worldAdapter) {\n * var worldChannel = cm.getWorldChannel(worldObject);\n * worldChannel.subscribe('', function (data) {\n * console.log(data);\n * });\n * });\n *\n * **Return Value**\n *\n * * *Channel* Returns the channel (an instance of the [Channel Service](../channel-service/)).\n *\n * **Parameters**\n *\n * @param {String|Object} world The world object or id.\n * @param {String} groupName (Optional) Group the world exists in. If not provided, picks up group from current session if end user is logged in.\n * @return {Channel} Channel instance\n */\n getWorldChannel: function (world, groupName) {\n var worldid = ($.isPlainObject(world) && world.id) ? world.id : world;\n if (!worldid) {\n throw new Error('Please specify a world id');\n }\n var session = this.sessionManager.getMergedOptions(this.options);\n\n groupName = getFromSessionOrError(groupName, 'groupName', session);\n var account = getFromSessionOrError('', 'account', session);\n var project = getFromSessionOrError('', 'project', session);\n\n var baseTopic = ['/world', account, project, groupName, worldid].join('/');\n return __super.getChannel.call(this, { base: baseTopic });\n },\n\n /**\n * Create and return a publish/subscribe channel (from the underlying [Channel Manager](../channel-manager/)) for the current [end user](../../../glossary/#users) in that user's current [world](../../../glossary/#world).\n *\n * This is typically used together with the [World Manager](../world-manager). Note that this channel only gets notifications for worlds currently in memory. (See more background on [persistence](../../../run_persistence).)\n *\n * **Example**\n *\n * var cm = new F.manager.ChannelManager();\n * var worldManager = new F.manager.WorldManager({\n * account: 'acme-simulations',\n * project: 'supply-chain-game',\n * group: 'team1',\n * run: { model: 'model.eqn' }\n * });\n * worldManager.getCurrentWorld().then(function (worldObject, worldAdapter) {\n * var userChannel = cm.getUserChannel(worldObject);\n * userChannel.subscribe('', function (data) {\n * console.log(data);\n * });\n * });\n *\n *\n * **Return Value**\n *\n * * *Channel* Returns the channel (an instance of the [Channel Service](../channel-service/)).\n *\n * **Parameters**\n *\n * @param {String|Object} world World object or id.\n * @param {String|Object} user (Optional) User object or id. If not provided, picks up user id from current session if end user is logged in.\n * @param {String} groupName (Optional) Group the world exists in. If not provided, picks up group from current session if end user is logged in.\n * @return {Channel} Channel instance\n */\n getUserChannel: function (world, user, groupName) {\n var worldid = ($.isPlainObject(world) && world.id) ? world.id : world;\n if (!worldid) {\n throw new Error('Please specify a world id');\n }\n var session = this.sessionManager.getMergedOptions(this.options);\n\n var userid = ($.isPlainObject(user) && user.id) ? user.id : user;\n userid = getFromSessionOrError(userid, 'userId', session);\n groupName = getFromSessionOrError(groupName, 'groupName', session);\n\n var account = getFromSessionOrError('', 'account', session);\n var project = getFromSessionOrError('', 'project', session);\n\n var baseTopic = ['/user', account, project, groupName, worldid, userid].join('/');\n return __super.getChannel.call(this, { base: baseTopic });\n },\n\n /**\n * Create and return a publish/subscribe channel (from the underlying [Channel Manager](../channel-manager/)) that automatically tracks the presence of an [end user](../../../glossary/#users), that is, whether the end user is currently online in this group. Notifications are automatically sent when the end user comes online, and when the end user goes offline (not present for more than 2 minutes). Useful in multiplayer games for letting each end user know whether other users in their group are also online.\n *\n * Note that the presence channel is tracking all end users in a group. In particular, if the project additionally splits each group into [worlds](../world-manager/), this channel continues to show notifications for all end users in the group (not restricted by worlds).\n *\n * **Example**\n *\n * var cm = new F.manager.ChannelManager();\n * var pc = cm.getPresenceChannel();\n * pc.subscribe('', function (data) {\n * // 'data' is the entire message object to the channel;\n * // parse for information of interest\n * if (data.data.subType === 'disconnect') {\n * console.log('user ', data.data.user.userName, 'disconnected at ', data.data.date);\n * }\n * if (data.data.subType === 'connect') {\n * console.log('user ', data.data.user.userName, 'connected at ', data.data.date);\n * }\n * });\n *\n *\n * **Return Value**\n *\n * * *Channel* Returns the channel (an instance of the [Channel Service](../channel-service/)).\n *\n * **Parameters**\n *\n * @param {String} groupName (Optional) Group the end user is in. If not provided, picks up group from current session if end user is logged in.\n * @return {Channel} Channel instance\n */\n getPresenceChannel: function (groupName) {\n var session = this.sessionManager.getMergedOptions(this.options);\n groupName = getFromSessionOrError(groupName, 'groupName', session);\n var account = getFromSessionOrError('', 'account', session);\n var project = getFromSessionOrError('', 'project', session);\n\n var baseTopic = ['/group', account, project, groupName].join('/');\n var channel = __super.getChannel.call(this, { base: baseTopic });\n var oldsubs = channel.subscribe;\n channel.subscribe = function (topic, callback, context, options) {\n var callbackWithOnlyPresenceData = function (payload) {\n if (isPresenceData(payload)) {\n callback.call(context, payload);\n }\n };\n return oldsubs.call(channel, topic, callbackWithOnlyPresenceData, context, options);\n };\n return channel;\n },\n\n /**\n * Create and return a publish/subscribe channel (from the underlying [Channel Manager](../channel-manager/)) for the given collection. (The collection name is specified in the `root` argument when the [Data Service](../data-api-service/) is instantiated.) Must be one of the collections in this account (team) and project.\n *\n * There are automatic notifications from Epicenter on this channel when data is created, updated, or deleted in this collection. See more on [automatic messages to the data channel](../../../rest_apis/multiplayer/channel/#data-messages).\n *\n * **Example**\n *\n * var cm = new F.manager.ChannelManager();\n * var dc = cm.getDataChannel('survey-responses');\n * dc.subscribe('', function(data, meta) {\n * console.log(data);\n *\n * // meta.date is time of change,\n * // meta.subType is the kind of change: new, update, or delete\n * // meta.path is the full path to the changed data\n * console.log(meta);\n * });\n *\n * **Return Value**\n *\n * * *Channel* Returns the channel (an instance of the [Channel Service](../channel-service/)).\n *\n * **Parameters**\n *\n * @param {String} collection Name of collection whose automatic notifications you want to receive.\n * @return {Channel} Channel instance\n */\n getDataChannel: function (collection) {\n if (!collection) {\n throw new Error('Please specify a collection to listen on.');\n }\n\n var session = this.sessionManager.getMergedOptions(this.options);\n var account = getFromSessionOrError('', 'account', session);\n var project = getFromSessionOrError('', 'project', session);\n var baseTopic = ['/data', account, project, collection].join('/');\n var channel = __super.getChannel.call(this, { base: baseTopic });\n\n //TODO: Fix after Epicenter bug is resolved\n var oldsubs = channel.subscribe;\n channel.subscribe = function (topic, callback, context, options) {\n var callbackWithCleanData = function (payload) {\n var meta = {\n path: payload.channel,\n subType: payload.data.subType,\n date: payload.data.date,\n dataPath: payload.data.data.path,\n };\n var actualData = payload.data.data;\n if (actualData.data !== undefined) { //Delete notifications are one data-level behind of course\n actualData = actualData.data;\n }\n\n callback.call(context, actualData, meta);\n };\n return oldsubs.call(channel, topic, callbackWithCleanData, context, options);\n };\n\n return channel;\n }\n});\n\nmodule.exports = EpicenterChannelManager;\n","'use strict';\n\nmodule.exports = {\n EPI_SESSION_KEY: 'epicenterjs.session',\n STRATEGY_SESSION_KEY: 'epicenter-scenario'\n};","/**\n* ## Run Manager\n*\n* The Run Manager gives you access to runs for your project. This allows you to read and update variables, call operations, etc. Additionally, the Run Manager gives you control over run creation depending on run states. Specifically, you can select [run creation strategies (rules)](../strategies/) for which runs end users of your project work with when they log in to your project.\n*\n* There are many ways to create new runs, including the Epicenter.js [Run Service](../run-api-service/) and the RESFTful [Run API](../../../rest_apis/aggregate_run_api). However, for some projects it makes more sense to pick up where the user left off, using an existing run. And in some projects, whether to create a new run or use an existing one is conditional, for example based on characteristics of the existing run or your own knowledge about the model. The Run Manager provides this level of control: your call to `getRun()`, rather than always returning a new run, returns a run based on the strategy you've specified.\n*\n*\n* ### Using the Run Manager to create and access runs\n*\n* To use the Run Manager, instantiate it by passing in:\n*\n* * `run`: (required) Run object. Must contain:\n* * `account`: Epicenter account id (**Team ID** for team projects, **User ID** for personal projects).\n* * `project`: Epicenter project id.\n* * `model`: The name of your primary model file. (See more on [Writing your Model](../../../writing_your_model/).)\n* * `scope`: (optional) Scope object for the run, for example `scope.group` with value of the name of the group.\n* * `server`: (optional) An object with one field, `host`. The value of `host` is the string `api.forio.com`, the URI of the Forio server. This is automatically set, but you can pass it explicitly if desired. It is most commonly used for clarity when you are [hosting an Epicenter project on your own server](../../../how_to/self_hosting/).\n* * `files`: (optional) If and only if you are using a Vensim model and you have additional data to pass in to your model, you can optionally pass a `files` object with the names of the files, for example: `\"files\": {\"data\": \"myExtraData.xls\"}`. (See more on [Using External Data in Vensim](../../../model_code/vensim/vensim_example_xls/).)\n*\n* * `strategy`: (optional) Run creation strategy for when to create a new run and when to reuse an end user's existing run. This is *optional*; by default, the Run Manager selects `reuse-per-session`, or `reuse-last-initialized` if you also pass in an initial operation. See [below](#using-the-run-manager-to-access-and-register-strategies) for more information on strategies.\n*\n* * `strategyOptions`: (optional) Additional options passed directly to the [run creation strategy](../strategies/).\n*\n* * `sessionKey`: (optional) Name of browser cookie in which to store run information, including run id. Many conditional strategies, including the provided strategies, rely on this browser cookie to store the run id and help make the decision of whether to create a new run or use an existing one. The name of this cookie defaults to `epicenter-scenario` and can be set with the `sessionKey` parameter. This can also be a function which returns a string, if you'd like to control this at runtime.\n*\n*\n* After instantiating a Run Manager, make a call to `getRun()` whenever you need to access a run for this end user. The `RunManager.run` contains the instantiated [Run Service](../run-api-service/). The Run Service allows you to access variables, call operations, etc.\n*\n* **Example**\n*\n* var rm = new F.manager.RunManager({\n* run: {\n* account: 'acme-simulations',\n* project: 'supply-chain-game',\n* model: 'supply-chain-model.jl',\n* server: { host: 'api.forio.com' }\n* },\n* strategy: 'reuse-never',\n* sessionKey: 'epicenter-session'\n* });\n* rm.getRun()\n* .then(function(run) {\n* // the return value of getRun() is a run object\n* var thisRunId = run.id;\n* // the RunManager.run also contains the instantiated Run Service,\n* // so any Run Service method is valid here\n* rm.run.do('runModel');\n* })\n*\n*\n* ### Using the Run Manager to access and register strategies\n*\n* The `strategy` for a Run Manager describes when to create a new run and when to reuse an end user's existing run. The Run Manager is responsible for passing a strategy everything it might need to determine the 'correct' run, that is, how to find the best existing run and how to decide when to create a new run.\n*\n* There are several common strategies provided as part of Epicenter.js, which you can list by accessing `F.manager.RunManager.strategies`. You can also create your own strategies, and register them to use with Run Managers. See [Run Manager Strategies](../strategies/) for details.\n* \n*/\n\n'use strict';\nvar strategies = require('./run-strategies');\nvar specialOperations = require('./special-operations');\n\nvar RunService = require('../service/run-api-service');\nvar SessionManager = require('../store/session-manager');\n\nvar util = require('../util/object-util');\nvar keyNames = require('./key-names');\n\nfunction patchRunService(service, manager) {\n if (service.patched) {\n return service;\n }\n\n var orig = service.do;\n service.do = function (operation, params, options) {\n var reservedOps = Object.keys(specialOperations);\n if (reservedOps.indexOf(operation) === -1) {\n return orig.apply(service, arguments);\n } else {\n return specialOperations[operation].call(service, params, options, manager);\n }\n };\n\n service.patched = true;\n\n return service;\n}\n\nfunction sessionKeyFromOptions(options, runService) {\n var config = runService.getCurrentConfig();\n var sessionKey = $.isFunction(options.sessionKey) ? options.sessionKey(config) : options.sessionKey;\n return sessionKey;\n}\n\nfunction setRunInSession(sessionKey, run, sessionManager) {\n if (sessionKey) {\n delete run.variables;\n sessionManager.getStore().set(sessionKey, JSON.stringify(run));\n }\n}\n\nvar defaults = {\n sessionKey: function (config) { \n var baseKey = keyNames.STRATEGY_SESSION_KEY;\n var key = ['account', 'project', 'model'].reduce(function (accum, key) {\n return config[key] ? accum + '-' + config[key] : accum; \n }, baseKey);\n return key;\n }\n};\n\nfunction RunManager(options) {\n this.options = $.extend(true, {}, defaults, options);\n\n if (this.options.run instanceof RunService) {\n this.run = this.options.run;\n } else if (!util.isEmpty(this.options.run)) {\n this.run = new RunService(this.options.run);\n } else {\n throw new Error('No run options passed to RunManager');\n }\n patchRunService(this.run, this);\n\n this.strategy = strategies.getBestStrategy(this.options);\n this.sessionManager = new SessionManager(this.options);\n}\n\nRunManager.prototype = {\n /**\n * Returns the run object for the 'correct' run. The correct run is defined by the strategy. \n *\n * For example, if the strategy is `reuse-never`, the call\n * to `getRun()` always returns a newly created run; if the strategy is `reuse-per-session`,\n * `getRun()` returns the run currently referenced in the browser cookie, and if there is none, creates a new run. \n * See [Run Manager Strategies](../strategies/) for more on strategies.\n *\n * **Example**\n *\n * rm.getRun().then(function (run) {\n * // use the run object\n * var thisRunId = run.id;\n *\n * // use the Run Service object\n * rm.run.do('runModel');\n * });\n *\n * rm.getRun(['sample_int']).then(function (run) {\n * // an object whose fields are the name : value pairs of the variables passed to getRun()\n * console.log(run.variables);\n * // the value of sample_int\n * console.log(run.variables.sample_int); \n * });\n *\n * @param {Array} variables (Optional) The run object is populated with the provided model variables, if provided. Note: `getRun()` does not throw an error if you try to get a variable which doesn't exist. Instead, the variables list is empty, and any errors are logged to the console.\n * @param {Object} options (Optional) Configuration options; passed on to [RunService#create](../run-api-service/#create) if the strategy does create a new run.\n * @return {$promise} Promise to complete the call.\n */\n getRun: function (variables, options) {\n var me = this;\n var sessionStore = this.sessionManager.getStore();\n\n var sessionContents = sessionStore.get(sessionKeyFromOptions(this.options, me.run));\n var runSession = JSON.parse(sessionContents || '{}');\n \n if (runSession.runId) {\n //EpiJS < 2.2 used runId as key, so maintain comptaibility. Remove at some future date (Summer `17?)\n runSession.id = runSession.runId;\n }\n\n var authSession = this.sessionManager.getSession();\n if (this.strategy.requiresAuth && util.isEmpty(authSession)) {\n console.error('No user-session available', this.options.strategy, 'requires authentication.');\n return $.Deferred().reject('No user-session available').promise();\n }\n return this.strategy\n .getRun(this.run, authSession, runSession, options).then(function (run) {\n if (run && run.id) {\n me.run.updateConfig({ filter: run.id });\n var sessionKey = sessionKeyFromOptions(me.options, me.run);\n setRunInSession(sessionKey, run, me.sessionManager);\n\n if (variables && variables.length) {\n return me.run.variables().query(variables).then(function (results) {\n run.variables = results;\n return run;\n }).catch(function (err) {\n run.variables = {};\n console.error(err);\n return run;\n });\n }\n }\n return run;\n });\n },\n\n /**\n * Returns the run object for a 'reset' run. The definition of a reset is defined by the strategy, but typically means forcing the creation of a new run. For example, `reset()` for the default strategies `reuse-per-session` and `reuse-last-initialized` both create new runs.\n *\n * **Example**\n *\n * rm.reset().then(function (run) {\n * // use the (new) run object\n * var thisRunId = run.id;\n *\n * // use the Run Service object\n * rm.run.do('runModel');\n * });\n *\n * **Parameters**\n * @param {Object} options (Optional) Configuration options; passed on to [RunService#create](../run-api-service/#create).\n * @return {Promise}\n */\n reset: function (options) {\n var me = this;\n var authSession = this.sessionManager.getSession();\n if (this.strategy.requiresAuth && util.isEmpty(authSession)) {\n console.error('No user-session available', this.options.strategy, 'requires authentication.');\n return $.Deferred().reject('No user-session available').promise();\n }\n return this.strategy.reset(this.run, authSession, options).then(function (run) {\n if (run && run.id) {\n me.run.updateConfig({ filter: run.id });\n var sessionKey = sessionKeyFromOptions(me.options, me.run);\n setRunInSession(sessionKey, run.id, me.sessionManager);\n }\n return run;\n });\n }\n};\n\nRunManager.strategies = strategies;\nmodule.exports = RunManager;\n","'use strict';\n\nvar Base = require('./none-strategy');\nvar classFrom = require('../../util/inherit');\n\n/**\n* ## Conditional Creation Strategy\n*\n* This strategy will try to get the run stored in the cookie and\n* evaluate if it needs to create a new run by calling the `condition` function.\n*/\n\nvar Strategy = classFrom(Base, {\n constructor: function Strategy(condition) {\n if (condition == null) { //eslint-disable-line\n throw new Error('Conditional strategy needs a condition to create a run');\n }\n this.condition = typeof condition !== 'function' ? function () { return condition; } : condition;\n },\n\n /**\n * Gets a new 'correct' run, or updates the existing one (the definition of 'correct' depends on strategy implementation).\n * @param {RunService} runService A Run Service instance for the current run, as determined by the Run Manager.\n * @param {Object} userSession Information about the current user session. See [AuthManager#getCurrentUserSessionInfo](../auth-manager/#getcurrentusersessioninfo) for format.\n * @param {Object} options (Optional) See [RunService#create](../run-api-service/#create) for supported options.\n * @return {Promise} \n */\n reset: function (runService, userSession, options) {\n var group = userSession && userSession.groupName;\n var opt = $.extend({\n scope: { group: group }\n }, runService.getCurrentConfig());\n\n return runService\n .create(opt, options)\n .then(function (run) {\n run.freshlyCreated = true;\n return run;\n });\n },\n\n /**\n * Gets the 'correct' run (the definition of 'correct' depends on strategy implementation).\n * @param {RunService} runService A Run Service instance for the current run, as determined by the Run Manager.\n * @param {Object} userSession Information about the current user session. See [AuthManager#getCurrentUserSessionInfo](../auth-manager/#getcurrentusersessioninfo) for format.\n * @param {Object} runSession The Run Manager stores the 'last accessed' run in a cookie and passes it back here.\n * @param {Object} options (Optional) See [RunService#create](../run-api-service/#create) for supported options.\n * @return {Promise} \n */\n getRun: function (runService, userSession, runSession, options) {\n var me = this;\n if (runSession && runSession.id) {\n return this.loadAndCheck(runService, userSession, runSession, options).catch(function () {\n return me.reset(runService, userSession, options); //if it got the wrong cookie for e.g.\n });\n } else {\n return this.reset(runService, userSession, options);\n }\n },\n\n loadAndCheck: function (runService, userSession, runSession, options) {\n var shouldCreate = false;\n var me = this;\n\n return runService\n .load(runSession.id, null, {\n success: function (run, msg, headers) {\n shouldCreate = me.condition(run, headers, userSession, runSession);\n }\n })\n .then(function (run) {\n if (shouldCreate) {\n return me.reset(runService, userSession, options);\n }\n return run;\n });\n }\n});\n\nmodule.exports = Strategy;\n","/**\n * The `new-if-initialized` strategy creates a new run if the current one is in memory or has its `initialized` field set to `true`. The `initialized` field in the run record is automatically set to `true` at run creation, but can be changed.\n * \n * This strategy is useful if your project is structured such that immediately after a run is created, the model is executed completely (for example, a Vensim model is stepped to the end). It is similar to the `new-if-missing` strategy, except that it checks a field of the run record.\n * \n * Specifically, the strategy is:\n *\n * * Check the `sessionKey` cookie. \n * * This cookie is set by the [Run Manager](../run-manager/) and configurable through its options.\n * * If the cookie exists, check whether the run is in memory or only persisted in the database. Additionally, check whether the run's `initialized` field is `true`. \n * * If the run is in memory, create a new run.\n * * If the run's `initialized` field is `true`, create a new run.\n * * Otherwise, use the existing run.\n * * If the cookie does not exist, create a new run for this end user.\n * \n * @deprecated Consider using `reuse-last-initialized` instead\n */\n\n'use strict';\nvar classFrom = require('../../../util/inherit');\nvar ConditionalStrategy = require('../conditional-creation-strategy');\n\nvar __super = ConditionalStrategy.prototype;\n\nvar Strategy = classFrom(ConditionalStrategy, {\n constructor: function (options) {\n __super.constructor.call(this, this.createIf, options);\n console.warn('This strategy is deprecated; all runs now default to being initialized by default making this redundant. Consider using `reuse-last-initialized` instead.');\n },\n\n createIf: function (run, headers) {\n return headers.getResponseHeader('pragma') === 'persistent' || run.initialized;\n }\n});\n\nmodule.exports = Strategy;\n","/**\n * The `new-if-persisted` strategy creates a new run when the current one becomes persisted (end user is idle for a set period), but otherwise uses the current one. \n * \n * Using this strategy means that when end users navigate between pages in your project, or refresh their browsers, they will still be working with the same run. \n * \n * However, if they are idle for longer than your project's **Model Session Timeout** (configured in your project's [Settings](../../../updating_your_settings/)), then their run is persisted; the next time they interact with the project, they will get a new run. (See more background on [Run Persistence](../../../run_persistence/).)\n * \n * This strategy is useful for multi-page projects where end users play through a simulation in one sitting, stepping through the model sequentially (for example, a Vensim model that uses the `step` operation) or calling specific functions until the model is \"complete.\" However, you will need to guarantee that your end users will remain engaged with the project from beginning to end — or at least, that if they are idle for longer than the **Model Session Timeout**, it is okay for them to start the project from scratch (with an uninitialized model). \n * \n * Specifically, the strategy is:\n *\n * * Check the `sessionKey` cookie.\n * * This cookie is set by the [Run Manager](../run-manager/) and configurable through its options.\n * * If the cookie exists, check whether the run is in memory or only persisted in the database. \n * * If the run is in memory, use the run.\n * * If the run is only persisted (and not still in memory), create a new run for this end user.\n * * If the cookie does not exist, create a new run for this end user.\n *\n * @deprecated The run-service now sets a header to automatically bring back runs into memory\n */\n\n'use strict';\nvar classFrom = require('../../../util/inherit');\nvar ConditionalStrategy = require('../conditional-creation-strategy');\n\nvar __super = ConditionalStrategy.prototype;\n\nvar Strategy = classFrom(ConditionalStrategy, {\n constructor: function (options) {\n __super.constructor.call(this, this.createIf, options);\n console.warn('This strategy is deprecated; the run-service now sets a header to automatically bring back runs into memory');\n },\n\n createIf: function (run, headers) {\n return headers.getResponseHeader('pragma') === 'persistent';\n }\n});\n\nmodule.exports = Strategy;\n","/**\n * ### Working with Run Strategies\n *\n * You can access a list of available strategies using `F.manager.RunManager.strategies.list`. You can also ask for a particular strategy by name.\n *\n * If you decide to [create your own run strategy](#create-your-own), you can register your strategy. Registering your strategy means that:\n *\n * * You can pass the strategy by name to a Run Manager (as opposed to passing the strategy function): `new F.manager.RunManager({ strategy: 'mynewname'})`.\n * * You can pass configuration options to your strategy.\n * * You can specify whether or not your strategy requires authorization (a valid user session) to work.\n */\n\n\nvar list = {\n 'conditional-creation': require('./conditional-creation-strategy'),\n 'new-if-initialized': require('./deprecated/new-if-initialized-strategy'), //deprecated\n 'new-if-persisted': require('./deprecated/new-if-persisted-strategy'), //deprecated\n\n none: require('./none-strategy'),\n\n multiplayer: require('./multiplayer-strategy'),\n 'reuse-never': require('./reuse-never'),\n 'reuse-per-session': require('./reuse-per-session'),\n 'reuse-across-sessions': require('./reuse-across-sessions'),\n 'reuse-last-initialized': require('./reuse-last-initialized'),\n};\n\n//Add back older aliases\nlist['always-new'] = list['reuse-never'];\nlist['new-if-missing'] = list['reuse-per-session'];\nlist['persistent-single-player'] = list['reuse-across-sessions'];\n\n\nmodule.exports = {\n /**\n * List of available strategies. Within this object, each key is the strategy name and the associated value is the strategy constructor.\n * @type {Object} \n */\n list: list,\n\n /**\n * Gets strategy by name.\n *\n * **Example**\n *\n * var reuseStrat = F.manager.RunManager.strategies.byName('reuse-across-sessions');\n * // shows strategy function\n * console.log('reuseStrat = ', reuseStrat);\n * // create a new run manager using this strategy\n * var rm = new F.manager.RunManager({strategy: reuseStrat, run: { model: 'model.vmf'} });\n *\n * **Parameters**\n * @param {String} strategyName Name of strategy to get.\n * @return {Function} Strategy function.\n */\n byName: function (strategyName) {\n return list[strategyName];\n },\n\n getBestStrategy: function (options) {\n var strategy = options.strategy;\n if (!strategy) {\n if (options.strategyOptions && options.strategyOptions.initOperation) {\n strategy = 'reuse-last-initialized';\n } else {\n strategy = 'reuse-per-session';\n }\n }\n\n if (strategy.getRun) {\n return strategy;\n }\n var StrategyCtor = typeof strategy === 'function' ? strategy : this.byName(strategy);\n if (!StrategyCtor) {\n throw new Error('Specified run creation strategy was invalid:', strategy);\n }\n\n var strategyInstance = new StrategyCtor(options);\n if (!strategyInstance.getRun || !strategyInstance.reset) {\n throw new Error('All strategies should implement a `getRun` and `reset` interface', options.strategy);\n }\n strategyInstance.requiresAuth = StrategyCtor.requiresAuth;\n\n return strategyInstance;\n },\n\n /**\n * Adds a new strategy.\n *\n * **Example**\n *\n * // this \"favorite run\" strategy always returns the same run, no matter what\n * // (not a useful strategy, except as an example)\n * F.manager.RunManager.strategies.register(\n * 'favRun', \n * function() { \n * return { getRun: function() { return '0000015a4cd1700209cd0a7d207f44bac289'; },\n * reset: function() { return '0000015a4cd1700209cd0a7d207f44bac289'; } \n * } \n * }, \n * { requiresAuth: true }\n * );\n * \n * var rm = new F.manager.RunManager({strategy: 'favRun', run: { model: 'model.vmf'} });\n *\n * **Parameters**\n * @param {String} name Name for strategy. This string can then be passed to a Run Manager as `new F.manager.RunManager({ strategy: 'mynewname'})`.\n * @param {Function} strategy The strategy constructor. Will be called with `new` on Run Manager initialization.\n * @param {Object} options Options for strategy.\n * @param {Boolean} options.requiresAuth Specify if the strategy requires a valid user session to work.\n */\n register: function (name, strategy, options) {\n strategy.options = options;\n list[name] = strategy;\n }\n};","/**\n * The `multiplayer` strategy is for use with [multiplayer worlds](../../../glossary/#world). It checks the current world for this end user, and always returns the current run for that world. This is equivalent to calling `getCurrentWorldForUser()` and then `getCurrentRunId()` from the [World API Adapater](../world-api-adapter/). If you use the [World Manager](../world-manager/), you are automatically using this strategy.\n * \n * Using this strategy means that end users in projects with multiplayer worlds always see the most current world and run. This ensures that they are in sync with the other end users sharing their world and run. In turn, this allows for competitive or collaborative multiplayer projects.\n */\n'use strict';\n\nvar classFrom = require('../../util/inherit');\n\nvar IdentityStrategy = require('./none-strategy');\nvar WorldApiAdapter = require('../../service/world-api-adapter');\n\nvar defaults = {};\n\nvar Strategy = classFrom(IdentityStrategy, {\n constructor: function (options) {\n this.options = $.extend(true, {}, defaults, options);\n this.worldApi = new WorldApiAdapter(this.options.run);\n },\n\n reset: function (runService, session, options) {\n var curUserId = session.userId;\n var curGroupName = session.groupName;\n\n return this.worldApi\n .getCurrentWorldForUser(curUserId, curGroupName)\n .then(function (world) {\n return this.worldApi.newRunForWorld(world.id, options).then(function (runid) {\n return {\n id: runid\n };\n });\n }.bind(this));\n },\n\n getRun: function (runService, session) {\n var curUserId = session.userId;\n var curGroupName = session.groupName;\n var worldApi = this.worldApi;\n var model = this.options.model;\n var me = this;\n var dtd = $.Deferred();\n\n if (!curUserId) {\n return dtd.reject({ statusCode: 400, error: 'We need an authenticated user to join a multiplayer world. (ERR: no userId in session)' }, session).promise();\n }\n\n var loadRunFromWorld = function (world) {\n if (!world) {\n return dtd.reject({ statusCode: 404, error: 'The user is not in any world.' }, { options: me.options, session: session });\n }\n return worldApi.getCurrentRunId({ model: model, filter: world.id })\n .then(function (id) {\n return runService.load(id);\n })\n .then(dtd.resolve)\n .fail(dtd.reject);\n };\n\n var serverError = function (error) {\n // is this possible?\n return dtd.reject(error, session, me.options);\n };\n\n this.worldApi\n .getCurrentWorldForUser(curUserId, curGroupName)\n .then(loadRunFromWorld)\n .fail(serverError);\n\n return dtd.promise();\n },\n\n});\n\nmodule.exports = Strategy;\n","/**\n * The `none` strategy never returns a run or tries to create a new run. It simply returns the contents of the current [Run Service instance](../run-api-service/).\n * \n * This strategy is useful if you want to manually decide how to create your own runs and don't want any automatic assistance.\n */\n\n'use strict';\n\nvar classFrom = require('../../util/inherit');\nvar Base = {};\n\n// Interface that all strategies need to implement\nmodule.exports = classFrom(Base, {\n constructor: function (options) {\n\n },\n\n reset: function () {\n // return a newly created run\n return $.Deferred().resolve().promise();\n },\n\n getRun: function (runService) {\n // return a usable run\n return $.Deferred().resolve(runService).promise();\n }\n});\n","/**\n * The `reuse-across-sessions` strategy returns the latest (most recent) run for this user, whether it is in memory or not. If there are no runs for this user, it creates a new one.\n *\n * This strategy is useful if end users are using your project for an extended period of time, possibly over several sessions. This is most common in cases where a user of your project executes the model step by step (as opposed to a project where the model is executed completely, for example, a Vensim model that is immediately stepped to the end).\n *\n * Specifically, the strategy is:\n * \n * * Check if there are any runs for this end user.\n * * If there are no runs (either in memory or in the database), create a new one.\n * * If there are runs, take the latest (most recent) one.\n *\n * @name persistent-single-player\n */\n\n'use strict';\n\nvar classFrom = require('../../util/inherit');\nvar IdentityStrategy = require('./none-strategy');\nvar injectFiltersFromSession = require('../strategy-utils').injectFiltersFromSession;\nvar injectScopeFromSession = require('../strategy-utils').injectScopeFromSession;\n\nvar defaults = {\n /**\n * (Optional) Additional criteria to use while selecting the last run\n * @type {Object}\n */\n filter: {},\n};\n\nvar Strategy = classFrom(IdentityStrategy, {\n constructor: function Strategy(options) {\n var strategyOptions = options ? options.strategyOptions : {};\n this.options = $.extend(true, {}, defaults, strategyOptions);\n },\n\n reset: function (runService, userSession, options) {\n var opt = injectScopeFromSession(runService.getCurrentConfig(), userSession);\n return runService\n .create(opt, options)\n .then(function (run) {\n run.freshlyCreated = true;\n return run;\n });\n },\n\n getRun: function (runService, userSession, runSession, options) {\n var filter = injectFiltersFromSession(this.options.filter, userSession);\n var me = this;\n return runService.query(filter, { \n // startrecord: 0, //TODO: Uncomment when EPICENTER-2569 is fixed\n // endrecord: 0,\n sort: 'created', \n direction: 'desc'\n }).then(function (runs) {\n if (!runs.length) {\n return me.reset(runService, userSession, options);\n }\n return runs[0];\n });\n }\n});\n\nmodule.exports = Strategy;\n","/**\n * The `reuse-last-initialized` strategy looks for the most recent run that matches particular criteria; if it cannot find one, it creates a new run and immediately executes a set of \"initialization\" operations. \n *\n * This strategy is useful if you have a time-based model and always want the run you're operating on to start at a particular step. For example:\n *\n * var rm = new F.manager.RunManager({\n * strategy: 'reuse-last-initialized',\n * strategyOptions: {\n * initOperation: [{ step: 10 }]\n * }\n * });\n * \n * This strategy is also useful if you have a custom initialization function in your model, and want to make sure it's always executed for new runs.\n *\n * Specifically, the strategy is:\n *\n * * Look for the most recent run that matches the (optional) `flag` criteria\n * * If there are no runs that match the `flag` criteria, create a new run. Immediately \"initialize\" this new run by:\n * * Calling the model operation(s) specified in the `initOperation` array.\n * * Optionally, setting a `flag` in the run.\n *\n */\n\n'use strict';\nvar classFrom = require('../../util/inherit');\nvar injectFiltersFromSession = require('../strategy-utils').injectFiltersFromSession;\nvar injectScopeFromSession = require('../strategy-utils').injectScopeFromSession;\n\nvar Base = {};\n\nvar defaults = {\n /**\n * Operations to execute in the model for initialization to be considered complete.\n * @type {Array} Can be in any of the formats [Run Service's `serial()`](../run-api-service/#serial) supports.\n */\n initOperation: [],\n\n /**\n * (Optional) Flag to set in run after initialization operations are run. You typically would not override this unless you needed to set additional properties as well.\n * @type {Object}\n */\n flag: null,\n};\nmodule.exports = classFrom(Base, {\n constructor: function (options) {\n var strategyOptions = options ? options.strategyOptions : {};\n this.options = $.extend(true, {}, defaults, strategyOptions);\n if (!this.options.initOperation || !this.options.initOperation.length) {\n throw new Error('Specifying an init function is required for this strategy');\n }\n if (!this.options.flag) {\n this.options.flag = {\n isInitComplete: true\n };\n }\n },\n\n reset: function (runService, userSession, options) {\n var opt = injectScopeFromSession(runService.getCurrentConfig(), userSession);\n var me = this;\n return runService.create(opt, options).then(function (createResponse) {\n return runService.serial([].concat(me.options.initOperation)).then(function () {\n return createResponse;\n });\n }).then(function (createResponse) {\n return runService.save(me.options.flag).then(function (patchResponse) {\n return $.extend(true, {}, createResponse, patchResponse);\n });\n });\n },\n\n getRun: function (runService, userSession, runSession, options) {\n var sessionFilter = injectFiltersFromSession(this.options.flag, userSession);\n var runopts = runService.getCurrentConfig();\n var filter = $.extend(true, { trashed: false }, sessionFilter, { model: runopts.model });\n var me = this;\n return runService.query(filter, { \n // startrecord: 0, //TODO: Uncomment when EPICENTER-2569 is fixed\n // endrecord: 0,\n sort: 'created', \n direction: 'desc'\n }).then(function (runs) {\n if (!runs.length) {\n return me.reset(runService, userSession, options);\n }\n return runs[0];\n });\n }\n});","/**\n * The `reuse-never` strategy always creates a new run for this end user irrespective of current state. This is equivalent to calling `F.service.Run.create()` from the [Run Service](../run-api-service/) every time. \n * \n * This strategy means that every time your end users refresh their browsers, they get a new run. \n * \n * This strategy can be useful for basic, single-page projects. This strategy is also useful for prototyping or project development: it creates a new run each time you refresh the page, and you can easily check the outputs of the model. However, typically you will use one of the other strategies for a production project.\n *\n * @name always-new\n */\n\n'use strict';\n\nvar classFrom = require('../../util/inherit');\nvar ConditionalStrategy = require('./conditional-creation-strategy');\n\nvar __super = ConditionalStrategy.prototype;\n\nvar Strategy = classFrom(ConditionalStrategy, {\n constructor: function (options) {\n __super.constructor.call(this, this.createIf, options);\n },\n\n createIf: function (run, headers) {\n // always create a new run!\n return true;\n }\n});\n\nmodule.exports = Strategy;\n","/**\n * The `reuse-per-session` strategy creates a new run when the current one is not in the browser cookie.\n * \n * Using this strategy means that when end users navigate between pages in your project, or refresh their browsers, they will still be working with the same run. However, if end users log out and return to the project at a later date, a new run is created.\n *\n * This strategy is useful if your project is structured such that immediately after a run is created, the model is executed completely (for example, a Vensim model that is stepped to the end as soon as it is created). In contrast, if end users play with your project for an extended period of time, executing the model step by step, the `reuse-across-sessions` strategy is probably a better choice (it allows end users to pick up where they left off, rather than starting from scratch each browser session).\n * \n * Specifically, the strategy is:\n *\n * * Check the `sessionKey` cookie.\n * * This cookie is set by the [Run Manager](../run-manager/) and configurable through its options. \n * * If the cookie exists, use the run id stored there. \n * * If the cookie does not exist, create a new run for this end user.\n *\n * @name new-if-missing\n */\n\n'use strict';\n\nvar classFrom = require('../../util/inherit');\nvar ConditionalStrategy = require('./conditional-creation-strategy');\n\nvar __super = ConditionalStrategy.prototype;\n\n/*\n* create a new run only if nothing is stored in the cookie\n* this is useful for baseRuns.\n*/\nvar Strategy = classFrom(ConditionalStrategy, {\n constructor: function (options) {\n __super.constructor.call(this, this.createIf, options);\n },\n\n createIf: function (run, headers) {\n // if we are here, it means that the run exists... so we don't need a new one\n return false;\n }\n});\n\nmodule.exports = Strategy;\n","/**\n * ## Saved Runs Manager\n *\n * The Saved Runs Manager is a specific type of [Run Manager](../../run-manager/) which provides access to a list of runs (rather than just one run). It also provides utility functions for dealing with multiple runs (e.g. saving, deleting, listing).\n *\n * An instance of a Saved Runs Manager is included automatically in every instance of a [Scenario Manager](../), and is accessible from the Scenario Manager at `.savedRuns`. See [more information](../#properties) on using `.savedRuns` within the Scenario Manager.\n *\n */\n'use strict';\n\nvar RunService = require('../service/run-api-service');\nvar SessionManager = require('../store/session-manager');\n\nvar injectFiltersFromSession = require('./strategy-utils').injectFiltersFromSession;\n\nvar SavedRunsManager = function (config) {\n var defaults = {\n /**\n * If set, will only pull runs from current group. Defaults to `true`.\n * @type {Boolean}\n */\n scopeByGroup: true,\n\n /**\n * If set, will only pull runs from current user. Defaults to `true`.\n *\n * For multiplayer run comparison projects, set this to false so that all end users in a group can view the shared set of saved runs.\n * @type {Boolean}\n */\n scopeByUser: true,\n };\n\n this.sessionManager = new SessionManager();\n\n var options = $.extend(true, {}, defaults, config);\n if (options.run) {\n if (options.run instanceof RunService) {\n this.runService = options.run;\n } else {\n this.runService = new RunService(options.run);\n }\n this.options = options;\n } else {\n throw new Error('No run options passed to SavedRunsManager');\n }\n};\n\nSavedRunsManager.prototype = {\n /**\n * Marks a run as saved. \n *\n * Note that while any run can be saved, only runs which also match the configuration options `scopeByGroup` and `scopeByUser` are returned by the `getRuns()` method.\n *\n * **Example**\n *\n * var sm = new F.manager.ScenarioManager();\n * sm.savedRuns.save('0000015a4cd1700209cd0a7d207f44bac289');\n *\n * @param {String|RunService} run Run to save. Pass in either the run id, as a string, or the [Run Service](../../run-api-service/).\n * @param {Object} otherFields (Optional) Any other meta-data to save with the run.\n * @return {Promise}\n */\n save: function (run, otherFields) {\n var param = $.extend(true, {}, otherFields, { saved: true, trashed: false });\n return this.mark(run, param);\n },\n /**\n * Marks a run as removed; the inverse of marking as saved.\n *\n * **Example**\n *\n * var sm = new F.manager.ScenarioManager();\n * sm.savedRuns.remove('0000015a4cd1700209cd0a7d207f44bac289');\n *\n * @param {String|RunService} run Run to remove. Pass in either the run id, as a string, or the [Run Service](../../run-api-service/).\n * @param {Object} otherFields (Optional) any other meta-data to save with the run.\n * @return {Promise}\n */\n remove: function (run, otherFields) {\n var param = $.extend(true, {}, otherFields, { saved: false, trashed: true });\n return this.mark(run, param);\n },\n\n\n /**\n * Sets additional fields on a run. This is a convenience method for [RunService#save](../../run-api-service/#save).\n *\n * **Example**\n *\n * var sm = new F.manager.ScenarioManager();\n * sm.savedRuns.mark('0000015a4cd1700209cd0a7d207f44bac289', \n * { 'myRunName': 'sample policy decisions' });\n *\n * @param {String|RunService} run Run to operate on. Pass in either the run id, as a string, or the [Run Service](../../run-api-service/).\n * @param {Object} toMark Fields to set, as name : value pairs.\n * @return {Promise}\n */\n mark: function (run, toMark) {\n var rs;\n var existingOptions = this.runService.getCurrentConfig();\n if (run instanceof RunService) {\n rs = run;\n } else if (run && (typeof run === 'string')) {\n rs = new RunService($.extend(true, {}, existingOptions, { id: run, autoRestore: false }));\n } else if ($.isArray(run)) {\n var me = this;\n var proms = run.map(function (r) {\n return me.mark(r, toMark);\n });\n return $.when.apply(null, proms);\n } else {\n throw new Error('Invalid run object provided');\n }\n return rs.save(toMark);\n },\n\n /**\n * Returns a list of saved runs.\n *\n * **Example**\n *\n * var sm = new F.manager.ScenarioManager();\n * sm.savedRuns.getRuns().then(function (runs) {\n * for (var i=0; i\n * //\n * $('#fileupload').on('change', function (e) {\n * var file = e.target.files[0];\n * var data = new FormData();\n * data.append('file', file, file.name);\n * fa.create(file.name, data);\n * });\n */\n\n'use strict';\n\nvar ConfigService = require('./configuration-service');\nvar TransportFactory = require('../transport/http-transport-factory');\nvar SessionManager = require('../store/session-manager');\n\nmodule.exports = function (config) {\n var defaults = {\n /**\n * For projects that require authentication, pass in the user access token (defaults to empty string). If the user is already logged in to Epicenter, the user access token is already set in a cookie and automatically loaded from there. (See [more background on access tokens](../../../project_access/)).\n * @see [Authentication API Service](../auth-api-service/) for getting tokens.\n * @type {String}\n */\n token: undefined,\n\n /**\n * The account id. In the Epicenter UI, this is the **Team ID** (for team projects) or **User ID** (for personal projects). Defaults to undefined.\n * @type {String}\n */\n account: undefined,\n\n /**\n * The project id. Defaults to undefined.\n * @type {String}\n */\n project: undefined,\n\n /**\n * The folder type. One of `model` | `static` | `node`.\n * @type {String}\n */\n folderType: 'static',\n\n\n /**\n * Options to pass on to the underlying transport layer. All jquery.ajax options at http://api.jquery.com/jQuery.ajax/ are available. Defaults to empty object.\n * @type {Object}\n */\n transport: {}\n };\n\n this.sessionManager = new SessionManager();\n var serviceOptions = this.sessionManager.getMergedOptions(defaults, config);\n var urlConfig = new ConfigService(serviceOptions).get('server');\n if (serviceOptions.account) {\n urlConfig.accountPath = serviceOptions.account;\n }\n if (serviceOptions.project) {\n urlConfig.projectPath = serviceOptions.project;\n }\n\n var httpOptions = $.extend(true, {}, serviceOptions.transport, {\n url: urlConfig.getAPIPath('file')\n });\n\n if (serviceOptions.token) {\n httpOptions.headers = {\n Authorization: 'Bearer ' + serviceOptions.token\n };\n }\n var http = new TransportFactory(httpOptions);\n\n function uploadBody(fileName, contents) {\n var boundary = '---------------------------7da24f2e50046';\n\n return {\n body: '--' + boundary + '\\r\\n' +\n 'Content-Disposition: form-data; name=\"file\";' +\n 'filename=\"' + fileName + '\"\\r\\n' +\n 'Content-type: text/html\\r\\n\\r\\n' +\n contents + '\\r\\n' +\n '--' + boundary + '--',\n boundary: boundary\n };\n }\n\n function uploadFileOptions(filePath, contents, options) {\n filePath = filePath.split('/');\n var fileName = filePath.pop();\n filePath = filePath.join('/');\n var path = serviceOptions.folderType + '/' + filePath;\n\n var extraParams = {};\n if (contents instanceof FormData) {\n extraParams = {\n data: contents,\n processData: false,\n contentType: false,\n };\n } else {\n var upload = uploadBody(fileName, contents);\n extraParams = {\n data: upload.body,\n contentType: 'multipart/form-data; boundary=' + upload.boundary\n };\n }\n\n return $.extend(true, {}, serviceOptions, options, {\n url: urlConfig.getAPIPath('file') + path,\n }, extraParams);\n }\n\n var publicAsyncAPI = {\n /**\n * Get a directory listing, or contents of a file.\n * @param {String} filePath Path to the file\n * @param {Object} options (Optional) Overrides for configuration options.\n * @return {Promise}\n */\n getContents: function (filePath, options) {\n var path = serviceOptions.folderType + '/' + filePath;\n var httpOptions = $.extend(true, {}, serviceOptions, options, {\n url: urlConfig.getAPIPath('file') + path\n });\n return http.get('', httpOptions);\n },\n\n /**\n * Replaces the file at the given file path.\n * @param {String} filePath Path to the file\n * @param {String | FormData } contents Contents to write to file\n * @param {Object} options (Optional) Overrides for configuration options\n * @return {Promise}\n */\n replace: function (filePath, contents, options) {\n var httpOptions = uploadFileOptions(filePath, contents, options);\n return http.put(httpOptions.data, httpOptions);\n },\n\n /**\n * Creates a file in the given file path.\n * @param {String} filePath Path to the file\n * @param {String | FormData } contents Contents to write to file\n * @param {Boolean} replaceExisting Replace file if it already exists; defaults to false\n * @param {Object} options (Optional) Overrides for configuration options\n * @return {Promise}\n */\n create: function (filePath, contents, replaceExisting, options) {\n var httpOptions = uploadFileOptions(filePath, contents, options);\n var prom = http.post(httpOptions.data, httpOptions);\n var me = this;\n if (replaceExisting === true) {\n prom = prom.then(null, function (xhr) {\n var conflictStatus = 409;\n if (xhr.status === conflictStatus) {\n return me.replace(filePath, contents, options);\n }\n });\n }\n return prom;\n },\n\n /**\n * Removes the file.\n * @param {String} filePath Path to the file\n * @param {Object} options (Optional) Overrides for configuration options\n * @return {Promise}\n */\n remove: function (filePath, options) {\n var path = serviceOptions.folderType + '/' + filePath;\n var httpOptions = $.extend(true, {}, serviceOptions, options, {\n url: urlConfig.getAPIPath('file') + path\n });\n return http.delete(null, httpOptions);\n },\n\n /**\n * Renames the file.\n * @param {String} filePath Path to the file\n * @param {String} newName New name of file\n * @param {Object} options (Optional) Overrides for configuration options\n * @return {Promise}\n */\n rename: function (filePath, newName, options) {\n var path = serviceOptions.folderType + '/' + filePath;\n var httpOptions = $.extend(true, {}, serviceOptions, options, {\n url: urlConfig.getAPIPath('file') + path\n });\n return http.patch({ name: newName }, httpOptions);\n }\n };\n\n $.extend(this, publicAsyncAPI);\n};\n","/**\n * ## Asset API Adapter\n *\n * The Asset API Adapter allows you to store assets -- resources or files of any kind -- used by a project with a scope that is specific to project, group, or end user.\n *\n * Assets are used with [team projects](../../../project_admin/#team). One common use case is having end users in a [group](../../../glossary/#groups) or in a [multiplayer world](../../../glossary/#world) upload data -- videos created during game play, profile pictures for customizing their experience, etc. -- as part of playing through the project.\n *\n * Resources created using the Asset Adapter are scoped:\n *\n * * Project assets are writable only by [team members](../../../glossary/#team), that is, Epicenter authors.\n * * Group assets are writable by anyone with access to the project that is part of that particular [group](../../../glossary/#groups). This includes all [team members](../../../glossary/#team) (Epicenter authors) and any [end users](../../../glossary/#users) who are members of the group -- both facilitators and standard end users.\n * * User assets are writable by the specific end user, and by the facilitator of the group.\n * * All assets are readable by anyone with the exact URI.\n *\n * To use the Asset Adapter, instantiate it and then access the methods provided. Instantiating requires the account id (**Team ID** in the Epicenter user interface) and project id (**Project ID**). The group name is required for assets with a group scope, and the group name and userId are required for assets with a user scope. If not included, they are taken from the logged in user's session information if needed.\n *\n * When creating an asset, you can pass in text (encoded data) to the `create()` call. Alternatively, you can make the `create()` call as part of an HTML form and pass in a file uploaded via the form.\n *\n * // instantiate the Asset Adapter\n * var aa = new F.service.Asset({\n * account: 'acme-simulations',\n * project: 'supply-chain-game',\n * group: 'team1',\n * userId: '12345'\n * });\n *\n * // create a new asset using encoded text\n * aa.create('test.txt', {\n * encoding: 'BASE_64',\n * data: 'VGhpcyBpcyBhIHRlc3QgZmlsZS4=',\n * contentType: 'text/plain'\n * }, { scope: 'user' });\n *\n * // alternatively, create a new asset using a file uploaded through a form\n * // this sample code goes with an html form that looks like this:\n * //\n * //
\n * // \n * // \n * // \n * //
\n * //\n * $('#upload-file').on('submit', function (e) {\n * e.preventDefault();\n * var filename = $('#filename').val();\n * var data = new FormData();\n * var inputControl = $('#file')[0];\n * data.append('file', inputControl.files[0], filename);\n *\n * aa.create(filename, data, { scope: 'user' });\n * });\n *\n */\n\n'use strict';\n\nvar ConfigService = require('./configuration-service');\nvar TransportFactory = require('../transport/http-transport-factory');\nvar _pick = require('../util/object-util')._pick;\nvar SessionManager = require('../store/session-manager');\n\nvar apiEndpoint = 'asset';\n\nmodule.exports = function (config) {\n var defaults = {\n /**\n * For projects that require authentication, pass in the user access token (defaults to empty string). If the user is already logged in to Epicenter, the user access token is already set in a cookie and automatically loaded from there. (See [more background on access tokens](../../../project_access/)).\n * @see [Authentication API Service](../auth-api-service/) for getting tokens.\n * @type {String}\n */\n token: undefined,\n /**\n * The account id. In the Epicenter UI, this is the **Team ID** (for team projects). If left undefined, taken from the URL.\n * @type {String}\n */\n account: undefined,\n /**\n * The project id. If left undefined, taken from the URL.\n * @type {String}\n */\n project: undefined,\n /**\n * The group name. Defaults to session's `groupName`.\n * @type {String}\n */\n group: undefined,\n /**\n * The user id. Defaults to session's `userId`.\n * @type {String}\n */\n userId: undefined,\n /**\n * The scope for the asset. Valid values are: `user`, `group`, and `project`. See above for the required permissions to write to each scope. Defaults to `user`, meaning the current end user or a facilitator in the end user's group can edit the asset.\n * @type {String}\n */\n scope: 'user',\n /**\n * Determines if a request to list the assets in a scope includes the complete URL for each asset (`true`), or only the file names of the assets (`false`). Defaults to `true`.\n * @type {boolean}\n */\n fullUrl: true,\n /**\n * The transport object contains the options passed to the XHR request.\n * @type {object}\n */\n transport: {\n processData: false\n }\n };\n this.sessionManager = new SessionManager();\n var serviceOptions = this.sessionManager.getMergedOptions(defaults, config);\n var urlConfig = new ConfigService(serviceOptions).get('server');\n\n if (!serviceOptions.account) {\n serviceOptions.account = urlConfig.accountPath;\n }\n\n if (!serviceOptions.project) {\n serviceOptions.project = urlConfig.projectPath;\n }\n\n var transportOptions = $.extend(true, {}, serviceOptions.transport, {\n url: urlConfig.getAPIPath(apiEndpoint)\n });\n\n if (serviceOptions.token) {\n transportOptions.headers = {\n Authorization: 'Bearer ' + serviceOptions.token\n };\n }\n\n var http = new TransportFactory(transportOptions);\n\n var assetApiParams = ['encoding', 'data', 'contentType'];\n var scopeConfig = {\n user: ['scope', 'account', 'project', 'group', 'userId'],\n group: ['scope', 'account', 'project', 'group'],\n project: ['scope', 'account', 'project'],\n };\n\n var validateFilename = function (filename) {\n if (!filename) {\n throw new Error('filename is needed.');\n }\n };\n\n var validateUrlParams = function (options) {\n var partKeys = scopeConfig[options.scope];\n if (!partKeys) {\n throw new Error('scope parameter is needed.');\n }\n\n $.each(partKeys, function () {\n if (!options[this]) {\n throw new Error(this + ' parameter is needed.');\n }\n });\n };\n\n var buildUrl = function (filename, options) {\n validateUrlParams(options);\n var partKeys = scopeConfig[options.scope];\n var parts = $.map(partKeys, function (key) {\n return options[key];\n });\n if (filename) {\n // This prevents adding a trailing / in the URL as the Asset API\n // does not work correctly with it\n filename = '/' + filename;\n }\n return urlConfig.getAPIPath(apiEndpoint) + parts.join('/') + filename;\n };\n\n // Private function, all requests follow a more or less same approach to\n // use the Asset API and the difference is the HTTP verb\n //\n // @param {string} method` (Required) HTTP verb\n // @param {string} filename` (Required) Name of the file to delete/replace/create\n // @param {object} params` (Optional) Body parameters to send to the Asset API\n // @param {object} options` (Optional) Options object to override global options.\n var upload = function (method, filename, params, options) {\n validateFilename(filename);\n // make sure the parameter is clean\n method = method.toLowerCase();\n var contentType = params instanceof FormData === true ? false : 'application/json';\n if (contentType === 'application/json') {\n // whitelist the fields that we actually can send to the api\n params = _pick(params, assetApiParams);\n } else { // else we're sending form data which goes directly in request body\n // For multipart/form-data uploads the filename is not set in the URL,\n // it's getting picked by the FormData field filename.\n filename = method === 'post' || method === 'put' ? '' : filename;\n }\n var urlOptions = $.extend({}, serviceOptions, options);\n var url = buildUrl(filename, urlOptions);\n var createOptions = $.extend(true, {}, urlOptions, { url: url, contentType: contentType });\n\n return http[method](params, createOptions);\n };\n\n var publicAPI = {\n /**\n * Creates a file in the Asset API. The server returns an error (status code `409`, conflict) if the file already exists, so\n * check first with a `list()` or a `get()`.\n *\n * **Example**\n *\n * var aa = new F.service.Asset({\n * account: 'acme-simulations',\n * project: 'supply-chain-game',\n * group: 'team1',\n * userId: ''\n * });\n *\n * // create a new asset using encoded text\n * aa.create('test.txt', {\n * encoding: 'BASE_64',\n * data: 'VGhpcyBpcyBhIHRlc3QgZmlsZS4=',\n * contentType: 'text/plain'\n * }, { scope: 'user' });\n *\n * // alternatively, create a new asset using a file uploaded through a form\n * // this sample code goes with an html form that looks like this:\n * //\n * //
\n * // \n * // \n * // \n * //
\n * //\n * $('#upload-file').on('submit', function (e) {\n * e.preventDefault();\n * var filename = $('#filename').val();\n * var data = new FormData();\n * var inputControl = $('#file')[0];\n * data.append('file', inputControl.files[0], filename);\n *\n * aa.create(filename, data, { scope: 'user' });\n * });\n *\n *\n * **Parameters**\n * @param {string} filename (Required) Name of the file to create.\n * @param {object} params (Optional) Body parameters to send to the Asset API. Required if the `options.transport.contentType` is `application/json`, otherwise ignored.\n * @param {string} params.encoding Either `HEX` or `BASE_64`. Required if `options.transport.contentType` is `application/json`.\n * @param {string} params.data The encoded data for the file. Required if `options.transport.contentType` is `application/json`.\n * @param {string} params.contentType The mime type of the file. Optional.\n * @param {object} options (Optional) Options object to override global options.\n * @return {Promise}\n */\n create: function (filename, params, options) {\n return upload('post', filename, params, options);\n },\n\n /**\n * Gets a file from the Asset API, fetching the asset content. (To get a list\n * of the assets in a scope, use `list()`.)\n *\n * **Parameters**\n * @param {string} filename (Required) Name of the file to retrieve.\n * @param {object} options (Optional) Options object to override global options.\n * @return {Promise}\n */\n get: function (filename, options) {\n var getServiceOptions = _pick(serviceOptions, ['scope', 'account', 'project', 'group', 'userId']);\n var urlOptions = $.extend({}, getServiceOptions, options);\n var url = buildUrl(filename, urlOptions);\n var getOptions = $.extend(true, {}, urlOptions, { url: url });\n\n return http.get({}, getOptions);\n },\n\n /**\n * Gets the list of the assets in a scope.\n *\n * **Example**\n *\n * aa.list({ fullUrl: true }).then(function(fileList){\n * console.log('array of files = ', fileList);\n * });\n *\n * **Parameters**\n * @param {object} options (Optional) Options object to override global options.\n * @param {string} options.scope (Optional) The scope (`user`, `group`, `project`).\n * @param {boolean} options.fullUrl (Optional) Determines if the list of assets in a scope includes the complete URL for each asset (`true`), or only the file names of the assets (`false`).\n * @return {Promise}\n */\n list: function (options) {\n var dtd = $.Deferred();\n var me = this;\n var urlOptions = $.extend({}, serviceOptions, options);\n var url = buildUrl('', urlOptions);\n var getOptions = $.extend(true, {}, urlOptions, { url: url });\n var fullUrl = getOptions.fullUrl;\n\n if (!fullUrl) {\n return http.get({}, getOptions);\n }\n\n http.get({}, getOptions)\n .then(function (files) {\n var fullPathFiles = $.map(files, function (file) {\n return buildUrl(file, urlOptions);\n });\n dtd.resolveWith(me, [fullPathFiles]);\n })\n .fail(dtd.reject);\n\n return dtd.promise();\n },\n\n /**\n * Replaces an existing file in the Asset API.\n *\n * **Example**\n *\n * // replace an asset using encoded text\n * aa.replace('test.txt', {\n * encoding: 'BASE_64',\n * data: 'VGhpcyBpcyBhIHNlY29uZCB0ZXN0IGZpbGUu',\n * contentType: 'text/plain'\n * }, { scope: 'user' });\n *\n * // alternatively, replace an asset using a file uploaded through a form\n * // this sample code goes with an html form that looks like this:\n * //\n * //
\n * // \n * // \n * // \n * //
\n * //\n * $('#replace-file').on('submit', function (e) {\n * e.preventDefault();\n * var filename = $('#replace-filename').val();\n * var data = new FormData();\n * var inputControl = $('#file')[0];\n * data.append('file', inputControl.files[0], filename);\n *\n * aa.replace(filename, data, { scope: 'user' });\n * });\n *\n * **Parameters**\n * @param {string} filename (Required) Name of the file being replaced.\n * @param {object} params (Optional) Body parameters to send to the Asset API. Required if the `options.transport.contentType` is `application/json`, otherwise ignored.\n * @param {string} params.encoding Either `HEX` or `BASE_64`. Required if `options.transport.contentType` is `application/json`.\n * @param {string} params.data The encoded data for the file. Required if `options.transport.contentType` is `application/json`.\n * @param {string} params.contentType The mime type of the file. Optional.\n * @param {object} options (Optional) Options object to override global options.\n * @return {Promise}\n */\n replace: function (filename, params, options) {\n return upload('put', filename, params, options);\n },\n\n /**\n * Deletes a file from the Asset API.\n *\n * **Example**\n *\n * aa.delete(sampleFileName);\n *\n * **Parameters**\n * @param {string} filename (Required) Name of the file to delete.\n * @param {object} options (Optional) Options object to override global options.\n * @return {Promise}\n */\n delete: function (filename, options) {\n return upload('delete', filename, {}, options);\n },\n\n assetUrl: function (filename, options) {\n var urlOptions = $.extend({}, serviceOptions, options);\n return buildUrl(filename, urlOptions);\n }\n };\n $.extend(this, publicAPI);\n};\n","/**\n *\n * ## Authentication API Service\n *\n * The Authentication API Service provides a method for logging in, which creates and returns a user access token.\n *\n * User access tokens are required for each call to Epicenter. (See [Project Access](../../../project_access/) for more information.)\n *\n * If you need additional functionality -- such as tracking session information, easily retrieving the user token, or getting the groups to which an end user belongs -- consider using the [Authorization Manager](../auth-manager/) instead.\n *\n * var auth = new F.service.Auth();\n * auth.login({ userName: 'jsmith@acmesimulations.com',\n * password: 'passw0rd' });\n */\n\n'use strict';\n\nvar ConfigService = require('./configuration-service');\nvar TransportFactory = require('../transport/http-transport-factory');\n\nmodule.exports = function (config) {\n var defaults = {\n /**\n * Email or username to use for logging in. Defaults to empty string.\n * @type {String}\n */\n userName: '',\n\n /**\n * Password for specified `userName`. Defaults to empty string.\n * @type {String}\n */\n password: '',\n\n /**\n * The account id for this `userName`. In the Epicenter UI, this is the **Team ID** (for team projects) or the **User ID** (for personal projects). Required if the `userName` is for an [end user](../../../glossary/#users). Defaults to empty string.\n * @type {String}\n */\n account: '',\n\n /**\n * Options to pass on to the underlying transport layer. All jquery.ajax options at http://api.jquery.com/jQuery.ajax/ are available. Defaults to empty object.\n * @type {Object}\n */\n transport: {}\n };\n var serviceOptions = $.extend({}, defaults, config);\n var urlConfig = new ConfigService(serviceOptions).get('server');\n\n var transportOptions = $.extend(true, {}, serviceOptions.transport, {\n url: urlConfig.getAPIPath('authentication')\n });\n var http = new TransportFactory(transportOptions);\n\n var publicAPI = {\n\n /**\n * Logs user in, returning the user access token.\n *\n * If no `userName` or `password` were provided in the initial configuration options, they are required in the `options` here. If no `account` was provided in the initial configuration options and the `userName` is for an [end user](../../../glossary/#users), the `account` is required as well.\n *\n * **Example**\n *\n * auth.login({\n * userName: 'jsmith',\n * password: 'passw0rd',\n * account: 'acme-simulations' })\n * .then(function (token) {\n * console.log(\"user access token is: \", token.access_token);\n * });\n *\n * **Parameters**\n * @param {Object} options (Optional) Overrides for configuration options.\n * @return {Promise}\n */\n login: function (options) {\n var httpOptions = $.extend(true, { success: $.noop }, serviceOptions, options);\n if (!httpOptions.userName || !httpOptions.password) {\n var resp = { status: 401, statusMessage: 'No username or password specified.' };\n if (options.error) {\n options.error.call(this, resp);\n }\n\n return $.Deferred().reject(resp).promise();\n }\n\n var postParams = {\n userName: httpOptions.userName,\n password: httpOptions.password,\n };\n if (httpOptions.account) {\n //pass in null for account under options if you don't want it to be sent\n postParams.account = httpOptions.account;\n }\n\n return http.post(postParams, httpOptions);\n },\n\n // (replace with /* */ comment block, to make visible in docs, once this is more than a noop)\n //\n // Logs user out from specified accounts.\n //\n // Epicenter logout is not implemented yet, so for now this is a dummy promise that gets automatically resolved.\n //\n // **Example**\n //\n // auth.logout();\n //\n // **Parameters**\n // @param {Object} `options` (Optional) Overrides for configuration options.\n //\n logout: function (options) {\n var dtd = $.Deferred();\n dtd.resolve();\n return dtd.promise();\n }\n };\n\n $.extend(this, publicAPI);\n};\n","/**\n * ## Channel Service\n *\n * The Epicenter platform provides a push channel, which allows you to publish and subscribe to messages within a [project](../../../glossary/#projects), [group](../../../glossary/#groups), or [multiplayer world](../../../glossary/#world). There are two main use cases for the channel: event notifications and chat messages.\n *\n * If you are developing with Epicenter.js, you should use the [Epicenter Channel Manager](../epicenter-channel-manager/) directly. The Epicenter Channel Manager documentation also has more [background](../epicenter-channel-manager/#background) information on channels and their use.\n *\n * The Channel Service is a building block for this functionality. It creates a publish-subscribe object, allowing you to publish messages, subscribe to messages, or unsubscribe from messages for a given 'topic' on a `$.cometd` transport instance.\n *\n * You'll need to include the `epicenter-multiplayer-dependencies.js` library in addition to the `epicenter.js` library in your project to use the Channel Service. See [Including Epicenter.js](../../#include).\n *\n * To use the Channel Service, instantiate it, then make calls to any of the methods you need.\n *\n * var cs = new F.service.Channel();\n * cs.publish('/acme-simulations/supply-chain-game/fall-seminar/run/variables', { price: 50 });\n *\n * If you are working through the [Epicenter Channel Manager](../epicenter-channel-manager/), when you ask to \"get\" a particular channel, you are really asking for an instance of the Channel Service with a topic already set, for example to the appropriate group or world:\n *\n * var cm = new F.manager.ChannelManager();\n * var gc = cm.getGroupChannel();\n * // because we used an Epicenter Channel Manager to get the group channel,\n * // subscribe() and publish() here default to the base topic for the group\n * gc.subscribe('', function(data) { console.log(data); });\n * gc.publish('', { message: 'a new message to the group' });\n *\n * The parameters for instantiating a Channel Service include:\n *\n * * `options` The options object to configure the Channel Service.\n * * `options.base` The base topic. This is added as a prefix to all further topics you publish or subscribe to while working with this Channel Service.\n * * `options.topicResolver` A function that processes all 'topics' passed into the `publish` and `subscribe` methods. This is useful if you want to implement your own serialize functions for converting custom objects to topic names. Returns a String. By default, it just echoes the topic.\n * * `options.transport` The instance of `$.cometd` to hook onto. See http://docs.cometd.org/reference/javascript.html for additional background on cometd.\n */\n\n'use strict';\nvar Channel = function (options) {\n var defaults = {\n\n /**\n * The base topic. This is added as a prefix to all further topics you publish or subscribe to while working with this Channel Service.\n * @type {string}\n */\n base: '',\n\n /**\n * A function that processes all 'topics' passed into the `publish` and `subscribe` methods. This is useful if you want to implement your own serialize functions for converting custom objects to topic names. By default, it just echoes the topic.\n *\n * **Parameters**\n *\n * * `topic` Topic to parse.\n *\n * **Return Value**\n *\n * * *String*: This function should return a string topic.\n *\n * @type {function}\n * @param {String} topic topic to resolve\n * @return {String}\n */\n topicResolver: function (topic) {\n return topic;\n },\n\n /**\n * The instance of `$.cometd` to hook onto.\n * @type {object}\n */\n transport: null\n };\n this.channelOptions = $.extend(true, {}, defaults, options);\n};\n\nvar makeName = function (channelName, topic) {\n //Replace trailing/double slashes\n var newName = (channelName ? (channelName + '/' + topic) : topic).replace(/\\/\\//g, '/').replace(/\\/$/, '');\n return newName;\n};\n\n\nChannel.prototype = $.extend(Channel.prototype, {\n\n // future functionality:\n // // Set the context for the callback\n // cs.subscribe('run', function () { this.innerHTML = 'Triggered'}, document.body);\n //\n // // Control the order of operations by setting the `priority`\n // cs.subscribe('run', cb, this, {priority: 9});\n //\n // // Only execute the callback, `cb`, if the value of the `price` variable is 50\n // cs.subscribe('run/variables/price', cb, this, {priority: 30, value: 50});\n //\n // // Only execute the callback, `cb`, if the value of the `price` variable is greater than 50\n // subscribe('run/variables/price', cb, this, {priority: 30, value: '>50'});\n //\n // // Only execute the callback, `cb`, if the value of the `price` variable is even\n // subscribe('run/variables/price', cb, this, {priority: 30, value: function (val) {return val % 2 === 0}});\n\n\n /**\n * Subscribe to changes on a topic.\n *\n * The topic should include the full path of the account id (**Team ID** for team projects), project id, and group name. (In most cases, it is simpler to use the [Epicenter Channel Manager](../epicenter-channel-manager/) instead, in which case this is configured for you.)\n *\n * **Examples**\n *\n * var cb = function(val) { console.log(val.data); };\n *\n * // Subscribe to changes on a top-level 'run' topic\n * cs.subscribe('/acme-simulations/supply-chain-game/fall-seminar/run', cb);\n *\n * // Subscribe to changes on children of the 'run' topic. Note this will also be triggered for changes to run.x.y.z.\n * cs.subscribe('/acme-simulations/supply-chain-game/fall-seminar/run/*', cb);\n *\n * // Subscribe to changes on both the top-level 'run' topic and its children\n * cs.subscribe(['/acme-simulations/supply-chain-game/fall-seminar/run',\n * '/acme-simulations/supply-chain-game/fall-seminar/run/*'], cb);\n *\n * // Subscribe to changes on a particular variable\n * subscribe('/acme-simulations/supply-chain-game/fall-seminar/run/variables/price', cb);\n *\n *\n * **Return Value**\n *\n * * *String* Returns a token you can later use to unsubscribe.\n *\n * **Parameters**\n * @param {String|Array} topic List of topics to listen for changes on.\n * @param {Function} callback Callback function to execute. Callback is called with signature `(evt, payload, metadata)`.\n * @param {Object} context Context in which the `callback` is executed.\n * @param {Object} options (Optional) Overrides for configuration options.\n * @param {Number} options.priority Used to control order of operations. Defaults to 0. Can be any +ve or -ve number.\n * @param {String|Number|Function} options.value The `callback` is only triggered if this condition matches. See examples for details.\n * @return {string} Subscription ID\n */\n subscribe: function (topic, callback, context, options) {\n\n var topics = [].concat(topic);\n var me = this;\n var subscriptionIds = [];\n var opts = me.channelOptions;\n\n opts.transport.batch(function () {\n $.each(topics, function (index, topic) {\n topic = makeName(opts.base, opts.topicResolver(topic));\n subscriptionIds.push(opts.transport.subscribe(topic, callback));\n });\n });\n return (subscriptionIds[1] ? subscriptionIds : subscriptionIds[0]);\n },\n\n /**\n * Publish data to a topic.\n *\n * **Examples**\n *\n * // Send data to all subscribers of the 'run' topic\n * cs.publish('/acme-simulations/supply-chain-game/fall-seminar/run', { completed: false });\n *\n * // Send data to all subscribers of the 'run/variables' topic\n * cs.publish('/acme-simulations/supply-chain-game/fall-seminar/run/variables', { price: 50 });\n *\n * **Parameters**\n *\n * @param {String} topic Topic to publish to.\n * @param {*} data Data to publish to topic.\n * @return {Array | Object} Responses to published data\n *\n */\n publish: function (topic, data) {\n var topics = [].concat(topic);\n var me = this;\n var returnObjs = [];\n var opts = me.channelOptions;\n\n\n opts.transport.batch(function () {\n $.each(topics, function (index, topic) {\n topic = makeName(opts.base, opts.topicResolver(topic));\n if (topic.charAt(topic.length - 1) === '*') {\n topic = topic.replace(/\\*+$/, '');\n console.warn('You can cannot publish to channels with wildcards. Publishing to ', topic, 'instead');\n }\n returnObjs.push(opts.transport.publish(topic, data));\n });\n });\n return (returnObjs[1] ? returnObjs : returnObjs[0]);\n },\n\n /**\n * Unsubscribe from changes to a topic.\n *\n * **Example**\n *\n * cs.unsubscribe('sampleToken');\n *\n * **Parameters**\n * @param {String} token The token for topic is returned when you initially subscribe. Pass it here to unsubscribe from that topic.\n * @return {Object} reference to current instance\n */\n unsubscribe: function (token) {\n this.channelOptions.transport.unsubscribe(token);\n return this;\n },\n\n /**\n * Start listening for events on this instance. Signature is same as for jQuery Events: http://api.jquery.com/on/.\n *\n * Supported events are: `connect`, `disconnect`, `subscribe`, `unsubscribe`, `publish`, `error`.\n *\n * **Parameters**\n *\n * @param {string} event The event type. See more detail at jQuery Events: http://api.jquery.com/on/.\n */\n on: function (event) {\n $(this).on.apply($(this), arguments);\n },\n\n /**\n * Stop listening for events on this instance. Signature is same as for jQuery Events: http://api.jquery.com/off/.\n *\n * **Parameters**\n *\n * @param {string} event The event type. See more detail at jQuery Events: http://api.jquery.com/off/.\n */\n off: function (event) {\n $(this).off.apply($(this), arguments);\n },\n\n /**\n * Trigger events and execute handlers. Signature is same as for jQuery Events: http://api.jquery.com/trigger/.\n *\n * **Parameters**\n *\n * @param {string} event The event type. See more detail at jQuery Events: http://api.jquery.com/trigger/.\n */\n trigger: function (event) {\n $(this).trigger.apply($(this), arguments);\n }\n\n});\n\nmodule.exports = Channel;\n","/**\n * @class ConfigurationService\n *\n * All services take in a configuration settings object to configure themselves. A JS hash {} is a valid configuration object, but optionally you can use the configuration service to toggle configs based on the environment\n *\n * @example\n * var cs = require('configuration-service')({\n * dev: { //environment\n port: 3000,\n host: 'localhost',\n },\n prod: {\n port: 8080,\n host: 'api.forio.com',\n logLevel: 'none'\n },\n logLevel: 'DEBUG' //global\n * });\n *\n * cs.get('logLevel'); //returns 'DEBUG'\n *\n * cs.setEnv('dev');\n * cs.get('logLevel'); //returns 'DEBUG'\n *\n * cs.setEnv('prod');\n * cs.get('logLevel'); //returns 'none'\n *\n */\n\n'use strict';\nvar urlService = require('./url-config-service');\n\nmodule.exports = function (config) {\n //TODO: Environments\n var defaults = {\n logLevel: 'NONE'\n };\n var serviceOptions = $.extend({}, defaults, config);\n serviceOptions.server = urlService(serviceOptions.server);\n\n return {\n\n data: serviceOptions,\n\n /**\n * Set the environment key to get configuration options from\n * @param { string} env\n */\n setEnv: function (env) {\n\n },\n\n /**\n * Get configuration.\n * @param { string} property optional\n * @return {*} Value of property if specified, the entire config object otherwise\n */\n get: function (property) {\n return serviceOptions[property];\n },\n\n /**\n * Set configuration.\n * @param { string|Object} key if a key is provided, set a key to that value. Otherwise merge object with current config\n * @param {*} value value for provided key\n */\n set: function (key, value) {\n serviceOptions[key] = value;\n }\n };\n};\n\n","/**\n * ## Data API Service\n *\n * The Data API Service allows you to create, access, and manipulate data related to any of your projects. Data are organized in collections. Each collection contains a document; each element of this top-level document is a JSON object. (See additional information on the underlying [Data API](../../../rest_apis/data_api/).)\n *\n * All API calls take in an \"options\" object as the last parameter. The options can be used to extend/override the Data API Service defaults. In particular, there are three required parameters when you instantiate the Data Service:\n *\n * * `account`: Epicenter account id (**Team ID** for team projects, **User ID** for personal projects).\n * * `project`: Epicenter project id.\n * * `root`: The the name of the collection. If you have multiple collections within each of your projects, you can also pass the collection name as an option for each call.\n *\n * var ds = new F.service.Data({\n * account: 'acme-simulations',\n * project: 'supply-chain-game',\n * root: 'survey-responses',\n * server: { host: 'api.forio.com' }\n * });\n * ds.saveAs('user1',\n * { 'question1': 2, 'question2': 10,\n * 'question3': false, 'question4': 'sometimes' } );\n * ds.saveAs('user2',\n * { 'question1': 3, 'question2': 8,\n * 'question3': true, 'question4': 'always' } );\n * ds.query('',{ 'question2': { '$gt': 9} });\n *\n * Note that in addition to the `account`, `project`, and `root`, the Data Service parameters optionally include a `server` object, whose `host` field contains the URI of the Forio server. This is automatically set, but you can pass it explicitly if desired. It is most commonly used for clarity when you are [hosting an Epicenter project on your own server](../../../how_to/self_hosting/).\n */\n\n'use strict';\n\nvar ConfigService = require('./configuration-service');\nvar qutil = require('../util/query-util');\nvar TransportFactory = require('../transport/http-transport-factory');\nvar SessionManager = require('../store/session-manager');\n\nmodule.exports = function (config) {\n var defaults = {\n /**\n * Name of collection. Required. Defaults to `/`, that is, the root level of your project at `forio.com/app/your-account-id/your-project-id/`, but must be set to a collection name.\n * @type {String}\n */\n root: '/',\n\n /**\n * The account id. In the Epicenter UI, this is the **Team ID** (for team projects) or **User ID** (for personal projects). Defaults to empty string. If left undefined, taken from the URL.\n * @type {String}\n */\n account: undefined,\n\n /**\n * The project id. Defaults to empty string. If left undefined, taken from the URL.\n * @type {String}\n */\n project: undefined,\n\n /**\n * For operations that require authentication, pass in the user access token (defaults to empty string). If the user is already logged in to Epicenter, the user access token is already set in a cookie and automatically loaded from there. (See [more background on access tokens](../../../project_access/)).\n * @see [Authentication API Service](../auth-api-service/) for getting tokens.\n * @type {String}\n */\n token: undefined,\n\n //Options to pass on to the underlying transport layer\n transport: {}\n };\n this.sessionManager = new SessionManager();\n var serviceOptions = this.sessionManager.getMergedOptions(defaults, config);\n\n var urlConfig = new ConfigService(serviceOptions).get('server');\n if (serviceOptions.account) {\n urlConfig.accountPath = serviceOptions.account;\n }\n if (serviceOptions.project) {\n urlConfig.projectPath = serviceOptions.project;\n }\n\n var getURL = function (key, root) {\n if (!root) {\n root = serviceOptions.root;\n }\n var url = urlConfig.getAPIPath('data') + qutil.addTrailingSlash(root);\n if (key) {\n url += qutil.addTrailingSlash(key);\n }\n return url;\n };\n\n var httpOptions = $.extend(true, {}, serviceOptions.transport, {\n url: getURL\n });\n if (serviceOptions.token) {\n httpOptions.headers = {\n Authorization: 'Bearer ' + serviceOptions.token\n };\n }\n var http = new TransportFactory(httpOptions);\n\n var publicAPI = {\n\n /**\n * Search for data within a collection.\n *\n * Searching using comparison or logical operators (as opposed to exact matches) requires MongoDB syntax. See the underlying [Data API](../../../rest_apis/data_api/#searching) for additional details.\n *\n * **Examples**\n *\n * // request all data associated with document 'user1'\n * ds.query('user1');\n *\n * // exact matching:\n * // request all documents in collection where 'question2' is 9\n * ds.query('', { 'question2': 9});\n *\n * // comparison operators:\n * // request all documents in collection\n * // where 'question2' is greater than 9\n * ds.query('', { 'question2': { '$gt': 9} });\n *\n * // logical operators:\n * // request all documents in collection\n * // where 'question2' is less than 10, and 'question3' is false\n * ds.query('', { '$and': [ { 'question2': { '$lt':10} }, { 'question3': false }] });\n *\n * // regular expresssions: use any Perl-compatible regular expressions\n * // request all documents in collection\n * // where 'question5' contains the string '.*day'\n * ds.query('', { 'question5': { '$regex': '.*day' } });\n *\n * **Parameters**\n * @param {String} key The name of the document to search. Pass the empty string ('') to search the entire collection.\n * @param {Object} query The query object. For exact matching, this object contains the field name and field value to match. For matching based on comparison, this object contains the field name and the comparison expression. For matching based on logical operators, this object contains an expression using MongoDB syntax. See the underlying [Data API](../../../rest_apis/data_api/#searching) for additional examples.\n * @param {Object} outputModifier (Optional) Available fields include: `startrecord`, `endrecord`, `sort`, and `direction` (`asc` or `desc`).\n * @param {Object} options (Optional) Overrides for configuration options.\n * @return {Promise} \n */\n query: function (key, query, outputModifier, options) {\n var params = $.extend(true, { q: query }, outputModifier);\n var httpOptions = $.extend(true, {}, serviceOptions, options);\n httpOptions.url = getURL(key, httpOptions.root);\n return http.get(params, httpOptions);\n },\n\n /**\n * Save data in an anonymous document within the collection.\n *\n * The `root` of the collection must be specified. By default the `root` is taken from the Data Service configuration options; you can also pass the `root` to the `save` call explicitly by overriding the options (third parameter).\n *\n * (Additional background: Documents are top-level elements within a collection. Collections must be unique within this account (team or personal account) and project and are set with the `root` field in the `option` parameter. See the underlying [Data API](../../../rest_apis/data_api/) for more information. The `save` method is making a `POST` request.)\n *\n * **Example**\n *\n * // Create a new document, with one element, at the default root level\n * ds.save('question1', 'yes');\n *\n * // Create a new document, with two elements, at the default root level\n * ds.save({ question1:'yes', question2: 32 });\n *\n * // Create a new document, with two elements, at `/students/`\n * ds.save({ name:'John', className: 'CS101' }, { root: 'students' });\n *\n * **Parameters**\n *\n * @param {String|Object} key If `key` is a string, it is the id of the element to save (create) in this document. If `key` is an object, the object is the data to save (create) in this document. In both cases, the id for the document is generated automatically.\n * @param {Object} value (Optional) The data to save. If `key` is a string, this is the value to save. If `key` is an object, the value(s) to save are already part of `key` and this argument is not required.\n * @param {Object} options (Optional) Overrides for configuration options. If you want to override the default `root` of the collection, do so here.\n * @return {Promise} \n */\n save: function (key, value, options) {\n var attrs;\n if (typeof key === 'object') {\n attrs = key;\n options = value;\n } else {\n (attrs = {})[key] = value;\n }\n var httpOptions = $.extend(true, {}, serviceOptions, options);\n httpOptions.url = getURL('', httpOptions.root);\n\n return http.post(attrs, httpOptions);\n },\n\n /**\n * Save (create or replace) data in a named document or element within the collection. \n * \n * The `root` of the collection must be specified. By default the `root` is taken from the Data Service configuration options; you can also pass the `root` to the `saveAs` call explicitly by overriding the options (third parameter).\n *\n * Optionally, the named document or element can include path information, so that you are saving just part of the document.\n *\n * (Additional background: Documents are top-level elements within a collection. Collections must be unique within this account (team or personal account) and project and are set with the `root` field in the `option` parameter. See the underlying [Data API](../../../rest_apis/data_api/) for more information. The `saveAs` method is making a `PUT` request.)\n *\n * **Example**\n *\n * // Create (or replace) the `user1` document at the default root level.\n * // Note that this replaces any existing content in the `user1` document.\n * ds.saveAs('user1',\n * { 'question1': 2, 'question2': 10,\n * 'question3': false, 'question4': 'sometimes' } );\n *\n * // Create (or replace) the `student1` document at the `students` root, \n * // that is, the data at `/students/student1/`.\n * // Note that this replaces any existing content in the `/students/student1/` document.\n * // However, this will keep existing content in other paths of this collection.\n * // For example, the data at `/students/student2/` is unchanged by this call.\n * ds.saveAs('student1',\n * { firstName: 'john', lastName: 'smith' },\n * { root: 'students' });\n *\n * // Create (or replace) the `mgmt100/groupB` document at the `myclasses` root,\n * // that is, the data at `/myclasses/mgmt100/groupB/`.\n * // Note that this replaces any existing content in the `/myclasses/mgmt100/groupB/` document.\n * // However, this will keep existing content in other paths of this collection.\n * // For example, the data at `/myclasses/mgmt100/groupA/` is unchanged by this call.\n * ds.saveAs('mgmt100/groupB',\n * { scenarioYear: '2015' },\n * { root: 'myclasses' });\n *\n * **Parameters**\n *\n * @param {String} key Id of the document.\n * @param {Object} value (Optional) The data to save, in key:value pairs.\n * @param {Object} options (Optional) Overrides for configuration options. If you want to override the default `root` of the collection, do so here.\n * @return {Promise} \n */\n saveAs: function (key, value, options) {\n var httpOptions = $.extend(true, {}, serviceOptions, options);\n httpOptions.url = getURL(key, httpOptions.root);\n\n return http.put(value, httpOptions);\n },\n\n /**\n * Get data for a specific document or field.\n *\n * **Example**\n *\n * ds.load('user1');\n * ds.load('user1/question3');\n *\n * **Parameters**\n * @param {String|Object} key The id of the data to return. Can be the id of a document, or a path to data within that document.\n * @param {Object} outputModifier (Optional) Available fields include: `startrecord`, `endrecord`, `sort`, and `direction` (`asc` or `desc`).\n * @param {Object} options Overrides for configuration options.\n * @return {Promise} \n */\n load: function (key, outputModifier, options) {\n var httpOptions = $.extend(true, {}, serviceOptions, options);\n httpOptions.url = getURL(key, httpOptions.root);\n return http.get(outputModifier, httpOptions);\n },\n\n /**\n * Removes data from collection. Only documents (top-level elements in each collection) can be deleted.\n *\n * **Example**\n *\n * ds.remove('user1');\n *\n *\n * **Parameters**\n *\n * @param {String|Array} keys The id of the document to remove from this collection, or an array of such ids.\n * @param {Object} options (Optional) Overrides for configuration options.\n * @return {Promise} \n */\n remove: function (keys, options) {\n var httpOptions = $.extend(true, {}, serviceOptions, options);\n var params;\n if ($.isArray(keys)) {\n params = { id: keys };\n } else {\n params = '';\n httpOptions.url = getURL(keys, httpOptions.root);\n }\n return http.delete(params, httpOptions);\n }\n\n // Epicenter doesn't allow nuking collections\n // /**\n // * Removes collection being referenced\n // * @return null\n // */\n // destroy: function (options) {\n // return this.remove('', options);\n // }\n };\n\n $.extend(this, publicAPI);\n};\n","/**\n *\n * ## Group API Adapter\n *\n * The Group API Adapter provides methods to look up, create, change or remove information about groups in a project. It is based on query capabilities of the underlying RESTful [Group API](../../../rest_apis/user_management/group/).\n *\n * This is only needed for Authenticated projects, that is, team projects with [end users and groups](../../../groups_and_end_users/).\n *\n * var ma = new F.service.Group({ token: 'user-or-project-access-token' });\n * ma.getGroupsForProject({ account: 'acme', project: 'sample' });\n */\n\n'use strict';\n\nvar serviceUtils = require('./service-utils');\nvar TransportFactory = require('../transport/http-transport-factory');\nvar objectAssign = require('object-assign');\n\nvar apiEndpoint = 'group/local';\n\nvar GroupService = function (config) {\n var defaults = {\n /**\n * Epicenter account name. Defaults to undefined.\n * @type {string}\n */\n account: undefined,\n\n /**\n * Epicenter project name. Defaults to undefined.\n * @type {string}\n */\n project: undefined,\n\n /**\n * Options to pass on to the underlying transport layer. All jquery.ajax options at http://api.jquery.com/jQuery.ajax/ are available. Defaults to empty object.\n * @type {object}\n */\n transport: {}\n };\n var serviceOptions = serviceUtils.getDefaultOptions(defaults, config, { apiEndpoint: apiEndpoint });\n var transportOptions = serviceOptions.transport;\n delete serviceOptions.transport;\n var http = new TransportFactory(transportOptions, serviceOptions);\n var publicAPI = {\n /**\n * Gets information for a group or multiple groups.\n * @param {Object} params object with query parameters\n * @patam {string} params.q partial match for name, organization or event.\n * @patam {string} params.account Epicenter's Team ID\n * @patam {string} params.project Epicenter's Project ID\n * @patam {string} params.name Epicenter's Group Name\n * @param {Object} options (Optional) Overrides for configuration options.\n * @return {Promise}\n */\n getGroups: function (params, options) {\n //groupID is part of the URL\n //q, account and project are part of the query string\n var finalOpts = objectAssign({}, serviceOptions, options);\n var finalParams;\n if (typeof params === 'string') {\n finalOpts.url = serviceUtils.getApiUrl(apiEndpoint + '/' + params, finalOpts);\n } else {\n finalParams = params;\n }\n return http.get(finalParams, finalOpts);\n }\n };\n objectAssign(this, publicAPI);\n};\n\nmodule.exports = GroupService;\n","/**\n *\n * ## Introspection API Service\n *\n * The Introspection API Service allows you to view a list of the variables and operations in a model. Typically used in conjunction with the [Run API Service](../run-api-service/).\n *\n * The Introspection API Service is not available for Forio SimLang.\n *\n * var intro = new F.service.Introspect({\n * account: 'acme-simulations',\n * project: 'supply-chain-game'\n * });\n * intro.byModel('supply-chain.py').then(function(data){ ... });\n * intro.byRunID('2b4d8f71-5c34-435a-8c16-9de674ab72e6').then(function(data){ ... });\n *\n */\n\n'use strict';\n\nvar ConfigService = require('./configuration-service');\nvar TransportFactory = require('../transport/http-transport-factory');\nvar SessionManager = require('../store/session-manager');\n\nvar apiEndpoint = 'model/introspect';\n\nmodule.exports = function (config) {\n var defaults = {\n /**\n * For projects that require authentication, pass in the user access token (defaults to empty string). If the user is already logged in to Epicenter, the user access token is already set in a cookie and automatically loaded from there. (See [more background on access tokens](../../../project_access/)).\n * @see [Authentication API Service](../auth-api-service/) for getting tokens.\n * @type {String}\n */\n token: undefined,\n\n /**\n * The account id. In the Epicenter UI, this is the **Team ID** (for team projects) or **User ID** (for personal projects). Defaults to empty string. If left undefined, taken from the URL.\n * @type {String}\n */\n account: undefined,\n\n /**\n * The project id. Defaults to empty string. If left undefined, taken from the URL.\n * @type {String}\n */\n project: undefined,\n\n };\n\n var sessionManager = new SessionManager();\n var serviceOptions = sessionManager.getMergedOptions(defaults, config);\n\n var urlConfig = new ConfigService(serviceOptions).get('server');\n if (serviceOptions.account) {\n urlConfig.accountPath = serviceOptions.account;\n }\n if (serviceOptions.project) {\n urlConfig.projectPath = serviceOptions.project;\n }\n\n var transportOptions = $.extend(true, {}, serviceOptions.transport, {\n url: urlConfig.getAPIPath(apiEndpoint)\n });\n if (serviceOptions.token) {\n transportOptions.headers = {\n Authorization: 'Bearer ' + serviceOptions.token\n };\n }\n var http = new TransportFactory(transportOptions);\n\n var publicAPI = {\n /**\n * Get the available variables and operations for a given model file.\n *\n * Note: This does not work for any model which requires additional parameters, such as `files`.\n *\n * **Example**\n *\n * intro.byModel('abc.vmf')\n * .then(function(data) {\n * // data contains an object with available functions (used with operations API) and available variables (used with variables API)\n * console.log(data.functions);\n * console.log(data.variables);\n * });\n *\n * **Parameters**\n * @param {String} modelFile Name of the model file to introspect.\n * @param {Object} options (Optional) Overrides for configuration options.\n * @return {Promise} \n */\n byModel: function (modelFile, options) {\n var opts = $.extend(true, {}, serviceOptions, options);\n if (!opts.account || !opts.project) {\n throw new Error('Account and project are required when using introspect#byModel');\n }\n if (!modelFile) {\n throw new Error('modelFile is required when using introspect#byModel');\n }\n var url = { url: urlConfig.getAPIPath(apiEndpoint) + [opts.account, opts.project, modelFile].join('/') };\n var httpOptions = $.extend(true, {}, serviceOptions, options, url);\n return http.get('', httpOptions);\n },\n\n /**\n * Get the available variables and operations for a given model file.\n *\n * Note: This does not work for any model which requires additional parameters such as `files`.\n *\n * **Example**\n *\n * intro.byRunID('2b4d8f71-5c34-435a-8c16-9de674ab72e6')\n * .then(function(data) {\n * // data contains an object with available functions (used with operations API) and available variables (used with variables API)\n * console.log(data.functions);\n * console.log(data.variables);\n * });\n *\n * **Parameters**\n * @param {String} runID Id of the run to introspect.\n * @param {Object} options (Optional) Overrides for configuration options.\n * @return {Promise} \n */\n byRunID: function (runID, options) {\n if (!runID) {\n throw new Error('runID is required when using introspect#byModel');\n }\n var url = { url: urlConfig.getAPIPath(apiEndpoint) + runID };\n var httpOptions = $.extend(true, {}, serviceOptions, options, url);\n return http.get('', httpOptions);\n }\n };\n $.extend(this, publicAPI);\n};\n","/**\n *\n * ## Member API Adapter\n *\n * The Member API Adapter provides methods to look up information about end users for your project and how they are divided across groups. It is based on query capabilities of the underlying RESTful [Member API](../../../rest_apis/user_management/member/).\n *\n * This is only needed for Authenticated projects, that is, team projects with [end users and groups](../../../groups_and_end_users/). For example, if some of your end users are facilitators, or if your end users should be treated differently based on which group they are in, use the Member API to find that information.\n *\n * var ma = new F.service.Member({ token: 'user-or-project-access-token' });\n * ma.getGroupsForUser({ userId: 'b6b313a3-ab84-479c-baea-206f6bff337' });\n * ma.getGroupDetails({ groupId: '00b53308-9833-47f2-b21e-1278c07d53b8' });\n */\n\n'use strict';\n\nvar ConfigService = require('./configuration-service');\nvar TransportFactory = require('../transport/http-transport-factory');\nvar SessionManager = require('../store/session-manager');\nvar _pick = require('../util/object-util')._pick;\nvar apiEndpoint = 'member/local';\n\nmodule.exports = function (config) {\n var defaults = {\n /**\n * Epicenter user id. Defaults to a blank string.\n * @type {string}\n */\n userId: undefined,\n\n /**\n * Epicenter group id. Defaults to a blank string. Note that this is the group *id*, not the group *name*.\n * @type {string}\n */\n groupId: undefined,\n\n /**\n * Options to pass on to the underlying transport layer. All jquery.ajax options at http://api.jquery.com/jQuery.ajax/ are available. Defaults to empty object.\n * @type {object}\n */\n transport: {}\n };\n this.sessionManager = new SessionManager();\n var serviceOptions = this.sessionManager.getMergedOptions(defaults, config);\n var urlConfig = new ConfigService(serviceOptions).get('server');\n\n var transportOptions = $.extend(true, {}, serviceOptions.transport, {\n url: urlConfig.getAPIPath(apiEndpoint)\n });\n\n if (serviceOptions.token) {\n transportOptions.headers = {\n Authorization: 'Bearer ' + serviceOptions.token\n };\n }\n var http = new TransportFactory(transportOptions, serviceOptions);\n\n var getFinalParams = function (params) {\n if (typeof params === 'object') {\n return $.extend(true, serviceOptions, params);\n }\n return serviceOptions;\n };\n\n var patchUserActiveField = function (params, active, options) {\n var httpOptions = $.extend(true, serviceOptions, options, {\n url: urlConfig.getAPIPath(apiEndpoint) + params.groupId + '/' + params.userId\n });\n\n return http.patch({ active: active }, httpOptions);\n };\n\n var publicAPI = {\n\n /**\n * Retrieve details about all of the group memberships for one end user. The membership details are returned in an array, with one element (group record) for each group to which the end user belongs.\n *\n * In the membership array, each group record includes the group id, project id, account (team) id, and an array of members. However, only the user whose userId is included in the call is listed in the members array (regardless of whether there are other members in this group).\n *\n * **Example**\n *\n * var ma = new F.service.Member({ token: 'user-or-project-access-token' });\n * ma.getGroupsForUser('42836d4b-5b61-4fe4-80eb-3136e956ee5c')\n * .then(function(memberships){\n * for (var i=0; i 1,\n * // where variables.price has been persisted (recorded)\n * // in the model.\n * rs.query({\n * 'saved': 'true',\n * '.price': '>1'\n * },\n * {\n * startrecord: 2,\n * endrecord: 5\n * });\n *\n * **Parameters**\n * @param {Object} qs Query object. Each key can be a property of the run or the name of variable that has been saved in the run (prefaced by `variables.`). Each value can be a literal value, or a comparison operator and value. (See [more on filtering](../../../rest_apis/aggregate_run_api/#filters) allowed in the underlying Run API.) Querying for variables is available for runs [in memory](../../../run_persistence/#runs-in-memory) and for runs [in the database](../../../run_persistence/#runs-in-memory) if the variables are persisted (e.g. that have been `record`ed in your model or marked for saving in your [model context file](../../../model_code/context/)).\n * @param {Object} outputModifier (Optional) Available fields include: `startrecord`, `endrecord`, `sort`, and `direction` (`asc` or `desc`).\n * @param {Object} options (Optional) Overrides for configuration options.\n * @return {Promise}\n */\n query: function (qs, outputModifier, options) {\n var httpOptions = $.extend(true, {}, serviceOptions, { url: urlConfig.getFilterURL(qs) }, options);\n httpOptions = urlConfig.addAutoRestoreHeader(httpOptions);\n\n return http.splitGet(outputModifier, httpOptions).then(function (r) {\n return ($.isPlainObject(r) && Object.keys(r).length === 0) ? [] : r;\n });\n },\n\n /**\n * Returns particular runs, based on conditions specified in the `qs` object.\n *\n * Similar to `.query()`.\n *\n * **Parameters**\n * @param {Object} filter Filter object. Each key can be a property of the run or the name of variable that has been saved in the run (prefaced by `variables.`). Each value can be a literal value, or a comparison operator and value. (See [more on filtering](../../../rest_apis/aggregate_run_api/#filters) allowed in the underlying Run API.) Filtering for variables is available for runs [in memory](../../../run_persistence/#runs-in-memory) and for runs [in the database](../../../run_persistence/#runs-in-memory) if the variables are persisted (e.g. that have been `record`ed in your model or marked for saving in your [model context file](../../../model_code/context/)).\n * @param {Object} outputModifier (Optional) Available fields include: `startrecord`, `endrecord`, `sort`, and `direction` (`asc` or `desc`).\n * @param {Object} options (Optional) Overrides for configuration options.\n * @return {Promise}\n */\n filter: function (filter, outputModifier, options) {\n if ($.isPlainObject(serviceOptions.filter)) {\n $.extend(serviceOptions.filter, filter);\n } else {\n serviceOptions.filter = filter;\n }\n var httpOptions = $.extend(true, {}, serviceOptions, options);\n httpOptions = urlConfig.addAutoRestoreHeader(httpOptions);\n return http.splitGet(outputModifier, httpOptions).then(function (r) {\n return ($.isPlainObject(r) && Object.keys(r).length === 0) ? [] : r;\n });\n },\n\n /**\n * Get data for a specific run. This includes standard run data such as the account, model, project, and created and last modified dates. To request specific model variables or run record variables, pass them as part of the `filters` parameter.\n *\n * Note that if the run is [in memory](../../../run_persistence/#runs-in-memory), any model variables are available; if the run is [in the database](../../../run_persistence/#runs-in-db), only model variables that have been persisted — that is, `record`ed or saved in your model — are available.\n *\n * **Example**\n *\n * rs.load('bb589677-d476-4971-a68e-0c58d191e450', { include: ['.price', '.sales'] });\n *\n * **Parameters**\n * @param {String} runID The run id.\n * @param {Object} filters (Optional) Object containing filters and operation modifiers. Use key `include` to list model variables that you want to include in the response. Other available fields include: `startrecord`, `endrecord`, `sort`, and `direction` (`asc` or `desc`).\n * @param {Object} options (Optional) Overrides for configuration options.\n * @return {Promise}\n */\n load: function (runID, filters, options) {\n if (runID) {\n serviceOptions.filter = runID; //shouldn't be able to over-ride\n }\n var httpOptions = $.extend(true, {}, serviceOptions, options);\n httpOptions = urlConfig.addAutoRestoreHeader(httpOptions);\n return http.get(filters, httpOptions);\n },\n\n /**\n * Removes specified runid from memory\n *\n * See [details on run persistence](../../../run_persistence/#runs-in-memory)\n * @param {String} [runID] id of run to remove\n * @param {Object} [filters] (Optional) Object containing filters and operation modifiers. Use key `include` to list model variables that you want to include in the response. Other available fields include: `startrecord`, `endrecord`, `sort`, and `direction` (`asc` or `desc`).\n * @param {Object} [options] (Optional) Overrides for configuration options.\n * @return {Promise}\n */\n removeFromMemory: function (runID, filters, options) {\n var httpOptions = $.extend(true, {}, serviceOptions, options);\n if (runID) {\n httpOptions.url = urlConfig.getAPIPath('run') + runID;\n }\n return http.delete({}, httpOptions);\n },\n\n /**\n * Save attributes (data, model variables) of the run.\n *\n * **Examples**\n *\n * // add 'completed' field to run record\n * rs.save({ completed: true });\n *\n * // update 'saved' field of run record, and update values of model variables for this run\n * rs.save({ saved: true, variables: { a: 23, b: 23 } });\n *\n * // update 'saved' field of run record for a particular run\n * rs.save({ saved: true }, { id: '0000015bf2a04995880df6b868d23eb3d229' });\n *\n * **Parameters**\n * @param {Object} attributes The run data and variables to save.\n * @param {Object} attributes.variables Model variables must be included in a `variables` field within the `attributes` object. (Otherwise they are treated as run data and added to the run record directly.)\n * @param {Object} options (Optional) Overrides for configuration options.\n * @return {Promise}\n */\n save: function (attributes, options) {\n var httpOptions = $.extend(true, {}, serviceOptions, options);\n setFilterOrThrowError(httpOptions);\n return http.patch(attributes, httpOptions);\n },\n\n /**\n * Call an operation from the model.\n *\n * Depending on the language in which you have written your model, the operation (function or method) may need to be exposed (e.g. `export` for a Julia model) in the model file in order to be called through the API. See [Writing your Model](../../../writing_your_model/)).\n *\n * The `params` argument is normally an array of arguments to the `operation`. In the special case where `operation` only takes one argument, you are not required to put that argument into an array.\n *\n * Note that you can combine the `operation` and `params` arguments into a single object if you prefer, as in the last example.\n *\n * **Examples**\n *\n * // operation \"solve\" takes no arguments\n * rs.do('solve');\n * // operation \"echo\" takes one argument, a string\n * rs.do('echo', ['hello']);\n * // operation \"echo\" takes one argument, a string\n * rs.do('echo', 'hello');\n * // operation \"sumArray\" takes one argument, an array\n * rs.do('sumArray', [[4,2,1]]);\n * // operation \"add\" takes two arguments, both integers\n * rs.do({ name:'add', params:[2,4] });\n * // call operation \"solve\" on a different run \n * rs.do('solve', { id: '0000015bf2a04995880df6b868d23eb3d229' });\n *\n * **Parameters**\n * @param {String} operation Name of operation.\n * @param {Array} params (Optional) Any parameters the operation takes, passed as an array. In the special case where `operation` only takes one argument, you are not required to put that argument into an array, and can just pass it directly.\n * @param {Object} options (Optional) Overrides for configuration options.\n * @return {Promise}\n */\n do: function (operation, params, options) {\n // console.log('do', operation, params);\n var opsArgs;\n var postOptions;\n if (options) {\n opsArgs = params;\n postOptions = options;\n } else if ($.isPlainObject(params)) {\n opsArgs = null;\n postOptions = params;\n } else {\n opsArgs = params;\n }\n var result = rutil.normalizeOperations(operation, opsArgs);\n var httpOptions = $.extend(true, {}, serviceOptions, postOptions);\n\n setFilterOrThrowError(httpOptions);\n\n var prms = (result.args[0].length && (result.args[0] !== null && result.args[0] !== undefined)) ? result.args[0] : [];\n return http.post({ arguments: prms }, $.extend(true, {}, httpOptions, {\n url: urlConfig.getFilterURL() + 'operations/' + result.ops[0] + '/'\n }));\n },\n\n /**\n * Call several operations from the model, sequentially.\n *\n * Depending on the language in which you have written your model, the operation (function or method) may need to be exposed (e.g. `export` for a Julia model) in the model file in order to be called through the API. See [Writing your Model](../../../writing_your_model/)).\n *\n * **Examples**\n *\n * // operations \"initialize\" and \"solve\" do not take any arguments\n * rs.serial(['initialize', 'solve']);\n * // operations \"init\" and \"reset\" take two arguments each\n * rs.serial([ { name: 'init', params: [1,2] },\n * { name: 'reset', params: [2,3] }]);\n * // operation \"init\" takes two arguments,\n * // operation \"runmodel\" takes none\n * rs.serial([ { name: 'init', params: [1,2] },\n * { name: 'runmodel', params: [] }]);\n *\n * **Parameters**\n * @param {Array} operations If none of the operations take parameters, pass an array of the operation names (strings). If any of the operations do take parameters, pass an array of objects, each of which contains an operation name and its own (possibly empty) array of parameters.\n * @param {*} params Parameters to pass to operations.\n * @param {Object} options (Optional) Overrides for configuration options.\n * @return {Promise} The parameter to the callback is an array. Each array element is an object containing the results of one operation.\n */\n serial: function (operations, params, options) {\n var opParams = rutil.normalizeOperations(operations, params);\n var ops = opParams.ops;\n var args = opParams.args;\n var me = this;\n\n var $d = $.Deferred();\n var postOptions = $.extend(true, {}, serviceOptions, options);\n\n var responses = [];\n var doSingleOp = function () {\n var op = ops.shift();\n var arg = args.shift();\n\n me.do(op, arg, {\n success: function (result) {\n responses.push(result);\n if (ops.length) {\n doSingleOp();\n } else {\n $d.resolve(responses);\n postOptions.success(responses, me);\n }\n },\n error: function (err) {\n responses.push(err);\n $d.reject(responses);\n postOptions.error(responses, me);\n }\n });\n };\n\n doSingleOp();\n\n return $d.promise();\n },\n\n /**\n * Call several operations from the model, executing them in parallel.\n *\n * Depending on the language in which you have written your model, the operation (function or method) may need to be exposed (e.g. `export` for a Julia model) in the model file in order to be called through the API. See [Writing your Model](../../../writing_your_model/)).\n *\n * **Example**\n *\n * // operations \"solve\" and \"reset\" do not take any arguments\n * rs.parallel(['solve', 'reset']);\n * // operations \"add\" and \"subtract\" take two arguments each\n * rs.parallel([ { name: 'add', params: [1,2] },\n * { name: 'subtract', params:[2,3] }]);\n * // operations \"add\" and \"subtract\" take two arguments each\n * rs.parallel({ add: [1,2], subtract: [2,4] });\n *\n * **Parameters**\n * @param {Array|Object} operations If none of the operations take parameters, pass an array of the operation names (as strings). If any of the operations do take parameters, you have two options. You can pass an array of objects, each of which contains an operation name and its own (possibly empty) array of parameters. Alternatively, you can pass a single object with the operation name and a (possibly empty) array of parameters.\n * @param {*} params Parameters to pass to operations.\n * @param {Object} options (Optional) Overrides for configuration options.\n * @return {Promise} The parameter to the callback is an array. Each array element is an object containing the results of one operation.\n */\n parallel: function (operations, params, options) {\n var $d = $.Deferred();\n\n var opParams = rutil.normalizeOperations(operations, params);\n var ops = opParams.ops;\n var args = opParams.args;\n var postOptions = $.extend(true, {}, serviceOptions, options);\n\n var queue = [];\n for (var i = 0; i < ops.length; i++) {\n queue.push(\n this.do(ops[i], args[i])\n );\n }\n\n var me = this;\n $.when.apply(this, queue)\n .then(function () {\n var args = Array.prototype.slice.call(arguments);\n var actualResponse = args.map(function (a) {\n return a[0];\n });\n $d.resolve(actualResponse);\n postOptions.success(actualResponse, me);\n })\n .fail(function () {\n var args = Array.prototype.slice.call(arguments);\n var actualResponse = args.map(function (a) {\n return a[0];\n });\n $d.reject(actualResponse);\n postOptions.error(actualResponse, me);\n });\n\n return $d.promise();\n },\n\n /**\n * Shortcut to using the [Introspection API Service](../introspection-api-service/). Allows you to view a list of the variables and operations in a model.\n *\n * **Example**\n *\n * rs.introspect({ runID: 'cbf85437-b539-4977-a1fc-23515cf071bb' }).then(function (data) {\n * console.log(data.functions);\n * console.log(data.variables);\n * });\n *\n * **Parameters**\n * @param {Object} options Options can either be of the form `{ runID: }` or `{ model: }`. Note that the `runID` is optional if the Run Service is already associated with a particular run (because `id` was passed in when the Run Service was initialized). If provided, the `runID` overrides the `id` currently associated with the Run Service.\n * @param {Object} introspectionConfig (Optional) Service options for Introspection Service\n * @return {Promise}\n */\n introspect: function (options, introspectionConfig) {\n var introspection = new IntrospectionService($.extend(true, {}, serviceOptions, introspectionConfig));\n if (options) {\n if (options.runID) {\n return introspection.byRunID(options.runID);\n } else if (options.model) {\n return introspection.byModel(options.model);\n }\n } else if (serviceOptions.id) {\n return introspection.byRunID(serviceOptions.id);\n } else {\n throw new Error('Please specify either the model or runid to introspect');\n }\n }\n };\n\n var publicSyncAPI = {\n getCurrentConfig: function () {\n return serviceOptions;\n },\n updateConfig: function (config) {\n if (config && config.id) {\n config.filter = config.id;\n } else if (config && config.filter) {\n config.id = config.filter;\n }\n serviceOptions = $.extend(true, {}, serviceOptions, config);\n urlConfig = updateURLConfig(serviceOptions);\n this.urlConfig = urlConfig;\n updateHTTPConfig(serviceOptions, urlConfig);\n },\n /**\n * Returns a Variables Service instance. Use the variables instance to load, save, and query for specific model variables. See the [Variable API Service](../variables-api-service/) for more information.\n *\n * **Example**\n *\n * var vs = rs.variables();\n * vs.save({ sample_int: 4 });\n *\n * **Parameters**\n * @param {Object} config (Optional) Overrides for configuration options.\n * @return {Object} variablesService Instance\n */\n variables: function (config) {\n var vs = new VariablesService($.extend(true, {}, serviceOptions, config, {\n runService: this\n }));\n return vs;\n }\n };\n\n $.extend(this, publicAsyncAPI);\n $.extend(this, publicSyncAPI);\n};\n","'use strict';\n\nvar ConfigService = require('./configuration-service');\nvar SessionManager = require('../store/session-manager');\nvar objectAssign = require('object-assign');\n\nvar serviceUtils = {\n /*\n * Gets the default options for a api service.\n * It will merge:\n * - The Session options (Using the Session Manager)\n * - The Authorization Header from the token option\n * - The full url from the endpoint option\n * With the supplied overrides and defaults\n *\n */\n getDefaultOptions: function (defaults) {\n var rest = Array.prototype.slice.call(arguments, 1);\n var sessionManager = new SessionManager();\n var serviceOptions = sessionManager.getMergedOptions.apply(sessionManager, [defaults].concat(rest));\n\n serviceOptions.transport = objectAssign({}, serviceOptions.transport, {\n url: this.getApiUrl(serviceOptions.apiEndpoint, serviceOptions)\n });\n\n if (serviceOptions.token) {\n serviceOptions.transport.headers = {\n Authorization: 'Bearer ' + serviceOptions.token\n };\n }\n return serviceOptions;\n },\n\n getApiUrl: function (apiEndpoint, serviceOptions) {\n var urlConfig = new ConfigService(serviceOptions).get('server');\n return urlConfig.getAPIPath(apiEndpoint);\n }\n};\n\nmodule.exports = serviceUtils;","'use strict';\n/**\n * ## State API Adapter\n *\n * The State API Adapter allows you to view the history of a run, and to replay or clone runs. \n *\n * The State API Adapter brings existing, persisted run data from the database back into memory, using the same run id (`replay`) or a new run id (`clone`). Runs must be in memory in order for you to update variables or call operations on them.\n *\n * Specifically, the State API Adapter works by \"re-running\" the run (user interactions) from the creation of the run up to the time it was last persisted in the database. This process uses the current version of the run's model. Therefore, if the model has changed since the original run was created, the retrieved run will use the new model — and may end up having different values or behavior as a result. Use with care!\n *\n * To use the State API Adapter, instantiate it and then call its methods:\n *\n * var sa = new F.service.State();\n * sa.replay({runId: '1842bb5c-83ad-4ba8-a955-bd13cc2fdb4f'});\n *\n * The constructor takes an optional `options` parameter in which you can specify the `account` and `project` if they are not already available in the current context.\n *\n */\n\nvar ConfigService = require('./configuration-service');\nvar TransportFactory = require('../transport/http-transport-factory');\nvar _pick = require('../util/object-util')._pick;\nvar SessionManager = require('../store/session-manager');\nvar apiEndpoint = 'model/state';\n\nmodule.exports = function (config) {\n\n var defaults = {\n\n };\n\n this.sessionManager = new SessionManager();\n var serviceOptions = this.sessionManager.getMergedOptions(defaults, config);\n var urlConfig = new ConfigService(serviceOptions).get('server');\n\n var transportOptions = $.extend(true, {}, serviceOptions.transport, {\n url: urlConfig.getAPIPath(apiEndpoint)\n });\n\n if (serviceOptions.token) {\n transportOptions.headers = {\n Authorization: 'Bearer ' + serviceOptions.token\n };\n }\n\n var http = new TransportFactory(transportOptions);\n var parseRunIdOrError = function (params) {\n if ($.isPlainObject(params) && params.runId) {\n return params.runId;\n } else {\n throw new Error('Please pass in a run id');\n }\n };\n\n var publicAPI = {\n\n /**\n * View the history of a run.\n * \n * **Example**\n *\n * var sa = new F.service.State();\n * sa.load('0000015a06bb58613b28b57365677ec89ec5').then(function(history) {\n * console.log('history = ', history);\n * });\n *\n * **Parameters**\n * @param {string} runId The id of the run.\n * @param {object} options (Optional) Overrides for configuration options.\n * @return {Promise}\n */\n load: function (runId, options) {\n var httpParams = $.extend(true, {},\n serviceOptions,\n options,\n { url: urlConfig.getAPIPath(apiEndpoint) + runId }\n );\n return http.get('', httpParams);\n },\n\n /**\n * Replay a run. After this call, the run, with its original run id, is now available [in memory](../../../run_persistence/#runs-in-memory). (It continues to be persisted into the Epicenter database at regular intervals.)\n *\n * **Example**\n *\n * var sa = new F.service.State();\n * sa.replay({runId: '1842bb5c-83ad-4ba8-a955-bd13cc2fdb4f', stopBefore: 'calculateScore'});\n *\n * **Parameters**\n * @param {object} params Parameters object.\n * @param {string} params.runId The id of the run to bring back to memory.\n * @param {string} params.stopBefore (Optional) The run is advanced only up to the first occurrence of this method.\n * @param {array} params.exclude (Optional) Array of methods to exclude when advancing the run.\n * @param {object} options (Optional) Overrides for configuration options.\n * @return {Promise}\n */\n replay: function (params, options) {\n var runId = parseRunIdOrError(params);\n\n var replayOptions = $.extend(true, {},\n serviceOptions,\n options,\n { url: urlConfig.getAPIPath(apiEndpoint) + runId }\n );\n\n params = $.extend(true, { action: 'replay' }, _pick(params, ['stopBefore', 'exclude']));\n\n return http.post(params, replayOptions);\n },\n\n /**\n * Clone a given run and return a new run in the same state as the given run.\n *\n * The new run id is now available [in memory](../../../run_persistence/#runs-in-memory). The new run includes a copy of all of the data from the original run, EXCEPT:\n *\n * * The `saved` field in the new run record is not copied from the original run record. It defaults to `false`.\n * * The `initialized` field in the new run record is not copied from the original run record. It defaults to `false` but may change to `true` as the new run is advanced. For example, if there has been a call to the `step` function (for Vensim models), the `initialized` field is set to `true`.\n * * The `created` field in the new run record is the date and time at which the clone was created (not the time that the original run was created.)\n *\n * The original run remains only [in the database](../../../run_persistence/#runs-in-db).\n *\n * **Example**\n *\n * var sa = new F.service.State();\n * sa.clone({runId: '1842bb5c-83ad-4ba8-a955-bd13cc2fdb4f', stopBefore: 'calculateScore', exclude: ['interimCalculation'] });\n *\n * **Parameters**\n * @param {object} params Parameters object.\n * @param {string} params.runId The id of the run to clone from memory.\n * @param {string} params.stopBefore (Optional) The newly cloned run is advanced only up to the first occurrence of this method.\n * @param {array} params.exclude (Optional) Array of methods to exclude when advancing the newly cloned run.\n * @param {object} options (Optional) Overrides for configuration options.\n * @return {Promise}\n */\n clone: function (params, options) {\n var runId = parseRunIdOrError(params);\n\n var replayOptions = $.extend(true, {},\n serviceOptions,\n options,\n { url: urlConfig.getAPIPath(apiEndpoint) + runId }\n );\n\n params = $.extend(true, { action: 'clone' }, _pick(params, ['stopBefore', 'exclude']));\n\n return http.post(params, replayOptions);\n }\n };\n\n $.extend(this, publicAPI);\n};\n","'use strict';\n\nvar epiVersion = require('../api-version.json');\n\n//TODO: urlutils to get host, since no window on node\nvar defaults = {\n host: window.location.host,\n pathname: window.location.pathname\n};\n\nfunction getLocalHost(existingFn, host) {\n var localHostFn;\n if (existingFn !== undefined) {\n if (!$.isFunction(existingFn)) {\n localHostFn = function () { return existingFn; };\n } else {\n localHostFn = existingFn;\n }\n } else {\n localHostFn = function () {\n var isLocal = !host || //phantomjs\n host === '127.0.0.1' || \n host.indexOf('local.') === 0 || \n host.indexOf('localhost') === 0;\n return isLocal;\n };\n }\n return localHostFn;\n}\n\nvar UrlConfigService = function (config) {\n var envConf = UrlConfigService.defaults;\n\n if (!config) {\n config = {};\n }\n // console.log(this.defaults);\n var overrides = $.extend({}, envConf, config);\n var options = $.extend({}, defaults, overrides);\n\n overrides.isLocalhost = options.isLocalhost = getLocalHost(options.isLocalhost, options.host);\n \n // console.log(isLocalhost(), '___________');\n var actingHost = config && config.host;\n if (!actingHost && options.isLocalhost()) {\n actingHost = 'forio.com';\n } else {\n actingHost = options.host;\n }\n\n var API_PROTOCOL = 'https';\n var HOST_API_MAPPING = {\n 'forio.com': 'api.forio.com',\n 'foriodev.com': 'api.epicenter.foriodev.com'\n };\n\n var publicExports = {\n protocol: API_PROTOCOL,\n\n api: '',\n\n //TODO: this should really be called 'apihost', but can't because that would break too many things\n host: (function () {\n var apiHost = (HOST_API_MAPPING[actingHost]) ? HOST_API_MAPPING[actingHost] : actingHost;\n // console.log(actingHost, config, apiHost);\n return apiHost;\n }()),\n\n isCustomDomain: (function () {\n var path = options.pathname.split('\\/');\n var pathHasApp = path && path[1] === 'app';\n return (!options.isLocalhost() && !pathHasApp);\n }()),\n\n appPath: (function () {\n var path = options.pathname.split('\\/');\n\n return path && path[1] || '';\n }()),\n\n accountPath: (function () {\n var accnt = '';\n var path = options.pathname.split('\\/');\n if (path && path[1] === 'app') {\n accnt = path[2];\n }\n return accnt;\n }()),\n\n projectPath: (function () {\n var prj = '';\n var path = options.pathname.split('\\/');\n if (path && path[1] === 'app') {\n prj = path[3]; //eslint-disable-line no-magic-numbers\n }\n return prj;\n }()),\n\n versionPath: (function () {\n var version = epiVersion.version ? epiVersion.version + '/' : '';\n return version;\n }()),\n\n getAPIPath: function (api) {\n var PROJECT_APIS = ['run', 'data', 'file', 'presence'];\n var apiMapping = {\n channel: 'channel/subscribe'\n };\n var apiEndpoint = apiMapping[api] || api;\n \n if (apiEndpoint === 'config') {\n var actualProtocol = window.location.protocol.replace(':', '');\n var configProtocol = (options.isLocalhost()) ? this.protocol : actualProtocol;\n return configProtocol + '://' + actingHost + '/epicenter/' + this.versionPath + 'config';\n }\n var apiPath = this.protocol + '://' + this.host + '/' + this.versionPath + apiEndpoint + '/';\n\n if ($.inArray(apiEndpoint, PROJECT_APIS) !== -1) {\n apiPath += this.accountPath + '/' + this.projectPath + '/';\n }\n return apiPath;\n }\n };\n\n\n $.extend(publicExports, overrides);\n return publicExports;\n};\n// This data can be set by external scripts, for loading from an env server for eg;\nUrlConfigService.defaults = {};\n\nmodule.exports = UrlConfigService;\n","'use strict';\n/**\n* ## User API Adapter\n*\n* The User API Adapter allows you to retrieve details about end users in your team (account). It is based on the querying capabilities of the underlying RESTful [User API](../../../rest_apis/user_management/user/).\n*\n* To use the User API Adapter, instantiate it and then call its methods.\n*\n* var ua = new F.service.User({\n* account: 'acme-simulations',\n* token: 'user-or-project-access-token'\n* });\n* ua.getById('42836d4b-5b61-4fe4-80eb-3136e956ee5c');\n* ua.get({ userName: 'jsmith' });\n* ua.get({ id: ['42836d4b-5b61-4fe4-80eb-3136e956ee5c',\n* '4ea75631-4c8d-4872-9d80-b4600146478e'] });\n*\n* The constructor takes an optional `options` parameter in which you can specify the `account` and `token` if they are not already available in the current context.\n*/\n\nvar ConfigService = require('./configuration-service');\nvar TransportFactory = require('../transport/http-transport-factory');\nvar SessionManager = require('../store/session-manager');\nvar qutil = require('../util/query-util');\n\nmodule.exports = function (config) {\n var defaults = {\n\n /**\n * The account id. In the Epicenter UI, this is the **Team ID** (for team projects) or **User ID** (for personal projects). Defaults to empty string.\n * @type {String}\n */\n account: undefined,\n\n /**\n * The access token to use when searching for end users. (See [more background on access tokens](../../../project_access/)).\n * @type {String}\n */\n token: undefined,\n\n /**\n * Options to pass on to the underlying transport layer. All jquery.ajax options at http://api.jquery.com/jQuery.ajax/ are available. Defaults to empty object.\n * @type {Object}\n */\n transport: {}\n };\n\n this.sessionManager = new SessionManager();\n var serviceOptions = this.sessionManager.getMergedOptions(defaults, config);\n var urlConfig = new ConfigService(serviceOptions).get('server');\n var transportOptions = $.extend(true, {}, serviceOptions.transport, {\n url: urlConfig.getAPIPath('user')\n });\n\n if (serviceOptions.token) {\n transportOptions.headers = {\n Authorization: 'Bearer ' + serviceOptions.token\n };\n }\n\n var http = new TransportFactory(transportOptions);\n\n var publicAPI = {\n\n /**\n * Retrieve details about particular end users in your team, based on user name or user id.\n *\n * **Example**\n *\n * var ua = new F.service.User({\n * account: 'acme-simulations',\n * token: 'user-or-project-access-token'\n * });\n * ua.get({ userName: 'jsmith' });\n * ua.get({ id: ['42836d4b-5b61-4fe4-80eb-3136e956ee5c',\n * '4ea75631-4c8d-4872-9d80-b4600146478e'] });\n *\n * **Parameters**\n * @param {object} filter Object with field `userName` and value of the username. Alternatively, object with field `id` and value of an array of user ids.\n * @param {object} options (Optional) Overrides for configuration options.\n * @return {Promise}\n */\n\n get: function (filter, options) {\n options = options || {};\n filter = filter || {};\n\n var getOptions = $.extend(true, {},\n serviceOptions,\n options\n );\n\n var toQFilter = function (filter) {\n var res = {};\n\n // API only supports filtering by username for now\n if (filter.userName) {\n res.q = filter.userName;\n }\n\n return res;\n };\n\n var toIdFilters = function (id) {\n if (!id) {\n return '';\n }\n\n id = $.isArray(id) ? id : [id];\n return 'id=' + id.join('&id=');\n };\n\n var getFilters = [\n 'account=' + getOptions.account,\n toIdFilters(filter.id),\n qutil.toQueryFormat(toQFilter(filter))\n ].join('&');\n\n // special case for queries with large number of ids\n // make it as a post with GET semantics\n var threshold = 30;\n if (filter.id && $.isArray(filter.id) && filter.id.length >= threshold) {\n getOptions.url = urlConfig.getAPIPath('user') + '?_method=GET';\n return http.post({ id: filter.id }, getOptions);\n } else {\n return http.get(getFilters, getOptions);\n }\n },\n\n /**\n * Retrieve details about a single end user in your team, based on user id.\n *\n * **Example**\n *\n * var ua = new F.service.User({\n * account: 'acme-simulations',\n * token: 'user-or-project-access-token'\n * });\n * ua.getById('42836d4b-5b61-4fe4-80eb-3136e956ee5c');\n *\n * **Parameters**\n * @param {string} userId The user id for the end user in your team.\n * @param {object} options (Optional) Overrides for configuration options.\n * @return {Promise}\n */\n\n getById: function (userId, options) {\n return publicAPI.get({ id: userId }, options);\n }\n };\n\n $.extend(this, publicAPI);\n};\n\n\n","/**\n *\n * ## Variables API Service\n *\n * Used in conjunction with the [Run API Service](../run-api-service/) to read, write, and search for specific model variables.\n *\n * var rm = new F.manager.RunManager({\n * run: {\n * account: 'acme-simulations',\n * project: 'supply-chain-game',\n * model: 'supply-chain-model.jl'\n * }\n * });\n * rm.getRun()\n * .then(function() {\n * var vs = rm.run.variables();\n * vs.save({sample_int: 4});\n * });\n *\n */\n\n\n 'use strict';\n\n var TransportFactory = require('../transport/http-transport-factory');\n var rutil = require('../util/run-util');\n\n module.exports = function (config) {\n var defaults = {\n /**\n * The runs object to which the variable filters apply. Defaults to null.\n * @type {runService}\n */\n runService: null\n };\n var serviceOptions = $.extend({}, defaults, config);\n\n var getURL = function () {\n //TODO: Replace with getCurrentconfig instead?\n return serviceOptions.runService.urlConfig.getFilterURL() + 'variables/';\n };\n\n var addAutoRestoreHeader = function (options) {\n return serviceOptions.runService.urlConfig.addAutoRestoreHeader(options);\n };\n\n var httpOptions = {\n url: getURL\n };\n if (serviceOptions.token) {\n httpOptions.headers = {\n Authorization: 'Bearer ' + serviceOptions.token\n };\n }\n var http = new TransportFactory(httpOptions);\n http.splitGet = rutil.splitGetFactory(httpOptions);\n\n var publicAPI = {\n\n /**\n * Get values for a variable.\n *\n * **Example**\n *\n * vs.load('sample_int')\n * .then(function(val){\n * // val contains the value of sample_int\n * });\n *\n * **Parameters**\n * @param {String} variable Name of variable to load.\n * @param {Object} outputModifier (Optional) Available fields include: `startrecord`, `endrecord`, `sort`, and `direction` (`asc` or `desc`).\n * @param {Object} options (Optional) Overrides for configuration options.\n * @return {Promise}\n */\n load: function (variable, outputModifier, options) {\n var httpOptions = $.extend(true, {}, serviceOptions, options);\n httpOptions = addAutoRestoreHeader(httpOptions);\n return http.get(outputModifier, $.extend({}, httpOptions, {\n url: getURL() + variable + '/'\n }));\n },\n\n /**\n * Returns particular variables, based on conditions specified in the `query` object.\n *\n * **Example**\n *\n * vs.query(['price', 'sales'])\n * .then(function(val) {\n * // val is an object with the values of the requested variables: val.price, val.sales\n * });\n *\n * vs.query({ include:['price', 'sales'] });\n *\n * **Parameters**\n * @param {Object|Array} query The names of the variables requested.\n * @param {Object} outputModifier (Optional) Available fields include: `startrecord`, `endrecord`, `sort`, and `direction` (`asc` or `desc`).\n * @param {Object} options (Optional) Overrides for configuration options.\n * @return {Promise}\n */\n query: function (query, outputModifier, options) {\n //Query and outputModifier are both querystrings in the url; only calling them out separately here to be consistent with the other calls\n var httpOptions = $.extend(true, {}, serviceOptions, options);\n httpOptions = addAutoRestoreHeader(httpOptions);\n\n if ($.isArray(query)) {\n query = { include: query };\n }\n $.extend(query, outputModifier);\n return http.splitGet(query, httpOptions);\n },\n\n /**\n * Save values to model variables. Overwrites existing values. Note that you can only update model variables if the run is [in memory](../../../run_persistence/#runs-in-memory). (An alternate way to update model variables is to call a method from the model and make sure that the method persists the variables. See `do`, `serial`, and `parallel` in the [Run API Service](../run-api-service/) for calling methods from the model.)\n *\n * **Example**\n *\n * vs.save('price', 4);\n * vs.save({ price: 4, quantity: 5, products: [2,3,4] });\n *\n * **Parameters**\n * @param {Object|String} variable An object composed of the model variables and the values to save. Alternatively, a string with the name of the variable.\n * @param {Object} val (Optional) If passing a string for `variable`, use this argument for the value to save.\n * @param {Object} options (Optional) Overrides for configuration options.\n * @return {Promise}\n */\n save: function (variable, val, options) {\n var attrs;\n if (typeof variable === 'object') {\n attrs = variable;\n options = val;\n } else {\n (attrs = {})[variable] = val;\n }\n var httpOptions = $.extend(true, {}, serviceOptions, options);\n\n return http.patch.call(this, attrs, httpOptions);\n }\n\n // Not Available until underlying API supports PUT. Otherwise save would be PUT and merge would be PATCH\n // *\n // * Save values to the api. Merges arrays, but otherwise same as save\n // * @param {Object|String} variable Object with attributes, or string key\n // * @param {Object} val Optional if prev parameter was a string, set value here\n // * @param {Object} options Overrides for configuration options\n // *\n // * @example\n // * vs.merge({ price: 4, quantity: 5, products: [2,3,4] })\n // * vs.merge('price', 4);\n\n // merge: function (variable, val, options) {\n // var attrs;\n // if (typeof variable === 'object') {\n // attrs = variable;\n // options = val;\n // } else {\n // (attrs = {})[variable] = val;\n // }\n // var httpOptions = $.extend(true, {}, serviceOptions, options);\n\n // return http.patch.call(this, attrs, httpOptions);\n // }\n };\n $.extend(this, publicAPI);\n };\n","/**\n * ## World API Adapter\n *\n * A [run](../../../glossary/#run) is a collection of end user interactions with a project and its model -- including setting variables, making decisions, and calling operations. For building multiplayer simulations you typically want multiple end users to share the same set of interactions, and work within a common state. Epicenter allows you to create \"worlds\" to handle such cases. Only [team projects](../../../glossary/#team) can be multiplayer.\n *\n * The World API Adapter allows you to create, access, and manipulate multiplayer worlds within your Epicenter project. You can use this to add and remove end users from the world, and to create, access, and remove their runs. Because of this, typically the World Adapter is used for facilitator pages in your project. (The related [World Manager](../world-manager/) provides an easy way to access runs and worlds for particular end users, so is typically used in pages that end users will interact with.)\n *\n * As with all the other [API Adapters](../../), all methods take in an \"options\" object as the last parameter. The options can be used to extend/override the World API Service defaults.\n *\n * To use the World Adapter, instantiate it and then access the methods provided. Instantiating requires the account id (**Team ID** in the Epicenter user interface), project id (**Project ID**), and group (**Group Name**).\n *\n * var wa = new F.service.World({\n * account: 'acme-simulations',\n * project: 'supply-chain-game',\n * group: 'team1' });\n * wa.create()\n * .then(function(world) {\n * // call methods, e.g. wa.addUsers()\n * });\n */\n\n'use strict';\n\nvar ConfigService = require('./configuration-service');\n// var qutil = require('../util/query-util');\nvar TransportFactory = require('../transport/http-transport-factory');\nvar SessionManager = require('../store/session-manager');\nvar _pick = require('../util/object-util')._pick;\n\nvar apiBase = 'multiplayer/';\nvar assignmentEndpoint = apiBase + 'assign';\nvar apiEndpoint = apiBase + 'world';\nvar projectEndpoint = apiBase + 'project';\n\nmodule.exports = function (config) {\n var defaults = {\n /**\n * For projects that require authentication, pass in the user access token (defaults to empty string). If the user is already logged in to Epicenter, the user access token is already set in a cookie and automatically loaded from there. (See [more background on access tokens](../../../project_access/)).\n * @see [Authentication API Service](../auth-api-service/) for getting tokens.\n * @type {String}\n */\n token: undefined,\n\n /**\n * The project id. If left undefined, taken from the URL.\n * @type {String}\n */\n project: undefined,\n\n /**\n * The account id. In the Epicenter UI, this is the **Team ID** (for team projects). If left undefined, taken from the URL.\n * @type {String}\n */\n account: undefined,\n\n /**\n * The group name. Defaults to undefined.\n * @type {String}\n */\n group: undefined,\n\n /**\n * The model file to use to create runs in this world. Defaults to undefined.\n * @type {String}\n */\n model: undefined,\n\n /**\n * Criteria by which to filter world. Currently only supports world-ids as filters.\n * @type {String}\n */\n filter: '',\n\n /**\n * Convenience alias for filter\n * @type {String}\n */\n id: '',\n\n /**\n * Options to pass on to the underlying transport layer. All jquery.ajax options at http://api.jquery.com/jQuery.ajax/ are available. Defaults to empty object.\n * @type {Object}\n */\n transport: {},\n\n /**\n * Called when the call completes successfully. Defaults to `$.noop`.\n * @type {function}\n */\n success: $.noop,\n\n /**\n * Called when the call fails. Defaults to `$.noop`.\n * @type {function}\n */\n error: $.noop\n };\n\n this.sessionManager = new SessionManager();\n var serviceOptions = this.sessionManager.getMergedOptions(defaults, config);\n if (serviceOptions.id) {\n serviceOptions.filter = serviceOptions.id;\n }\n\n var urlConfig = new ConfigService(serviceOptions).get('server');\n\n if (!serviceOptions.account) {\n serviceOptions.account = urlConfig.accountPath;\n }\n\n if (!serviceOptions.project) {\n serviceOptions.project = urlConfig.projectPath;\n }\n\n var transportOptions = $.extend(true, {}, serviceOptions.transport, {\n url: urlConfig.getAPIPath(apiEndpoint)\n });\n\n if (serviceOptions.token) {\n transportOptions.headers = {\n Authorization: 'Bearer ' + serviceOptions.token\n };\n }\n\n var http = new TransportFactory(transportOptions);\n\n var setIdFilterOrThrowError = function (options) {\n if (options.id) {\n serviceOptions.filter = options.id;\n }\n if (options.filter) {\n serviceOptions.filter = options.filter;\n }\n if (!serviceOptions.filter) {\n throw new Error('No world id specified to apply operations against. This could happen if the user is not assigned to a world and is trying to work with runs from that world.');\n }\n };\n\n var validateModelOrThrowError = function (options) {\n if (!options.model) {\n throw new Error('No model specified to get the current run');\n }\n };\n\n var publicAPI = {\n\n /**\n * Creates a new World.\n *\n * Using this method is rare. It is more common to create worlds automatically while you `autoAssign()` end users to worlds. (In this case, configuration data for the world, such as the roles, are read from the project-level world configuration information, for example by `getProjectSettings()`.)\n *\n * **Example**\n *\n * var wa = new F.service.World({\n * account: 'acme-simulations',\n * project: 'supply-chain-game',\n * group: 'team1' });\n * wa.create({\n * roles: ['VP Marketing', 'VP Sales', 'VP Engineering']\n * });\n *\n * **Parameters**\n * @param {object} params Parameters to create the world.\n * @param {string} params.group (Optional) The **Group Name** to create this world under. Only end users in this group are eligible to join the world. Optional here; required when instantiating the service (`new F.service.World()`).\n * @param {object} params.roles (Optional) The list of roles (strings) for this world. Some worlds have specific roles that **must** be filled by end users. Listing the roles allows you to autoassign users to worlds and ensure that all roles are filled in each world.\n * @param {object} params.optionalRoles (Optional) The list of optional roles (strings) for this world. Some worlds have specific roles that **may** be filled by end users. Listing the optional roles as part of the world object allows you to autoassign users to worlds and ensure that all roles are filled in each world.\n * @param {integer} params.minUsers (Optional) The minimum number of users for the world. Including this number allows you to autoassign end users to worlds and ensure that the correct number of users are in each world.\n * @param {object} options (Optional) Options object to override global options.\n * @return {Promise}\n */\n create: function (params, options) {\n var createOptions = $.extend(true, {}, serviceOptions, options, { url: urlConfig.getAPIPath(apiEndpoint) });\n var worldApiParams = ['scope', 'files', 'roles', 'optionalRoles', 'minUsers', 'group', 'name'];\n var validParams = _pick(serviceOptions, ['account', 'project', 'group']);\n // whitelist the fields that we actually can send to the api\n params = _pick(params, worldApiParams);\n\n // account and project go in the body, not in the url\n params = $.extend({}, validParams, params);\n\n var oldSuccess = createOptions.success;\n createOptions.success = function (response) {\n serviceOptions.filter = response.id; //all future chained calls to operate on this id\n return oldSuccess.apply(this, arguments);\n };\n\n return http.post(params, createOptions);\n },\n\n /**\n * Updates a World, for example to replace the roles in the world.\n *\n * Typically, you complete world configuration at the project level, rather than at the world level. For example, each world in your project probably has the same roles for end users. And your project is probably either configured so that all end users share the same world (and run), or smaller sets of end users share worlds — but not both. However, this method is available if you need to update the configuration of a particular world.\n *\n * **Example**\n *\n * var wa = new F.service.World({\n * account: 'acme-simulations',\n * project: 'supply-chain-game',\n * group: 'team1' });\n * wa.create()\n * .then(function(world) {\n * wa.update({ roles: ['VP Marketing', 'VP Sales', 'VP Engineering'] });\n * });\n *\n * **Parameters**\n * @param {object} params Parameters to update the world.\n * @param {string} params.name A string identifier for the linked end users, for example, \"name\": \"Our Team\".\n * @param {object} params.roles (Optional) The list of roles (strings) for this world. Some worlds have specific roles that **must** be filled by end users. Listing the roles allows you to autoassign users to worlds and ensure that all roles are filled in each world.\n * @param {object} params.optionalRoles (Optional) The list of optional roles (strings) for this world. Some worlds have specific roles that **may** be filled by end users. Listing the optional roles as part of the world object allows you to autoassign users to worlds and ensure that all roles are filled in each world.\n * @param {integer} params.minUsers (Optional) The minimum number of users for the world. Including this number allows you to autoassign end users to worlds and ensure that the correct number of users are in each world.\n * @param {object} options (Optional) Options object to override global options.\n * @return {Promise}\n */\n update: function (params, options) {\n var whitelist = ['roles', 'optionalRoles', 'minUsers'];\n options = options || {};\n setIdFilterOrThrowError(options);\n\n var updateOptions = $.extend(true, {},\n serviceOptions,\n options,\n { url: urlConfig.getAPIPath(apiEndpoint) + serviceOptions.filter }\n );\n\n params = _pick(params || {}, whitelist);\n\n return http.patch(params, updateOptions);\n },\n\n /**\n * Deletes an existing world.\n *\n * This function optionally takes one argument. If the argument is a string, it is the id of the world to delete. If the argument is an object, it is the override for global options.\n *\n * **Example**\n *\n * var wa = new F.service.World({\n * account: 'acme-simulations',\n * project: 'supply-chain-game',\n * group: 'team1' });\n * wa.create()\n * .then(function(world) {\n * wa.delete();\n * });\n *\n * **Parameters**\n * @param {String|Object} options (Optional) The id of the world to delete, or options object to override global options.\n * @return {Promise}\n */\n delete: function (options) {\n options = (options && (typeof options === 'string')) ? { filter: options } : {};\n setIdFilterOrThrowError(options);\n\n var deleteOptions = $.extend(true, {},\n serviceOptions,\n options,\n { url: urlConfig.getAPIPath(apiEndpoint) + serviceOptions.filter }\n );\n\n return http.delete(null, deleteOptions);\n },\n\n /**\n * Updates the configuration for the current instance of the World API Adapter (including all subsequent function calls, until the configuration is updated again).\n *\n * **Example**\n *\n * var wa = new F.service.World({...}).updateConfig({ filter: '123' }).addUser({ userId: '123' });\n *\n * **Parameters**\n * @param {object} config The configuration object to use in updating existing configuration.\n * @return {Object} reference to current instance\n */\n updateConfig: function (config) {\n $.extend(serviceOptions, config);\n\n return this;\n },\n\n /**\n * Lists all worlds for a given account, project, and group. All three are required, and if not specified as parameters, are read from the service.\n *\n * **Example**\n *\n * var wa = new F.service.World({\n * account: 'acme-simulations',\n * project: 'supply-chain-game',\n * group: 'team1' });\n * wa.create()\n * .then(function(world) {\n * // lists all worlds in group \"team1\"\n * wa.list();\n *\n * // lists all worlds in group \"other-group-name\"\n * wa.list({ group: 'other-group-name' });\n * });\n *\n * **Parameters**\n * @param {object} options (Optional) Options object to override global options.\n * @return {Promise}\n */\n list: function (options) {\n options = options || {};\n\n var getOptions = $.extend(true, {},\n serviceOptions,\n options,\n { url: urlConfig.getAPIPath(apiEndpoint) }\n );\n\n var filters = _pick(getOptions, ['account', 'project', 'group']);\n\n return http.get(filters, getOptions);\n },\n\n /**\n * Gets all worlds that an end user belongs to for a given account (team), project, and group.\n *\n * **Example**\n *\n * var wa = new F.service.World({\n * account: 'acme-simulations',\n * project: 'supply-chain-game',\n * group: 'team1' });\n * wa.create()\n * .then(function(world) {\n * wa.getWorldsForUser('b1c19dda-2d2e-4777-ad5d-3929f17e86d3')\n * });\n *\n * ** Parameters **\n * @param {string} userId The `userId` of the user whose worlds are being retrieved.\n * @param {object} options (Optional) Options object to override global options.\n * @return {Promise}\n */\n getWorldsForUser: function (userId, options) {\n options = options || {};\n\n var getOptions = $.extend(true, {},\n serviceOptions,\n options,\n { url: urlConfig.getAPIPath(apiEndpoint) }\n );\n\n var filters = $.extend(\n _pick(getOptions, ['account', 'project', 'group']),\n { userId: userId }\n );\n\n return http.get(filters, getOptions);\n },\n\n /**\n * Load information for a specific world. All further calls to the world service will use the id provided.\n *\n * **Parameters**\n * @param {String} worldId The id of the world to load.\n * @param {Object} options (Optional) Options object to override global options.\n * @return {Promise}\n */\n load: function (worldId, options) {\n if (worldId) {\n serviceOptions.filter = worldId;\n }\n if (!serviceOptions.filter) {\n throw new Error('Please provide a worldid to load');\n }\n var httpOptions = $.extend(true, {}, serviceOptions, options, { url: urlConfig.getAPIPath(apiEndpoint) + serviceOptions.filter + '/' });\n return http.get('', httpOptions);\n },\n\n /**\n * Adds an end user or list of end users to a given world. The end user must be a member of the `group` that is associated with this world.\n *\n * **Example**\n *\n * var wa = new F.service.World({\n * account: 'acme-simulations',\n * project: 'supply-chain-game',\n * group: 'team1' });\n * wa.create()\n * .then(function(world) {\n * // add one user\n * wa.addUsers('b1c19dda-2d2e-4777-ad5d-3929f17e86d3');\n * wa.addUsers(['b1c19dda-2d2e-4777-ad5d-3929f17e86d3']);\n * wa.addUsers({ userId: 'b1c19dda-2d2e-4777-ad5d-3929f17e86d3', role: 'VP Sales' });\n *\n * // add several users\n * wa.addUsers([\n * { userId: 'a6fe0c1e-f4b8-4f01-9f5f-01ccf4c2ed44',\n * role: 'VP Marketing' },\n * { userId: '8f2604cf-96cd-449f-82fa-e331530734ee',\n * role: 'VP Engineering' }\n * ]);\n *\n * // add one user to a specific world\n * wa.addUsers('b1c19dda-2d2e-4777-ad5d-3929f17e86d3', world.id);\n * wa.addUsers('b1c19dda-2d2e-4777-ad5d-3929f17e86d3', { filter: world.id });\n * });\n *\n * ** Parameters **\n * @param {string|object|array} users User id, array of user ids, object, or array of objects of the users to add to this world.\n * @param {string} users.role The `role` the user should have in the world. It is up to the caller to ensure, if needed, that the `role` passed in is one of the `roles` or `optionalRoles` of this world.\n * @param {string} worldId The world to which the users should be added. If not specified, the filter parameter of the `options` object is used.\n * @param {object} options (Optional) Options object to override global options.\n * @return {Promise}\n */\n addUsers: function (users, worldId, options) {\n\n if (!users) {\n throw new Error('Please provide a list of users to add to the world');\n }\n\n // normalize the list of users to an array of user objects\n users = $.map([].concat(users), function (u) {\n var isObject = $.isPlainObject(u);\n\n if (typeof u !== 'string' && !isObject) {\n throw new Error('Some of the users in the list are not in the valid format: ' + u);\n }\n\n return isObject ? u : { userId: u };\n });\n\n // check if options were passed as the second parameter\n if ($.isPlainObject(worldId) && !options) {\n options = worldId;\n worldId = null;\n }\n\n options = options || {};\n\n // we must have options by now\n if (typeof worldId === 'string') {\n options.filter = worldId;\n }\n\n setIdFilterOrThrowError(options);\n\n var updateOptions = $.extend(true, {},\n serviceOptions,\n options,\n { url: urlConfig.getAPIPath(apiEndpoint) + serviceOptions.filter + '/users' }\n );\n\n return http.post(users, updateOptions);\n },\n\n /**\n * Updates the role of an end user in a given world. (You can only update one end user at a time.)\n *\n * **Example**\n *\n * var wa = new F.service.World({\n * account: 'acme-simulations',\n * project: 'supply-chain-game',\n * group: 'team1' });\n *\n * wa.create().then(function(world) {\n * wa.addUsers('b1c19dda-2d2e-4777-ad5d-3929f17e86d3');\n * wa.updateUser({ userId: 'b1c19dda-2d2e-4777-ad5d-3929f17e86d3', role: 'leader' });\n * });\n *\n * **Parameters**\n * @param {object} user User object with `userId` and the new `role`.\n * @param {object} options (Optional) Options object to override global options.\n * @return {Promise}\n */\n updateUser: function (user, options) {\n options = options || {};\n\n if (!user || !user.userId) {\n throw new Error('You need to pass a userId to update from the world');\n }\n\n setIdFilterOrThrowError(options);\n\n var patchOptions = $.extend(true, {},\n serviceOptions,\n options,\n { url: urlConfig.getAPIPath(apiEndpoint) + serviceOptions.filter + '/users/' + user.userId }\n );\n\n return http.patch(_pick(user, 'role'), patchOptions);\n },\n\n /**\n * Removes an end user from a given world.\n *\n * **Example**\n *\n * var wa = new F.service.World({\n * account: 'acme-simulations',\n * project: 'supply-chain-game',\n * group: 'team1' });\n * wa.create()\n * .then(function(world) {\n * wa.addUsers(['a6fe0c1e-f4b8-4f01-9f5f-01ccf4c2ed44', '8f2604cf-96cd-449f-82fa-e331530734ee']);\n * wa.removeUser('a6fe0c1e-f4b8-4f01-9f5f-01ccf4c2ed44');\n * wa.removeUser({ userId: '8f2604cf-96cd-449f-82fa-e331530734ee' });\n * });\n *\n * ** Parameters **\n * @param {object|string} user The `userId` of the user to remove from the world, or an object containing the `userId` field.\n * @param {object} options (Optional) Options object to override global options.\n * @return {Promise}\n */\n removeUser: function (user, options) {\n options = options || {};\n\n if (typeof user === 'string') {\n user = { userId: user };\n }\n\n if (!user.userId) {\n throw new Error('You need to pass a userId to remove from the world');\n }\n\n setIdFilterOrThrowError(options);\n\n var getOptions = $.extend(true, {},\n serviceOptions,\n options,\n { url: urlConfig.getAPIPath(apiEndpoint) + serviceOptions.filter + '/users/' + user.userId }\n );\n\n return http.delete(null, getOptions);\n },\n\n /**\n * Gets the run id of current run for the given world. If the world does not have a run, creates a new one and returns the run id.\n *\n * Remember that a [run](../../glossary/#run) is a collection of interactions with a project and its model. In the case of multiplayer projects, the run is shared by all end users in the world.\n *\n * **Example**\n *\n * var wa = new F.service.World({\n * account: 'acme-simulations',\n * project: 'supply-chain-game',\n * group: 'team1' });\n * wa.create()\n * .then(function(world) {\n * wa.getCurrentRunId({ model: 'model.py' });\n * });\n *\n * ** Parameters **\n * @param {object} options (Optional) Options object to override global options.\n * @param {object} options.model The model file to use to create a run if needed.\n * @return {Promise}\n */\n getCurrentRunId: function (options) {\n options = options || {};\n\n setIdFilterOrThrowError(options);\n\n var getOptions = $.extend(true, {},\n serviceOptions,\n options,\n { url: urlConfig.getAPIPath(apiEndpoint) + serviceOptions.filter + '/run' }\n );\n\n validateModelOrThrowError(getOptions);\n return http.post(_pick(getOptions, 'model'), getOptions);\n },\n\n /**\n * Gets the current (most recent) world for the given end user in the given group. Brings this most recent world into memory if needed.\n *\n * **Example**\n *\n * var wa = new F.service.World({\n * account: 'acme-simulations',\n * project: 'supply-chain-game',\n * group: 'team1' });\n * wa.getCurrentWorldForUser('8f2604cf-96cd-449f-82fa-e331530734ee')\n * .then(function(world) {\n * // use data from world\n * });\n *\n * ** Parameters **\n * @param {string} userId The `userId` of the user whose current (most recent) world is being retrieved.\n * @param {string} groupName (Optional) The name of the group. If not provided, defaults to the group used to create the service.\n * @return {Promise}\n */\n getCurrentWorldForUser: function (userId, groupName) {\n var dtd = $.Deferred();\n var me = this;\n this.getWorldsForUser(userId, { group: groupName })\n .then(function (worlds) {\n // assume the most recent world as the 'active' world\n worlds.sort(function (a, b) { return new Date(b.lastModified) - new Date(a.lastModified); });\n var currentWorld = worlds[0];\n\n if (currentWorld) {\n serviceOptions.filter = currentWorld.id;\n }\n\n dtd.resolveWith(me, [currentWorld]);\n })\n .fail(dtd.reject);\n\n return dtd.promise();\n },\n\n /**\n * Deletes the current run from the world.\n *\n * (Note that the world id remains part of the run record, indicating that the run was formerly an active run for the world.)\n *\n * **Example**\n *\n * var wa = new F.service.World({\n * account: 'acme-simulations',\n * project: 'supply-chain-game',\n * group: 'team1' });\n *\n * wa.deleteRun('sample-world-id');\n *\n * **Parameters**\n * @param {string} worldId The `worldId` of the world from which the current run is being deleted.\n * @param {object} options (Optional) Options object to override global options.\n * @return {Promise}\n */\n deleteRun: function (worldId, options) {\n options = options || {};\n\n if (worldId) {\n options.filter = worldId;\n }\n\n setIdFilterOrThrowError(options);\n\n var deleteOptions = $.extend(true, {},\n serviceOptions,\n options,\n { url: urlConfig.getAPIPath(apiEndpoint) + serviceOptions.filter + '/run' }\n );\n\n return http.delete(null, deleteOptions);\n },\n\n /**\n * Creates a new run for the world.\n *\n * **Example**\n *\n * var wa = new F.service.World({\n * account: 'acme-simulations',\n * project: 'supply-chain-game',\n * group: 'team1' });\n *\n * wa.getCurrentWorldForUser('8f2604cf-96cd-449f-82fa-e331530734ee')\n * .then(function (world) {\n * wa.newRunForWorld(world.id);\n * });\n *\n * **Parameters**\n * @param {string} worldId worldId in which we create the new run.\n * @param {object} options (Optional) Options object to override global options.\n * @param {object} options.model The model file to use to create a run if needed.\n * @return {Promise}\n */\n newRunForWorld: function (worldId, options) {\n var currentRunOptions = $.extend(true, {},\n serviceOptions,\n options,\n { filter: worldId || serviceOptions.filter }\n );\n var me = this;\n\n validateModelOrThrowError(currentRunOptions);\n\n return this.deleteRun(worldId, options)\n .then(function () {\n return me.getCurrentRunId(currentRunOptions);\n });\n },\n\n /**\n * Assigns end users to worlds, creating new worlds as appropriate, automatically. Assigns all end users in the group, and creates new worlds as needed based on the project-level world configuration (roles, optional roles, and minimum end users per world).\n *\n * **Example**\n *\n * var wa = new F.service.World({\n * account: 'acme-simulations',\n * project: 'supply-chain-game',\n * group: 'team1' });\n *\n * wa.autoAssign();\n *\n * **Parameters**\n * @param {object} options (Optional) Options object to override global options.\n * @param {number} options.maxUsers Sets the maximum number of users in a world.\n * @param {string[]} options.userIds A list of users to be assigned be assigned instead of all end users in the group.\n * @return {Promise}\n *\n */\n autoAssign: function (options) {\n options = options || {};\n\n var opt = $.extend(true, {},\n serviceOptions,\n options,\n { url: urlConfig.getAPIPath(assignmentEndpoint) }\n );\n\n var params = {\n account: opt.account,\n project: opt.project,\n group: opt.group\n };\n\n if (opt.maxUsers) {\n params.maxUsers = opt.maxUsers;\n }\n\n if (opt.userIds) {\n params.userIds = opt.userIds;\n }\n\n return http.post(params, opt);\n },\n\n /**\n * Gets the project's world configuration.\n *\n * Typically, every interaction with your project uses the same configuration of each world. For example, each world in your project probably has the same roles for end users. And your project is probably either configured so that all end users share the same world (and run), or smaller sets of end users share worlds — but not both.\n *\n * (The [Multiplayer Project REST API](../../../rest_apis/multiplayer/multiplayer_project/) allows you to set these project-level world configurations. The World Adapter simply retrieves them, for example so they can be used in auto-assignment of end users to worlds.)\n *\n * **Example**\n *\n * var wa = new F.service.World({\n * account: 'acme-simulations',\n * project: 'supply-chain-game',\n * group: 'team1' });\n *\n * wa.getProjectSettings()\n * .then(function(settings) {\n * console.log(settings.roles);\n * console.log(settings.optionalRoles);\n * });\n *\n * **Parameters**\n * @param {object} options (Optional) Options object to override global options.\n * @return {Promise}\n */\n getProjectSettings: function (options) {\n options = options || {};\n\n var opt = $.extend(true, {},\n serviceOptions,\n options,\n { url: urlConfig.getAPIPath(projectEndpoint) }\n );\n\n opt.url += [opt.account, opt.project].join('/');\n return http.get(null, opt);\n }\n\n };\n\n $.extend(this, publicAPI);\n};\n","/**\n * @class Cookie Storage Service\n *\n * @example\n * var people = require('cookie-store')({ root: 'people' });\n people\n .save({lastName: 'smith' })\n\n */\n\n\n'use strict';\n\n// Thin document.cookie wrapper to allow unit testing\nvar Cookie = function () {\n this.get = function () {\n return document.cookie;\n };\n\n this.set = function (newCookie) {\n document.cookie = newCookie;\n };\n};\n\nmodule.exports = function (config) {\n var host = window.location.hostname;\n var validHost = host.split('.').length > 1;\n var domain = validHost ? '.' + host : null;\n\n var defaults = {\n /**\n * Name of collection\n * @type { string}\n */\n root: '/',\n\n domain: domain,\n cookie: new Cookie()\n };\n this.serviceOptions = $.extend({}, defaults, config);\n\n var publicAPI = {\n // * TBD\n // * Query collection; uses MongoDB syntax\n // * @see \n // *\n // * @param { string} qs Query Filter\n // * @param { string} limiters @see \n // *\n // * @example\n // * cs.query(\n // * { name: 'John', className: 'CSC101' },\n // * {limit: 10}\n // * )\n\n // query: function (qs, limiters) {\n\n // },\n\n /**\n * Save cookie value\n * @param { string|Object} key If given a key save values under it, if given an object directly, save to top-level api\n * @param {Object} value (Optional)\n * @param {Object} options Overrides for service options\n *\n * @return {*} The saved value\n *\n * @example\n * cs.set('person', { firstName: 'john', lastName: 'smith' });\n * cs.set({ name:'smith', age:'32' });\n */\n set: function (key, value, options) {\n var setOptions = $.extend(true, {}, this.serviceOptions, options);\n\n var domain = setOptions.domain;\n var path = setOptions.root;\n var cookie = setOptions.cookie;\n\n cookie.set(encodeURIComponent(key) + '=' +\n encodeURIComponent(value) +\n (domain ? '; domain=' + domain : '') +\n (path ? '; path=' + path : '')\n );\n\n return value;\n },\n\n /**\n * Load cookie value\n * @param { string|Object} key If given a key save values under it, if given an object directly, save to top-level api\n * @return {*} The value stored\n *\n * @example\n * cs.get('person');\n */\n get: function (key) {\n var cookie = this.serviceOptions.cookie;\n var cookieReg = new RegExp('(?:^|;)\\\\s*' + encodeURIComponent(key).replace(/[\\-\\.\\+\\*]/g, '\\\\$&') + '\\\\s*\\\\=\\\\s*([^;]*).*$');\n var res = cookieReg.exec(cookie.get());\n var val = res ? decodeURIComponent(res[1]) : null;\n return val;\n },\n\n /**\n * Removes key from collection\n * @param { string} key key to remove\n * @param {object} options (optional) overrides for service options\n * @return { string} key The key removed\n *\n * @example\n * cs.remove('person');\n */\n remove: function (key, options) {\n var remOptions = $.extend(true, {}, this.serviceOptions, options);\n\n var domain = remOptions.domain;\n var path = remOptions.root;\n var cookie = remOptions.cookie;\n\n cookie.set(encodeURIComponent(key) +\n '=; expires=Thu, 01 Jan 1970 00:00:00 GMT' +\n (domain ? '; domain=' + domain : '') +\n (path ? '; path=' + path : '')\n );\n return key;\n },\n\n /**\n * Removes collection being referenced\n * @return { array} keys All the keys removed\n */\n destroy: function () {\n var cookie = this.serviceOptions.cookie;\n var aKeys = cookie.get().replace(/((?:^|\\s*;)[^\\=]+)(?=;|$)|^\\s*|\\s*(?:\\=[^;]*)?(?:\\1|$)/g, '').split(/\\s*(?:\\=[^;]*)?;\\s*/);\n for (var nIdx = 0; nIdx < aKeys.length; nIdx++) {\n var cookieKey = decodeURIComponent(aKeys[nIdx]);\n this.remove(cookieKey);\n }\n return aKeys;\n }\n };\n\n $.extend(this, publicAPI);\n};\n","'use strict';\n\nvar keyNames = require('../managers/key-names');\nvar StorageFactory = require('./store-factory');\nvar optionUtils = require('../util/option-utils');\n\nvar EPI_SESSION_KEY = keyNames.EPI_SESSION_KEY;\nvar EPI_MANAGER_KEY = 'epicenter.token'; //can't be under key-names, or logout will clear this too\n\nvar defaults = {\n /**\n * Where to store user access tokens for temporary access. Defaults to storing in a cookie in the browser.\n * @type {string}\n */\n store: { synchronous: true }\n};\n\nvar SessionManager = function (managerOptions) {\n managerOptions = managerOptions || {};\n function getBaseOptions(overrides) {\n overrides = overrides || {};\n var libOptions = optionUtils.getOptions();\n var finalOptions = $.extend(true, {}, defaults, libOptions, managerOptions, overrides);\n return finalOptions;\n }\n\n function getStore(overrides) {\n var baseOptions = getBaseOptions(overrides);\n var storeOpts = baseOptions.store || {};\n var isEpicenterDomain = !baseOptions.isLocal && !baseOptions.isCustomDomain;\n if (storeOpts.root === undefined && baseOptions.account && baseOptions.project && isEpicenterDomain) {\n storeOpts.root = '/app/' + baseOptions.account + '/' + baseOptions.project;\n }\n return new StorageFactory(storeOpts);\n }\n\n var publicAPI = {\n saveSession: function (userInfo, options) {\n var serialized = JSON.stringify(userInfo);\n getStore(options).set(EPI_SESSION_KEY, serialized);\n },\n getSession: function (options) {\n var store = getStore(options);\n var finalOpts = store.serviceOptions;\n var serialized = store.get(EPI_SESSION_KEY) || '{}';\n var session = JSON.parse(serialized);\n // If the url contains the project and account\n // validate the account and project in the session\n // and override project, groupName, groupId and isFac\n // Otherwise (i.e. localhost) use the saved session values\n var account = finalOpts.account;\n var project = finalOpts.project;\n if (account && session.account !== account) {\n // This means that the token was not used to login to the same account\n return {};\n }\n if (session.groups && account && project) {\n var group = session.groups[project] || { groupId: '', groupName: '', isFac: false };\n $.extend(session, { project: project }, group);\n }\n return session;\n },\n removeSession: function (options) {\n var store = getStore(options);\n Object.keys(keyNames).forEach(function (cookieKey) {\n var cookieName = keyNames[cookieKey];\n store.remove(cookieName);\n });\n return true;\n },\n getStore: function (options) {\n return getStore(options);\n },\n\n getMergedOptions: function () {\n var args = Array.prototype.slice.call(arguments);\n var overrides = $.extend.apply($, [true, {}].concat(args));\n var baseOptions = getBaseOptions(overrides);\n var session = this.getSession(overrides);\n\n var token = session.auth_token;\n if (!token) {\n var factory = new StorageFactory();\n token = factory.get(EPI_MANAGER_KEY);\n }\n\n var sessionDefaults = {\n /**\n * For projects that require authentication, pass in the user access token (defaults to empty string). If the user is already logged in to Epicenter, the user access token is already set in a cookie and automatically loaded from there. (See [more background on access tokens](../../../project_access/)).\n * @see [Authentication API Service](../auth-api-service/) for getting tokens.\n * @type {String}\n */\n token: token,\n\n /**\n * The account. If left undefined, taken from the cookie session.\n * @type {String}\n */\n account: session.account,\n\n /**\n * The project. If left undefined, taken from the cookie session.\n * @type {String}\n */\n project: session.project,\n\n\n /**\n * The group name. If left undefined, taken from the cookie session.\n * @type {String}\n */\n group: session.groupName,\n /**\n * Alias for group. \n * @type {String}\n */\n groupName: session.groupName, //It's a little weird that it's called groupName in the cookie, but 'group' in all the service options, so normalize for both\n /**\n * The group id. If left undefined, taken from the cookie session.\n * @type {String}\n */\n groupId: session.groupId,\n userId: session.userId,\n userName: session.userName,\n };\n return $.extend(true, sessionDefaults, baseOptions);\n }\n };\n $.extend(this, publicAPI);\n};\n\nmodule.exports = SessionManager;","/**\n Decides type of store to provide\n*/\n\n'use strict';\n// var isNode = false; FIXME: Browserify/minifyify has issues with the next link\n// var store = (isNode) ? require('./session-store') : require('./cookie-store');\nvar store = require('./cookie-store');\n\nmodule.exports = store;\n","'use strict';\n\nvar qutils = require('../util/query-util');\n\nmodule.exports = function (config) {\n\n var defaults = {\n url: '',\n\n contentType: 'application/json',\n headers: {},\n statusCode: {\n 404: $.noop\n },\n\n /**\n * ONLY for strings in the url. All GET & DELETE params are run through this\n * @type {[type] }\n */\n parameterParser: qutils.toQueryFormat,\n\n // To allow epicenter.token and other session cookies to be passed\n // with the requests\n xhrFields: {\n withCredentials: true\n }\n };\n\n var transportOptions = $.extend({}, defaults, config);\n\n var result = function (d) {\n return ($.isFunction(d)) ? d() : d;\n };\n\n var connect = function (method, params, connectOptions) {\n params = result(params);\n params = ($.isPlainObject(params) || $.isArray(params)) ? JSON.stringify(params) : params;\n\n var options = $.extend(true, {}, transportOptions, connectOptions, {\n type: method,\n data: params\n });\n var ALLOWED_TO_BE_FUNCTIONS = ['data', 'url'];\n $.each(options, function (key, value) {\n if ($.isFunction(value) && $.inArray(key, ALLOWED_TO_BE_FUNCTIONS) !== -1) {\n options[key] = value();\n }\n });\n\n if (options.logLevel && options.logLevel === 'DEBUG') {\n console.log(options.url);\n var oldSuccessFn = options.success || $.noop;\n options.success = function (response, ajaxStatus, ajaxReq) {\n console.log(response);\n oldSuccessFn.apply(this, arguments);\n };\n }\n\n var beforeSend = options.beforeSend;\n options.beforeSend = function (xhr, settings) {\n xhr.requestUrl = (connectOptions || {}).url;\n if (beforeSend) {\n beforeSend.apply(this, arguments);\n }\n };\n\n return $.ajax(options);\n };\n\n var publicAPI = {\n get: function (params, ajaxOptions) {\n var options = $.extend({}, transportOptions, ajaxOptions);\n params = options.parameterParser(result(params));\n return connect.call(this, 'GET', params, options);\n },\n splitGet: function () {\n\n },\n post: function () {\n return connect.apply(this, ['post'].concat([].slice.call(arguments)));\n },\n patch: function () {\n return connect.apply(this, ['patch'].concat([].slice.call(arguments)));\n },\n put: function () {\n return connect.apply(this, ['put'].concat([].slice.call(arguments)));\n },\n delete: function (params, ajaxOptions) {\n //DELETE doesn't support body params, but jQuery thinks it does.\n var options = $.extend({}, transportOptions, ajaxOptions);\n params = options.parameterParser(result(params));\n if ($.trim(params)) {\n var delimiter = (result(options.url).indexOf('?') === -1) ? '?' : '&';\n options.url = result(options.url) + delimiter + params;\n }\n return connect.call(this, 'DELETE', null, options);\n },\n head: function () {\n return connect.apply(this, ['head'].concat([].slice.call(arguments)));\n },\n options: function () {\n return connect.apply(this, ['options'].concat([].slice.call(arguments)));\n }\n };\n\n return $.extend(this, publicAPI);\n};\n","'use strict';\n\n// var isNode = false; FIXME: Browserify/minifyify has issues with the next link\n// var transport = (isNode) ? require('./node-http-transport') : require('./ajax-http-transport');\nvar transport = require('./ajax-http-transport');\nmodule.exports = transport;\n","/**\n/* Inherit from a class (using prototype borrowing)\n*/\n'use strict';\n\nfunction inherit(C, P) {\n var F = function () {};\n F.prototype = P.prototype;\n C.prototype = new F();\n C.__super = P.prototype;\n C.prototype.constructor = C;\n}\n\n/**\n* Shallow copy of an object\n* @param {Object} dest object to extend\n* @return {Object} extended object\n*/\nvar extend = function (dest /*, var_args*/) {\n var obj = Array.prototype.slice.call(arguments, 1);\n var current;\n for (var j = 0; j < obj.length; j++) {\n if (!(current = obj[j])) { //eslint-disable-line\n continue;\n }\n\n // do not wrap inner in dest.hasOwnProperty or bad things will happen\n for (var key in current) { //eslint-disable-line\n dest[key] = current[key];\n }\n }\n\n return dest;\n};\n\nmodule.exports = function (base, props, staticProps) {\n var parent = base;\n var child;\n\n child = props && props.hasOwnProperty('constructor') ? props.constructor : function () { return parent.apply(this, arguments); };\n\n // add static properties to the child constructor function\n extend(child, parent, staticProps);\n\n // associate prototype chain\n inherit(child, parent);\n\n // add instance properties\n if (props) {\n extend(child.prototype, props);\n }\n\n // done\n return child;\n};\n","'use strict';\n\nmodule.exports = {\n _pick: function (obj, props) {\n var res = {};\n for (var p in obj) {\n if (props.indexOf(p) !== -1) {\n res[p] = obj[p];\n }\n }\n\n return res;\n },\n isEmpty: function isEmpty(value) {\n return (!value || ($.isPlainObject(value) && Object.keys(value).length === 0));\n }\n};\n","'use strict';\n\nvar ConfigService = require('../service/configuration-service');\n\nvar urlConfig = new ConfigService().get('server');\nvar customDefaults = {};\nvar libDefaults = {\n /**\n * The account id. In the Epicenter UI, this is the **Team ID** (for team projects) or **User ID** (for personal projects). Defaults to empty string. If left undefined, taken from the URL.\n * @type {String}\n */\n account: urlConfig.accountPath || undefined,\n /**\n * The account id. In the Epicenter UI, this is the **Team ID** (for team projects) or **User ID** (for personal projects). Defaults to empty string. If left undefined, taken from the URL.\n * @type {String}\n */\n project: urlConfig.projectPath || undefined,\n isLocal: urlConfig.isLocalhost(),\n isCustomDomain: urlConfig.isCustomDomain,\n store: {}\n};\n\nvar optionUtils = {\n /**\n * Gets the final options by overriding the global options set with\n * optionUtils#setDefaults() and the lib defaults.\n * @param {object} options The final options object.\n * @return {object} Extended object\n */\n getOptions: function (options) {\n return $.extend(true, {}, libDefaults, customDefaults, options);\n },\n /**\n * Sets the global defaults for the optionUtils#getOptions() method.\n * @param {object} defaults The defaults object.\n */\n setDefaults: function (defaults) {\n customDefaults = defaults;\n }\n};\nmodule.exports = optionUtils;\n","/**\n * Utilities for working with query strings\n*/\n'use strict';\n\nmodule.exports = (function () {\n\n return {\n /**\n * Converts to matrix format\n * @param {Object} qs Object to convert to query string\n * @return { string} Matrix-format query parameters\n */\n toMatrixFormat: function (qs) {\n if (qs === null || qs === undefined || qs === '') {\n return ';';\n }\n if (typeof qs === 'string' || qs instanceof String) {\n return qs;\n }\n\n var returnArray = [];\n var OPERATORS = ['<', '>', '!'];\n $.each(qs, function (key, value) {\n if (typeof value !== 'string' || $.inArray($.trim(value).charAt(0), OPERATORS) === -1) {\n value = '=' + value;\n }\n returnArray.push(key + value);\n });\n\n var mtrx = ';' + returnArray.join(';');\n return mtrx;\n },\n\n /**\n * Converts strings/arrays/objects to type 'a=b&b=c'\n * @param { string|Array|Object} qs\n * @return { string}\n */\n toQueryFormat: function (qs) {\n if (qs === null || qs === undefined) {\n return '';\n }\n if (typeof qs === 'string' || qs instanceof String) {\n return qs;\n }\n\n var returnArray = [];\n $.each(qs, function (key, value) {\n if ($.isArray(value)) {\n value = value.join(',');\n }\n if ($.isPlainObject(value)) {\n //Mostly for data api\n value = JSON.stringify(value);\n }\n returnArray.push(key + '=' + value);\n });\n\n var result = returnArray.join('&');\n return result;\n },\n\n /**\n * Converts strings of type 'a=b&b=c' to { a:b, b:c}\n * @param { string} qs\n * @return {object}\n */\n qsToObject: function (qs) {\n if (qs === null || qs === undefined || qs === '') {\n return {};\n }\n\n var qsArray = qs.split('&');\n var returnObj = {};\n $.each(qsArray, function (index, value) {\n var qKey = value.split('=')[0];\n var qVal = value.split('=')[1];\n\n if (qVal.indexOf(',') !== -1) {\n qVal = qVal.split(',');\n }\n\n returnObj[qKey] = qVal;\n });\n\n return returnObj;\n },\n\n /**\n * Normalizes and merges strings of type 'a=b', { b:c} to { a:b, b:c}\n * @param { string|Array|Object} qs1\n * @param { string|Array|Object} qs2\n * @return {Object}\n */\n mergeQS: function (qs1, qs2) {\n var obj1 = this.qsToObject(this.toQueryFormat(qs1));\n var obj2 = this.qsToObject(this.toQueryFormat(qs2));\n return $.extend(true, {}, obj1, obj2);\n },\n\n addTrailingSlash: function (url) {\n if (!url) {\n return '';\n }\n return (url.charAt(url.length - 1) === '/') ? url : (url + '/');\n }\n };\n}());\n\n\n","/**\n * Utilities for working with the run service\n*/\n'use strict';\nvar qutil = require('./query-util');\nvar MAX_URL_LENGTH = 2048;\n\nmodule.exports = (function () {\n return {\n /**\n * normalizes different types of operation inputs\n * @param {Object|Array|String} operations operations to perform\n * @param {Array} args arguments for operation\n * @return {String} operations of the form `{ ops: [], args: [] }`\n */\n normalizeOperations: function (operations, args) {\n if (!args) {\n args = [];\n }\n var returnList = {\n ops: [],\n args: []\n };\n\n var _concat = function (arr) {\n return (arr !== null && arr !== undefined) ? [].concat(arr) : [];\n };\n\n //{ add: [1,2], subtract: [2,4] }\n var _normalizePlainObjects = function (operations, returnList) {\n if (!returnList) {\n returnList = { ops: [], args: [] };\n }\n $.each(operations, function (opn, arg) {\n returnList.ops.push(opn);\n returnList.args.push(_concat(arg));\n });\n return returnList;\n };\n //{ name: 'add', params: [1] }\n var _normalizeStructuredObjects = function (operation, returnList) {\n if (!returnList) {\n returnList = { ops: [], args: [] };\n }\n returnList.ops.push(operation.name);\n returnList.args.push(_concat(operation.params));\n return returnList;\n };\n\n var _normalizeObject = function (operation, returnList) {\n return ((operation.name) ? _normalizeStructuredObjects : _normalizePlainObjects)(operation, returnList);\n };\n\n var _normalizeLiterals = function (operation, args, returnList) {\n if (!returnList) {\n returnList = { ops: [], args: [] };\n }\n returnList.ops.push(operation);\n returnList.args.push(_concat(args));\n return returnList;\n };\n\n\n var _normalizeArrays = function (operations, arg, returnList) {\n if (!returnList) {\n returnList = { ops: [], args: [] };\n }\n $.each(operations, function (index, opn) {\n if ($.isPlainObject(opn)) {\n _normalizeObject(opn, returnList);\n } else {\n _normalizeLiterals(opn, args[index], returnList);\n }\n });\n return returnList;\n };\n\n if ($.isPlainObject(operations)) {\n _normalizeObject(operations, returnList);\n } else if ($.isArray(operations)) {\n _normalizeArrays(operations, args, returnList);\n } else {\n _normalizeLiterals(operations, args, returnList);\n }\n\n return returnList;\n },\n\n splitGetFactory: function (httpOptions) {\n return function (params, options) {\n var http = this; //eslint-disable-line\n var getValue = function (name) {\n var value = options[name] || httpOptions[name];\n if (typeof value === 'function') {\n value = value();\n }\n return value;\n };\n var getFinalUrl = function (params) {\n var url = getValue('url', options);\n var data = params;\n // There is easy (or known) way to get the final URL jquery is going to send so\n // we're replicating it. The process might change at some point but it probably will not.\n // 1. Remove hash\n url = url.replace(/#.*$/, '');\n // 1. Append query string\n var queryParams = qutil.toQueryFormat(data);\n var questionIdx = url.indexOf('?');\n if (queryParams && questionIdx > -1) {\n return url + '&' + queryParams;\n } else if (queryParams) {\n return url + '?' + queryParams;\n }\n return url;\n };\n var url = getFinalUrl(params);\n // We must split the GET in multiple short URL's\n // The only property allowed to be split is \"include\"\n if (params && params.include && encodeURI(url).length > MAX_URL_LENGTH) {\n var dtd = $.Deferred();\n var paramsCopy = $.extend(true, {}, params);\n delete paramsCopy.include;\n var urlNoIncludes = getFinalUrl(paramsCopy);\n var diff = MAX_URL_LENGTH - urlNoIncludes.length;\n var oldSuccess = options.success || httpOptions.success || $.noop;\n var oldError = options.error || httpOptions.error || $.noop;\n // remove the original success and error callbacks\n options.success = $.noop;\n options.error = $.noop;\n\n var include = params.include;\n var currIncludes = [];\n var includeOpts = [currIncludes];\n var currLength = encodeURIComponent('?include=').length;\n var variable = include.pop();\n while (variable) {\n var varLenght = encodeURIComponent(variable).length;\n // Use a greedy approach for now, can be optimized to be solved in a more\n // efficient way\n // + 1 is the comma\n if (currLength + varLenght + 1 < diff) {\n currIncludes.push(variable);\n currLength += varLenght + 1;\n } else {\n currIncludes = [variable];\n includeOpts.push(currIncludes);\n currLength = '?include='.length + varLenght;\n }\n variable = include.pop();\n }\n var reqs = $.map(includeOpts, function (include) {\n var reqParams = $.extend({}, params, { include: include });\n return http.get(reqParams, options);\n });\n $.when.apply($, reqs).then(function () {\n // Each argument are arrays of the arguments of each done request\n // So the first argument of the first array of arguments is the data\n var isValid = arguments[0] && arguments[0][0];\n if (!isValid) {\n // Should never happen...\n oldError();\n return dtd.reject();\n }\n var firstResponse = arguments[0][0];\n var isObject = $.isPlainObject(firstResponse);\n var isRunAPI = (isObject && $.isPlainObject(firstResponse.variables)) || !isObject;\n if (isRunAPI) {\n if (isObject) {\n // aggregate the variables property only\n var aggregateRun = arguments[0][0];\n $.each(arguments, function (idx, args) {\n var run = args[0];\n $.extend(true, aggregateRun.variables, run.variables);\n });\n oldSuccess(aggregateRun, arguments[0][1], arguments[0][2]);\n dtd.resolve(aggregateRun, arguments[0][1], arguments[0][2]);\n } else {\n // array of runs\n // Agregate variables in each run\n var aggregatedRuns = {};\n $.each(arguments, function (idx, args) {\n var runs = args[0];\n if (!$.isArray(runs)) {\n return;\n }\n $.each(runs, function (idxRun, run) {\n if (run.id && !aggregatedRuns[run.id]) {\n run.variables = run.variables || {};\n aggregatedRuns[run.id] = run;\n } else if (run.id) {\n $.extend(true, aggregatedRuns[run.id].variables, run.variables);\n }\n });\n });\n // turn it into an array\n aggregatedRuns = $.map(aggregatedRuns, function (run) { return run; });\n oldSuccess(aggregatedRuns, arguments[0][1], arguments[0][2]);\n dtd.resolve(aggregatedRuns, arguments[0][1], arguments[0][2]);\n }\n } else {\n // is variables API\n // aggregate the response\n var aggregatedVariables = {};\n $.each(arguments, function (idx, args) {\n var vars = args[0];\n $.extend(true, aggregatedVariables, vars);\n });\n oldSuccess(aggregatedVariables, arguments[0][1], arguments[0][2]);\n dtd.resolve(aggregatedVariables, arguments[0][1], arguments[0][2]);\n }\n }, function () {\n oldError.apply(http, arguments);\n dtd.reject.apply(dtd, arguments);\n });\n return dtd.promise();\n } else {\n return http.get(params, options);\n }\n };\n }\n };\n}());\n"]} \ No newline at end of file +{"version":3,"sources":["node_modules/browser-pack/_prelude.js","src/api-version.json","node_modules/Base64/base64.js","node_modules/object-assign/index.js","src/app.js","src/env-load.js","src/managers/auth-manager.js","src/managers/channel-manager.js","src/managers/epicenter-channel-manager.js","src/managers/key-names.js","src/managers/run-manager.js","src/managers/run-strategies/conditional-creation-strategy.js","src/managers/run-strategies/deprecated/new-if-initialized-strategy.js","src/managers/run-strategies/deprecated/new-if-persisted-strategy.js","src/managers/run-strategies/index.js","src/managers/run-strategies/multiplayer-strategy.js","src/managers/run-strategies/none-strategy.js","src/managers/run-strategies/reuse-across-sessions.js","src/managers/run-strategies/reuse-last-initialized.js","src/managers/run-strategies/reuse-never.js","src/managers/run-strategies/reuse-per-session.js","src/managers/saved-runs-manager.js","src/managers/scenario-manager.js","src/managers/scenario-strategies/baseline-strategy.js","src/managers/scenario-strategies/reuse-last-unsaved.js","src/managers/special-operations.js","src/managers/strategy-utils.js","src/managers/world-manager.js","src/service/admin-file-service.js","src/service/asset-api-adapter.js","src/service/auth-api-service.js","src/service/channel-service.js","src/service/configuration-service.js","src/service/data-api-service.js","src/service/group-api-service.js","src/service/introspection-api-service.js","src/service/member-api-adapter.js","src/service/presence-api-service.js","src/service/run-api-service.js","src/service/service-utils.js","src/service/state-api-adapter.js","src/service/url-config-service.js","src/service/user-api-adapter.js","src/service/variables-api-service.js","src/service/world-api-adapter.js","src/store/cookie-store.js","src/store/session-manager.js","src/store/store-factory.js","src/transport/ajax-http-transport.js","src/transport/http-transport-factory.js","src/util/inherit.js","src/util/object-util.js","src/util/option-utils.js","src/util/query-util.js","src/util/run-util.js"],"names":["InvalidCharacterError","message","this","object","exports","self","chars","prototype","Error","name","btoa","input","str","String","block","charCode","idx","map","output","charAt","charCodeAt","atob","replace","length","bs","buffer","bc","fromCharCode","indexOf","toObject","val","undefined","TypeError","Object","shouldUseNative","assign","test1","getOwnPropertyNames","test2","i","order2","n","join","test3","split","forEach","letter","keys","e","hasOwnProperty","propIsEnumerable","propertyIsEnumerable","module","target","source","from","to","symbols","s","arguments","key","call","getOwnPropertySymbols","F","_private","util","factory","transport","store","service","manager","strategy","load","require","global","SKIP_ENV_LOAD","query","run","classFrom","strategyutils","Transport","Ajax","URL","Config","Run","File","Variables","Data","Auth","World","State","User","Member","Asset","Group","Introspect","Presence","Cookie","Store","ScenarioManager","RunManager","AuthManager","WorldManager","SavedRunsManager","strategies","list","ChannelManager","Channel","version","api","constants","URLConfigService","envLoad","callback","urlService","infoUrl","getAPIPath","envPromise","$","ajax","url","async","then","res","overrides","defaults","extend","fail","options","sessionManager","SessionManager","getMergedOptions","authAdapter","AuthAdapter","MemberAdapter","GroupService","_pick","objectAssign","window","requiresGroup","_findUserInGroup","members","id","j","userId","login","me","$d","Deferred","adapterOptions","success","noop","error","outSuccess","outError","groupId","decodeToken","token","encoded","JSON","parse","handleGroupError","statusCode","data","type","logout","statusText","status","reject","handleSuccess","response","access_token","userInfo","oldGroups","getSession","groups","userGroupOpts","auth","user","project","isTeamMember","parent_account_id","userName","user_name","sessionInfo","auth_token","account","user_id","saveSession","apply","resolve","handleGroupList","groupList","userGroups","group","filteredGroups","grep","resGroup","isFac","role","groupData","groupName","sessionInfoWithGroup","opts","groupService","getGroups","getUserGroups","promise","removeCookieFn","removeSession","getToken","httpOptions","session","params","memberInfo","memberAdapter","server","getGroupsForUser","isLoggedIn","getCurrentUserSessionInfo","addGroups","isArray","Array","each","index","extendedGroup","validProps","cometd","logLevel","websocketEnabled","ackEnabled","shareConnection","channel","handshake","defaultCometOptions","currentSubscriptions","_cometd","CometD","isConnected","connectionBroken","trigger","connectionSucceeded","configure","addListener","wasConnected","successful","bind","batch","subs","resubscribe","getChannel","isPlainObject","base","subscribe","subid","concat","unsubs","unsubscribe","removed","splice","on","event","off","ConfigService","validTypes","world","general","chat","getFromSessionOrError","value","sessionKeyName","settings","isPresenceData","payload","__super","EpicenterChannelManager","constructor","urlConfig","get","userProp","ext","authorization","channelOpts","allowAllChannels","baseParts","channelType","getGroupChannel","baseTopic","oldsubs","topic","context","callbackWithoutPresenceData","getWorldChannel","worldid","getUserChannel","userid","getPresenceChannel","callbackWithOnlyPresenceData","getDataChannel","collection","callbackWithCleanData","meta","path","subType","date","dataPath","actualData","EPI_SESSION_KEY","STRATEGY_SESSION_KEY","patchRunService","patched","orig","do","operation","reservedOps","specialOperations","sessionKeyFromOptions","runService","config","getCurrentConfig","sessionKey","isFunction","setRunInSession","variables","getStore","set","stringify","RunService","isEmpty","getBestStrategy","keyNames","baseKey","reduce","accum","getRun","sessionStore","sessionContents","runSession","runId","authSession","requiresAuth","console","updateConfig","filter","results","catch","err","reset","Base","Strategy","condition","userSession","opt","scope","create","freshlyCreated","loadAndCheck","shouldCreate","msg","headers","ConditionalStrategy","createIf","warn","getResponseHeader","initialized","conditional-creation","new-if-initialized","new-if-persisted","none","multiplayer","reuse-never","reuse-per-session","reuse-across-sessions","reuse-last-initialized","byName","strategyName","strategyOptions","initOperation","StrategyCtor","strategyInstance","register","IdentityStrategy","WorldApiAdapter","worldApi","curUserId","curGroupName","getCurrentWorldForUser","newRunForWorld","runid","model","dtd","loadRunFromWorld","getCurrentRunId","serverError","injectFiltersFromSession","injectScopeFromSession","startrecord","endrecord","sort","direction","runs","flag","isInitComplete","createResponse","serial","save","patchResponse","sessionFilter","runopts","trashed","scopeByGroup","scopeByUser","otherFields","param","saved","mark","remove","toMark","rs","existingOptions","proms","r","when","autoRestore","getRuns","modifiers","scopedFilter","opModifiers","include","cookieNameFromOptions","prefix","advanceOperation","BaselineStrategyToUse","includeBaseLine","BaselineStrategy","NoneStrategy","baseline","strategyUtils","mergeRunOptions","baselineName","runName","savedRuns","origGetRuns","args","current","LastUnsavedStrategy","saveAndAdvance","metadata","clone","sa","StateService","advanceOpns","rutil","normalizeOperations","exclude","ops","markSaved","savedResponse","advance","ReuseinitStrategy","stepTo","outputModifiers","currentFilter","currentParams","buildStrategy","worldId","resolveWith","WorldApi","_auth","getCurrentWorld","getCurrentRun","getAndRestoreLatestRun","currentWorldId","runOpts","rm","TransportFactory","uploadBody","fileName","contents","boundary","body","uploadFileOptions","filePath","pop","serviceOptions","folderType","extraParams","FormData","processData","contentType","upload","accountPath","projectPath","Authorization","http","publicAsyncAPI","getContents","put","replaceExisting","prom","post","xhr","conflictStatus","delete","rename","newName","patch","apiEndpoint","fullUrl","transportOptions","assetApiParams","scopeConfig","validateFilename","filename","validateUrlParams","partKeys","buildUrl","parts","method","toLowerCase","urlOptions","createOptions","publicAPI","getServiceOptions","getOptions","files","fullPathFiles","file","assetUrl","password","resp","statusMessage","postParams","topicResolver","channelOptions","makeName","channelName","topics","subscriptionIds","push","publish","returnObjs","setEnv","env","property","qutil","root","getURL","addTrailingSlash","outputModifier","q","attrs","saveAs","serviceUtils","getDefaultOptions","finalOpts","finalParams","getApiUrl","byModel","modelFile","byRunID","runID","getFinalParams","patchUserActiveField","active","isString","objParams","getParms","getGroupDetails","makeUserActive","makeUserInactive","markOnline","markOffline","getStatus","cm","VariablesService","IntrospectionService","updateURLConfig","getFilterURL","filterMatrix","toMatrixFormat","addAutoRestoreHeader","isFilterRunId","autorestoreOpts","X-AutoRestore","updateHTTPConfig","splitGet","splitGetFactory","setFilterOrThrowError","runApiParams","oldSuccess","qs","filters","removeFromMemory","attributes","opsArgs","postOptions","result","prms","operations","opParams","responses","doSingleOp","op","shift","arg","parallel","queue","slice","actualResponse","a","introspect","introspectionConfig","introspection","publicSyncAPI","vs","rest","parseRunIdOrError","httpParams","replay","replayOptions","action","getLocalHost","existingFn","host","localHostFn","isLocal","epiVersion","location","pathname","UrlConfigService","envConf","isLocalhost","actingHost","API_PROTOCOL","HOST_API_MAPPING","forio.com","foriodev.com","publicExports","protocol","apiHost","isCustomDomain","pathHasApp","appPath","accnt","prj","versionPath","PROJECT_APIS","apiMapping","actualProtocol","configProtocol","apiPath","inArray","toQFilter","toIdFilters","getFilters","toQueryFormat","threshold","getById","variable","apiBase","assignmentEndpoint","projectEndpoint","setIdFilterOrThrowError","validateModelOrThrowError","worldApiParams","validParams","update","whitelist","updateOptions","deleteOptions","getWorldsForUser","addUsers","users","u","isObject","updateUser","patchOptions","removeUser","worlds","b","Date","lastModified","currentWorld","deleteRun","currentRunOptions","autoAssign","maxUsers","userIds","getProjectSettings","document","cookie","newCookie","hostname","validHost","domain","setOptions","encodeURIComponent","cookieReg","RegExp","exec","decodeURIComponent","remOptions","destroy","aKeys","nIdx","cookieKey","StorageFactory","optionUtils","EPI_MANAGER_KEY","synchronous","managerOptions","getBaseOptions","libOptions","finalOptions","baseOptions","storeOpts","isEpicenterDomain","serialized","cookieName","sessionDefaults","qutils","404","parameterParser","xhrFields","withCredentials","d","connect","connectOptions","ALLOWED_TO_BE_FUNCTIONS","log","oldSuccessFn","ajaxStatus","ajaxReq","beforeSend","requestUrl","ajaxOptions","trim","delimiter","head","inherit","C","P","dest","obj","props","staticProps","parent","child","p","customDefaults","libDefaults","setDefaults","returnArray","OPERATORS","mtrx","qsToObject","qsArray","returnObj","qKey","qVal","mergeQS","qs1","qs2","obj1","obj2","MAX_URL_LENGTH","returnList","_concat","arr","_normalizePlainObjects","opn","_normalizeStructuredObjects","_normalizeObject","_normalizeLiterals","_normalizeArrays","getValue","getFinalUrl","queryParams","questionIdx","encodeURI","paramsCopy","urlNoIncludes","diff","oldError","currIncludes","includeOpts","currLength","varLenght","reqs","reqParams","isValid","firstResponse","isRunAPI","aggregateRun","aggregatedRuns","idxRun","aggregatedVariables","vars"],"mappings":"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","file":"bundle.js","sourcesContent":["(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require==\"function\"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error(\"Cannot find module '\"+o+\"'\");throw f.code=\"MODULE_NOT_FOUND\",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require==\"function\"&&require;for(var o=0;o> 8 - idx % 1 * 8)\n ) {\n charCode = str.charCodeAt(idx += 3/4);\n if (charCode > 0xFF) {\n throw new InvalidCharacterError(\"'btoa' failed: The string to be encoded contains characters outside of the Latin1 range.\");\n }\n block = block << 8 | charCode;\n }\n return output;\n });\n\n // decoder\n // [https://gist.github.com/1020396] by [https://github.com/atk]\n object.atob || (\n object.atob = function (input) {\n var str = String(input).replace(/=+$/, '');\n if (str.length % 4 == 1) {\n throw new InvalidCharacterError(\"'atob' failed: The string to be decoded is not correctly encoded.\");\n }\n for (\n // initialize result and counters\n var bc = 0, bs, buffer, idx = 0, output = '';\n // get next character\n buffer = str.charAt(idx++);\n // character found in table? initialize bit storage and add its ascii value;\n ~buffer && (bs = bc % 4 ? bs * 64 + buffer : buffer,\n // and if not first of each 4 characters,\n // convert the first 8 bits to one ascii character\n bc++ % 4) ? output += String.fromCharCode(255 & bs >> (-2 * bc & 6)) : 0\n ) {\n // try to find character in table (0-63, not found => -1)\n buffer = chars.indexOf(buffer);\n }\n return output;\n });\n\n}());\n","'use strict';\n/* eslint-disable no-unused-vars */\nvar hasOwnProperty = Object.prototype.hasOwnProperty;\nvar propIsEnumerable = Object.prototype.propertyIsEnumerable;\n\nfunction toObject(val) {\n\tif (val === null || val === undefined) {\n\t\tthrow new TypeError('Object.assign cannot be called with null or undefined');\n\t}\n\n\treturn Object(val);\n}\n\nfunction shouldUseNative() {\n\ttry {\n\t\tif (!Object.assign) {\n\t\t\treturn false;\n\t\t}\n\n\t\t// Detect buggy property enumeration order in older V8 versions.\n\n\t\t// https://bugs.chromium.org/p/v8/issues/detail?id=4118\n\t\tvar test1 = new String('abc'); // eslint-disable-line\n\t\ttest1[5] = 'de';\n\t\tif (Object.getOwnPropertyNames(test1)[0] === '5') {\n\t\t\treturn false;\n\t\t}\n\n\t\t// https://bugs.chromium.org/p/v8/issues/detail?id=3056\n\t\tvar test2 = {};\n\t\tfor (var i = 0; i < 10; i++) {\n\t\t\ttest2['_' + String.fromCharCode(i)] = i;\n\t\t}\n\t\tvar order2 = Object.getOwnPropertyNames(test2).map(function (n) {\n\t\t\treturn test2[n];\n\t\t});\n\t\tif (order2.join('') !== '0123456789') {\n\t\t\treturn false;\n\t\t}\n\n\t\t// https://bugs.chromium.org/p/v8/issues/detail?id=3056\n\t\tvar test3 = {};\n\t\t'abcdefghijklmnopqrst'.split('').forEach(function (letter) {\n\t\t\ttest3[letter] = letter;\n\t\t});\n\t\tif (Object.keys(Object.assign({}, test3)).join('') !==\n\t\t\t\t'abcdefghijklmnopqrst') {\n\t\t\treturn false;\n\t\t}\n\n\t\treturn true;\n\t} catch (e) {\n\t\t// We don't expect any of the above to throw, but better to be safe.\n\t\treturn false;\n\t}\n}\n\nmodule.exports = shouldUseNative() ? Object.assign : function (target, source) {\n\tvar from;\n\tvar to = toObject(target);\n\tvar symbols;\n\n\tfor (var s = 1; s < arguments.length; s++) {\n\t\tfrom = Object(arguments[s]);\n\n\t\tfor (var key in from) {\n\t\t\tif (hasOwnProperty.call(from, key)) {\n\t\t\t\tto[key] = from[key];\n\t\t\t}\n\t\t}\n\n\t\tif (Object.getOwnPropertySymbols) {\n\t\t\tsymbols = Object.getOwnPropertySymbols(from);\n\t\t\tfor (var i = 0; i < symbols.length; i++) {\n\t\t\t\tif (propIsEnumerable.call(from, symbols[i])) {\n\t\t\t\t\tto[symbols[i]] = from[symbols[i]];\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\t}\n\n\treturn to;\n};\n","/**\n * Epicenter Javascript libraries\n * v<%= version %>\n * https://github.com/forio/epicenter-js-libs\n */\n\nvar F = {\n _private: {}, //need this hook now because tests expect everything to be global. Delete once tests are browserified\n util: {},\n factory: {},\n transport: {},\n store: {},\n service: {},\n manager: {\n strategy: {}\n },\n\n};\n\nF.load = require('./env-load');\n\nif (!global.SKIP_ENV_LOAD) {\n F.load();\n}\n\nF.util.query = require('./util/query-util');\nF.util.run = require('./util/run-util');\nF.util.classFrom = require('./util/inherit');\nF._private.strategyutils = require('./managers/strategy-utils');\n\nF.factory.Transport = require('./transport/http-transport-factory');\nF.transport.Ajax = require('./transport/ajax-http-transport');\n\nF.service.URL = require('./service/url-config-service');\nF.service.Config = require('./service/configuration-service');\nF.service.Run = require('./service/run-api-service');\nF.service.File = require('./service/admin-file-service');\nF.service.Variables = require('./service/variables-api-service');\nF.service.Data = require('./service/data-api-service');\nF.service.Auth = require('./service/auth-api-service');\nF.service.World = require('./service/world-api-adapter');\nF.service.State = require('./service/state-api-adapter');\nF.service.User = require('./service/user-api-adapter');\nF.service.Member = require('./service/member-api-adapter');\nF.service.Asset = require('./service/asset-api-adapter');\nF.service.Group = require('./service/group-api-service');\nF.service.Introspect = require('./service/introspection-api-service');\nF.service.Presence = require('./service/presence-api-service');\n\nF.store.Cookie = require('./store/cookie-store');\nF.factory.Store = require('./store/store-factory');\n\nF.manager.ScenarioManager = require('./managers/scenario-manager');\nF.manager.RunManager = require('./managers/run-manager');\nF.manager.AuthManager = require('./managers/auth-manager');\nF.manager.WorldManager = require('./managers/world-manager');\nF.manager.SavedRunsManager = require('./managers/saved-runs-manager');\n\nvar strategies = require('./managers/run-strategies');\nF.manager.strategy = strategies.list; //TODO: this is not really a manager so namespace this better\n\nF.manager.ChannelManager = require('./managers/epicenter-channel-manager');\nF.service.Channel = require('./service/channel-service');\n\nF.version = '<%= version %>';\nF.api = require('./api-version.json');\n\nF.constants = require('./managers/key-names');\n\nglobal.F = F;\nmodule.exports = F;\n","'use strict';\n\nvar URLConfigService = require('./service/url-config-service');\n\nvar envLoad = function (callback) {\n var urlService = new URLConfigService();\n var infoUrl = urlService.getAPIPath('config');\n var envPromise = $.ajax({ url: infoUrl, async: false });\n envPromise = envPromise.then(function (res) {\n var overrides = res.api;\n URLConfigService.defaults = $.extend(URLConfigService.defaults, overrides);\n });\n return envPromise.then(callback).fail(callback);\n};\n\nmodule.exports = envLoad;\n","/**\n* ## Authorization Manager\n*\n* The Authorization Manager provides an easy way to manage user authentication (logging in and out) and authorization (keeping track of tokens, sessions, and groups) for projects.\n*\n* The Authorization Manager is most useful for [team projects](../../../glossary/#team) with an access level of [Authenticated](../../../glossary/#access). These projects are accessed by [end users](../../../glossary/#users) who are members of one or more [groups](../../../glossary/#groups).\n*\n* #### Using the Authorization Manager\n*\n* To use the Authorization Manager, instantiate it. Then, make calls to any of the methods you need:\n*\n* var authMgr = new F.manager.AuthManager({\n* account: 'acme-simulations',\n* userName: 'enduser1',\n* password: 'passw0rd'\n* });\n* authMgr.login().then(function () {\n* authMgr.getCurrentUserSessionInfo();\n* });\n*\n*\n* The `options` object passed to the `F.manager.AuthManager()` call can include:\n*\n* * `account`: The account id for this `userName`. In the Epicenter UI, this is the **Team ID** (for team projects) or the **User ID** (for personal projects).\n* * `userName`: Email or username to use for logging in.\n* * `password`: Password for specified `userName`.\n* * `project`: The **Project ID** for the project to log this user into. Optional.\n* * `groupId`: Id of the group to which `userName` belongs. Required for end users if the `project` is specified.\n*\n* If you prefer starting from a template, the Epicenter JS Libs [Login Component](../../#components) uses the Authorization Manager as well. This sample HTML page (and associated CSS and JS files) provides a login form for team members and end users of your project. It also includes a group selector for end users that are members of multiple groups.\n*/\n\n'use strict';\nvar AuthAdapter = require('../service/auth-api-service');\nvar MemberAdapter = require('../service/member-api-adapter');\nvar GroupService = require('../service/group-api-service');\nvar SessionManager = require('../store/session-manager');\nvar _pick = require('../util/object-util')._pick;\nvar objectAssign = require('object-assign');\n\nvar atob = window.atob || require('Base64').atob;\n\nvar defaults = {\n requiresGroup: true\n};\n\nfunction AuthManager(options) {\n options = $.extend(true, {}, defaults, options);\n this.sessionManager = new SessionManager(options);\n this.options = this.sessionManager.getMergedOptions();\n\n this.authAdapter = new AuthAdapter(this.options);\n}\n\nvar _findUserInGroup = function (members, id) {\n for (var j = 0; j < members.length; j++) {\n if (members[j].userId === id) {\n return members[j];\n }\n }\n return null;\n};\n\nAuthManager.prototype = $.extend(AuthManager.prototype, {\n\n /**\n * Logs user in.\n *\n * **Example**\n *\n * authMgr.login({\n * account: 'acme-simulations',\n * project: 'supply-chain-game',\n * userName: 'enduser1',\n * password: 'passw0rd'\n * })\n * .then(function(statusObj) {\n * // if enduser1 belongs to exactly one group\n * // (or if the login() call is modified to include the group id)\n * // continue here\n * })\n * .fail(function(statusObj) {\n * // if enduser1 belongs to multiple groups,\n * // the login() call fails\n * // and returns all groups of which the user is a member\n * for (var i=0; i < statusObj.userGroups.length; i++) {\n * console.log(statusObj.userGroups[i].name, statusObj.userGroups[i].groupId);\n * }\n * });\n *\n * **Parameters**\n *\n * @param {Object} options (Optional) Overrides for configuration options. If not passed in when creating an instance of the manager (`F.manager.AuthManager()`), these options should include:\n * @param {string} options.account The account id for this `userName`. In the Epicenter UI, this is the **Team ID** (for team projects) or the **User ID** (for personal projects).\n * @param {string} options.userName Email or username to use for logging in.\n * @param {string} options.password Password for specified `userName`.\n * @param {string} options.project (Optional) The **Project ID** for the project to log this user into.\n * @param {string} options.groupId The id of the group to which `userName` belongs. Required for [end users](../../../glossary/#users) if the `project` is specified and if the end users are members of multiple [groups](../../../glossary/#groups), otherwise optional.\n * @return {Promise}\n */\n login: function (options) {\n var me = this;\n var $d = $.Deferred();\n var sessionManager = this.sessionManager;\n var adapterOptions = sessionManager.getMergedOptions({ success: $.noop, error: $.noop }, options);\n var outSuccess = adapterOptions.success;\n var outError = adapterOptions.error;\n var groupId = adapterOptions.groupId;\n\n var decodeToken = function (token) {\n var encoded = token.split('.')[1];\n while (encoded.length % 4 !== 0) { //eslint-disable-line\n encoded += '=';\n }\n return JSON.parse(atob(encoded));\n };\n\n var handleGroupError = function (message, statusCode, data, type) {\n // logout the user since it's in an invalid state with no group selected\n me.logout().then(function () {\n var error = $.extend(true, {}, data, { statusText: message, status: statusCode, type: type });\n $d.reject(error);\n });\n };\n\n var handleSuccess = function (response) {\n var token = response.access_token;\n var userInfo = decodeToken(token);\n var oldGroups = sessionManager.getSession(adapterOptions).groups || {};\n var userGroupOpts = $.extend(true, {}, adapterOptions, { success: $.noop });\n var data = { auth: response, user: userInfo };\n var project = adapterOptions.project;\n var isTeamMember = userInfo.parent_account_id === null;\n var requiresGroup = adapterOptions.requiresGroup && project;\n\n var userName = (userInfo.user_name || '').split('/')[0]; //of form /\n var sessionInfo = {\n auth_token: token,\n account: adapterOptions.account,\n project: project,\n userId: userInfo.user_id,\n groups: oldGroups,\n isTeamMember: isTeamMember,\n userName: userName,\n };\n // The group is not required if the user is not logging into a project\n if (!requiresGroup) {\n sessionManager.saveSession(sessionInfo);\n outSuccess.apply(this, [data]);\n $d.resolve(data);\n return;\n }\n\n var handleGroupList = function (groupList) {\n data.userGroups = groupList;\n\n var group = null;\n if (groupList.length === 0) {\n handleGroupError('The user has no groups associated in this account', 403, data, 'NO_GROUPS');\n return;\n } else if (groupList.length === 1) {\n // Select the only group\n group = groupList[0];\n } else if (groupList.length > 1) {\n if (groupId) {\n var filteredGroups = $.grep(groupList, function (resGroup) {\n return resGroup.groupId === groupId;\n });\n group = filteredGroups.length === 1 ? filteredGroups[0] : null;\n }\n }\n\n if (group) {\n // A team member does not get the group members because is calling the Group API\n // but it's automatically a fac user\n var isFac = isTeamMember ? true : _findUserInGroup(group.members, userInfo.user_id).role === 'facilitator';\n var groupData = {\n groupId: group.groupId,\n groupName: group.name,\n isFac: isFac\n };\n var sessionInfoWithGroup = objectAssign({}, sessionInfo, groupData);\n sessionInfo.groups[project] = groupData;\n me.sessionManager.saveSession(sessionInfoWithGroup, adapterOptions);\n outSuccess.apply(this, [data]);\n $d.resolve(data);\n } else {\n handleGroupError('This user is associated with more than one group. Please specify a group id to log into and try again', 403, data, 'MULTIPLE_GROUPS');\n }\n };\n\n if (!isTeamMember) {\n me.getUserGroups({ userId: userInfo.user_id, token: token }, userGroupOpts)\n .then(handleGroupList, $d.reject);\n } else {\n var opts = objectAssign({}, userGroupOpts, { token: token });\n var groupService = new GroupService(opts);\n groupService.getGroups({ account: adapterOptions.account, project: project })\n .then(function (groups) {\n // Group API returns id instead of groupId\n groups.forEach(function (group) {\n group.groupId = group.id;\n });\n\n if (groups.length) {\n handleGroupList(groups);\n } else {\n //either it's a private project or there are no groups\n sessionManager.saveSession(sessionInfo);\n outSuccess.apply(this, [data]);\n $d.resolve(data);\n return;\n }\n }, $d.reject);\n }\n };\n\n adapterOptions.success = handleSuccess;\n adapterOptions.error = function (response) {\n if (adapterOptions.account) {\n // Try to login as a system user\n adapterOptions.account = null;\n adapterOptions.error = function () {\n outError.apply(this, arguments);\n $d.reject(response);\n };\n\n me.authAdapter.login(adapterOptions);\n return;\n }\n\n outError.apply(this, arguments);\n $d.reject(response);\n };\n\n this.authAdapter.login(adapterOptions);\n return $d.promise();\n },\n\n /**\n * Logs user out by clearing all session information.\n *\n * **Example**\n *\n * authMgr.logout();\n *\n * **Parameters**\n *\n * @param {Object} options (Optional) Overrides for configuration options.\n * @return {Promise}\n */\n logout: function (options) {\n var me = this;\n var adapterOptions = this.sessionManager.getMergedOptions(options);\n\n var removeCookieFn = function (response) {\n me.sessionManager.removeSession();\n };\n\n return this.authAdapter.logout(adapterOptions).then(removeCookieFn);\n },\n\n /**\n * Returns the existing user access token if the user is already logged in. Otherwise, logs the user in, creating a new user access token, and returns the new token. (See [more background on access tokens](../../../project_access/)).\n *\n * **Example**\n *\n * authMgr.getToken()\n * .then(function (token) {\n * console.log('My token is ', token);\n * });\n *\n * **Parameters**\n * @param {Object} options (Optional) Overrides for configuration options.\n * @return {Promise}\n */\n getToken: function (options) {\n var httpOptions = this.sessionManager.getMergedOptions(options);\n\n var session = this.sessionManager.getSession(httpOptions);\n var $d = $.Deferred();\n if (session.auth_token) {\n $d.resolve(session.auth_token);\n } else {\n this.login(httpOptions).then($d.resolve);\n }\n return $d.promise();\n },\n\n /**\n * Returns an array of group records, one for each group of which the current user is a member. Each group record includes the group `name`, `account`, `project`, and `groupId`.\n *\n * If some end users in your project are members of multiple groups, this is a useful method to call on your project's login page. When the user attempts to log in, you can use this to display the groups of which the user is member, and have the user select the correct group to log in to for this session.\n *\n * **Example**\n *\n * // get groups for current user\n * var sessionObj = authMgr.getCurrentUserSessionInfo();\n * authMgr.getUserGroups({ userId: sessionObj.userId, token: sessionObj.auth_token })\n * .then(function (groups) {\n * for (var i=0; i < groups.length; i++)\n * { console.log(groups[i].name); }\n * });\n *\n * // get groups for particular user\n * authMgr.getUserGroups({userId: 'b1c19dda-2d2e-4777-ad5d-3929f17e86d3', token: savedProjAccessToken });\n *\n * **Parameters**\n * @param {Object} params Object with a userId and token properties.\n * @param {String} params.userId The userId. If looking up groups for the currently logged in user, this is in the session information. Otherwise, pass a string.\n * @param {String} params.token The authorization credentials (access token) to use for checking the groups for this user. If looking up groups for the currently logged in user, this is in the session information. A team member's token or a project access token can access all the groups for all end users in the team or project.\n * @param {Object} options (Optional) Overrides for configuration options.\n * @return {Promise}\n */\n getUserGroups: function (params, options) {\n var adapterOptions = this.sessionManager.getMergedOptions({ success: $.noop }, options);\n var $d = $.Deferred();\n var outSuccess = adapterOptions.success;\n\n adapterOptions.success = function (memberInfo) {\n // The member API is at the account scope, we filter by project\n if (adapterOptions.project) {\n memberInfo = $.grep(memberInfo, function (group) {\n return group.project === adapterOptions.project;\n });\n }\n\n outSuccess.apply(this, [memberInfo]);\n $d.resolve(memberInfo);\n };\n\n var memberAdapter = new MemberAdapter({ token: params.token, server: adapterOptions.server });\n memberAdapter.getGroupsForUser(params, adapterOptions).fail($d.reject);\n return $d.promise();\n },\n\n /**\n * Helper method to check if you're currently logged in\n *\n * **Example**\n * \n * var amILoggedIn = authMgr.isLoggedIn();\n *\n * **Parameters**\n * @param {none} none\n * @return {Boolean} true if you're logged in\n */\n isLoggedIn: function () {\n var session = this.getCurrentUserSessionInfo();\n return !!(session && session.userId);\n },\n\n /**\n * Returns session information for the current user, including the `userId`, `account`, `project`, `groupId`, `groupName`, `isFac` (whether the end user is a facilitator of this group), and `auth_token` (user access token).\n *\n * *Important*: This method is synchronous. The session information is returned immediately in an object; no callbacks or promises are needed.\n *\n * Session information is stored in a cookie in the browser.\n *\n * **Example**\n *\n * var sessionObj = authMgr.getCurrentUserSessionInfo();\n *\n * **Parameters**\n * @param {Object} options (Optional) Overrides for configuration options.\n * @return {Object} session information\n */\n getCurrentUserSessionInfo: function (options) {\n var adapterOptions = this.sessionManager.getMergedOptions({ success: $.noop }, options);\n return this.sessionManager.getSession(adapterOptions);\n },\n\n /*\n * Adds one or more groups to the current session. \n *\n * This method assumes that the project and group exist and the user specified in the session is part of this project and group.\n *\n * Returns the new session object.\n *\n * **Example**\n *\n * authMgr.addGroups({ project: 'hello-world', groupName: 'groupName', groupId: 'groupId' });\n * authMgr.addGroups([{ project: 'hello-world', groupName: 'groupName', groupId: 'groupId' }, { project: 'hello-world', groupName: '...' }]);\n *\n * **Parameters**\n * @param {object|array} groups (Required) The group object must contain the `project` (**Project ID**) and `groupName` properties. If passing an array of such objects, all of the objects must contain *different* `project` (**Project ID**) values: although end users may be logged in to multiple projects at once, they may only be logged in to one group per project at a time.\n * @param {string} group.isFac (optional) Defaults to `false`. Set to `true` if the user in the session should be a facilitator in this group.\n * @param {string} group.groupId (optional) Defaults to undefined. Needed mostly for the Members API.\n * @return {Object} session information\n */\n addGroups: function (groups) {\n var session = this.getCurrentUserSessionInfo();\n var isArray = Array.isArray(groups);\n groups = isArray ? groups : [groups];\n\n $.each(groups, function (index, group) {\n var extendedGroup = $.extend({}, { isFac: false }, group);\n var project = extendedGroup.project;\n var validProps = ['groupName', 'groupId', 'isFac'];\n if (!project || !extendedGroup.groupName) {\n throw new Error('No project or groupName specified.');\n }\n // filter object\n extendedGroup = _pick(extendedGroup, validProps);\n session.groups[project] = extendedGroup;\n });\n this.sessionManager.saveSession(session);\n return session;\n }\n});\n\nmodule.exports = AuthManager;\n","'use strict';\n\n/**\n * ## Channel Manager\n *\n * There are two main use cases for the channel: event notifications and chat messages.\n *\n * If you are developing with Epicenter.js, you should use the [Epicenter Channel Manager](../epicenter-channel-manager/) rather than this more generic Channel Manager. (The Epicenter Channel Manager is a wrapper that instantiates a Channel Manager with Epicenter-specific defaults.) The Epicenter Channel Manager documentation also has more [background](../epicenter-channel-manager/#background) information on channels and their use. \n *\n * However, you can work directly with the Channel Manager if you like. (This might be useful if you are working through Node.js, for example, `require('manager/channel-manager')`.)\n *\n * The Channel Manager is a wrapper around the default [cometd JavaScript library](http://docs.cometd.org/2/reference/javascript.html), `$.cometd`. It provides a few nice features that `$.cometd` doesn't, including:\n *\n * * Automatic re-subscription to channels if you lose your connection\n * * Online / Offline notifications\n * * 'Events' for cometd notifications (instead of having to listen on specific meta channels)\n *\n * You'll need to include the `epicenter-multiplayer-dependencies.js` library in addition to the `epicenter.js` library in your project to use the Channel Manager. (See [Including Epicenter.js](../../#include).)\n *\n * To use the Channel Manager in client-side JavaScript, instantiate the [Epicenter Channel Manager](../epicenter-channel-manager/), get a particular channel -- that is, an instance of a [Channel Service](../channel-service/) -- then use the channel's `subscribe()` and `publish()` methods to subscribe to topics or publish data to topics.\n *\n * var cm = new F.manager.ChannelManager();\n * var gc = cm.getGroupChannel();\n * // because we used an Epicenter Channel Manager to get the group channel,\n * // subscribe() and publish() here default to the base topic for the group;\n * gc.subscribe('', function(data) { console.log(data); });\n * gc.publish('', { message: 'a new message to the group' });\n *\n * The parameters for instantiating a Channel Manager include:\n *\n * * `options` The options object to configure the Channel Manager. Besides the common options listed here, see http://docs.cometd.org/reference/javascript.html for other supported options.\n * * `options.url` The Cometd endpoint URL.\n * * `options.websocketEnabled` Whether websocket support is active (boolean).\n * * `options.channel` Other defaults to pass on to instances of the underlying Channel Service. See [Channel Service](../channel-service/) for details.\n *\n */\n\nvar Channel = require('../service/channel-service');\nvar SessionManager = require('../store/session-manager');\n\nvar ChannelManager = function (options) {\n if (!$.cometd) {\n throw new Error('Cometd library not found. Please include epicenter-multiplayer-dependencies.js');\n }\n if (!options || !options.url) {\n throw new Error('Please provide an url for the cometd server');\n }\n\n var defaults = {\n /**\n * The Cometd endpoint URL.\n * @type {string}\n */\n url: '',\n\n /**\n * The log level for the channel (logs to console).\n * @type {string}\n */\n logLevel: 'info',\n\n /**\n * Whether websocket support is active. Defaults to `true`.\n * @type {boolean}\n */\n websocketEnabled: true,\n\n /**\n * Whether the ACK extension is enabled. Defaults to `true`. See [https://docs.cometd.org/current/reference/#_extensions_acknowledge](https://docs.cometd.org/current/reference/#_extensions_acknowledge) for more info.\n * @type {boolean}\n */\n ackEnabled: true,\n\n /**\n * If false each instance of Channel will have a separate cometd connection to server, which could be noisy. Set to true to re-use the same connection across instances.\n * @type {boolean}\n */\n shareConnection: false,\n\n /**\n * Other defaults to pass on to instances of the underlying [Channel Service](../channel-service/), which are created through `getChannel()`.\n * @type {object}\n */\n channel: {\n\n },\n\n /**\n * Options to pass to the channel handshake.\n *\n * For example, the [Epicenter Channel Manager](../epicenter-channel-manager/) passes `ext` and authorization information. More information on possible options is in the details of the underlying [Push Channel API](../../../rest_apis/multiplayer/channel/).\n *\n * @type {object}\n */\n handshake: undefined\n };\n this.sessionManager = new SessionManager();\n var defaultCometOptions = this.sessionManager.getMergedOptions(defaults, options);\n this.currentSubscriptions = [];\n this.options = defaultCometOptions;\n\n if (defaultCometOptions.shareConnection && ChannelManager.prototype._cometd) {\n this.cometd = ChannelManager.prototype._cometd;\n return this;\n }\n var cometd = new $.CometD();\n ChannelManager.prototype._cometd = cometd;\n\n cometd.websocketEnabled = defaultCometOptions.websocketEnabled;\n cometd.ackEnabled = defaultCometOptions.ackEnabled;\n\n this.isConnected = false;\n var connectionBroken = function (message) {\n $(this).trigger('disconnect', message);\n };\n var connectionSucceeded = function (message) {\n $(this).trigger('connect', message);\n };\n var me = this;\n\n cometd.configure(defaultCometOptions);\n\n cometd.addListener('/meta/connect', function (message) {\n var wasConnected = this.isConnected;\n this.isConnected = (message.successful === true);\n if (!wasConnected && this.isConnected) { //Connecting for the first time\n connectionSucceeded.call(this, message);\n } else if (wasConnected && !this.isConnected) { //Only throw disconnected message fro the first disconnect, not once per try\n connectionBroken.call(this, message);\n }\n }.bind(this));\n\n cometd.addListener('/meta/disconnect', connectionBroken);\n\n cometd.addListener('/meta/handshake', function (message) {\n if (message.successful) {\n //http://docs.cometd.org/reference/javascript_subscribe.html#javascript_subscribe_meta_channels\n // ^ \"dynamic subscriptions are cleared (like any other subscription) and the application needs to figure out which dynamic subscription must be performed again\"\n cometd.batch(function () {\n $(me.currentSubscriptions).each(function (index, subs) {\n cometd.resubscribe(subs);\n });\n });\n }\n });\n\n //Other interesting events for reference\n cometd.addListener('/meta/subscribe', function (message) {\n $(me).trigger('subscribe', message);\n });\n cometd.addListener('/meta/unsubscribe', function (message) {\n $(me).trigger('unsubscribe', message);\n });\n cometd.addListener('/meta/publish', function (message) {\n $(me).trigger('publish', message);\n });\n cometd.addListener('/meta/unsuccessful', function (message) {\n $(me).trigger('error', message);\n });\n\n cometd.handshake(defaultCometOptions.handshake);\n\n this.cometd = cometd;\n};\n\n\nChannelManager.prototype = $.extend(ChannelManager.prototype, {\n\n /**\n * Creates and returns a channel, that is, an instance of a [Channel Service](../channel-service/).\n *\n * **Example**\n *\n * var cm = new F.manager.ChannelManager();\n * var channel = cm.getChannel();\n *\n * channel.subscribe('topic', callback);\n * channel.publish('topic', { myData: 100 });\n *\n * **Parameters**\n * @param {Object|String} options (Optional) If string, assumed to be the base channel url. If object, assumed to be configuration options for the constructor.\n * @return {Channel} Channel instance\n */\n getChannel: function (options) {\n //If you just want to pass in a string\n if (options && !$.isPlainObject(options)) {\n options = {\n base: options\n };\n }\n var defaults = {\n transport: this.cometd\n };\n var channel = new Channel($.extend(true, {}, this.options.channel, defaults, options));\n\n\n //Wrap subs and unsubs so we can use it to re-attach handlers after being disconnected\n var subs = channel.subscribe;\n channel.subscribe = function () {\n var subid = subs.apply(channel, arguments);\n this.currentSubscriptions = this.currentSubscriptions.concat(subid);\n return subid;\n }.bind(this);\n\n\n var unsubs = channel.unsubscribe;\n channel.unsubscribe = function () {\n var removed = unsubs.apply(channel, arguments);\n for (var i = 0; i < this.currentSubscriptions.length; i++) {\n if (this.currentSubscriptions[i].id === removed.id) {\n this.currentSubscriptions.splice(i, 1);\n }\n }\n return removed;\n }.bind(this);\n\n return channel;\n },\n\n /**\n * Start listening for events on this instance. Signature is same as for jQuery Events: http://api.jquery.com/on/.\n *\n * Supported events are: `connect`, `disconnect`, `subscribe`, `unsubscribe`, `publish`, `error`.\n *\n * **Parameters**\n *\n * @param {string} event The event type. See more detail at jQuery Events: http://api.jquery.com/on/.\n */\n on: function (event) {\n $(this).on.apply($(this), arguments);\n },\n\n /**\n * Stop listening for events on this instance. Signature is same as for jQuery Events: http://api.jquery.com/off/.\n *\n * **Parameters**\n *\n * @param {string} event The event type. See more detail at jQuery Events: http://api.jquery.com/off/.\n */\n off: function (event) {\n $(this).off.apply($(this), arguments);\n },\n\n /**\n * Trigger events and execute handlers. Signature is same as for jQuery Events: http://api.jquery.com/trigger/.\n *\n * **Parameters**\n *\n * @param {string} event The event type. See more detail at jQuery Events: http://api.jquery.com/trigger/.\n */\n trigger: function (event) {\n $(this).trigger.apply($(this), arguments);\n }\n});\n\nmodule.exports = ChannelManager;\n","'use strict';\n\n/**\n * ## Epicenter Channel Manager\n *\n * The Epicenter platform provides a push channel, which allows you to publish and subscribe to messages within a [project](../../../glossary/#projects), [group](../../../glossary/#groups), or [multiplayer world](../../../glossary/#world).\n *\n * \n * ### Channel Background\n *\n * Channel notifications are only available for [team projects](../../../glossary/#team). There are two main use cases for the push channel: event notifications and chat messages.\n *\n * #### Event Notifications\n *\n * Within a [multiplayer simulation or world](../../../glossary/#world), it is often useful for your project's [model](../../../writing_your_model/) to alert the [user interface (browser)](../../../creating_your_interface/) that something new has happened.\n *\n * Usually, this \"something new\" is an event within the project, group, or world, such as:\n *\n * * An end user comes online (logs in) or goes offline. (This is especially interesting in a multiplayer world; only available if you have [enabled authorization](../../../updating_your_settings/#general-settings) for the channel.)\n * * An end user is assigned to a world.\n * * An end user updates a variable / makes a decision.\n * * An end user creates or updates data stored in the [Data API](../data-api-service/).\n * * An operation (method) is called. (This is especially interesting if the model is advanced, for instance, the Vensim `step` operation is called.)\n *\n * When these events occur, you often want to have the user interface for one or more end users automatically update with new information.\n *\n * #### Chat Messages\n *\n * Another reason to use the push channel is to allow players (end users) to send chat messages to other players, and to have those messages appear immediately.\n *\n * #### Getting Started\n *\n * For both the event notification and chat message use cases:\n *\n * * First, enable channel notifications for your project.\n * * Channel notifications are only available for [team projects](../../../glossary/#team). To enable notifications for your project, [update your project settings](../../../updating_your_settings/#general-settings) to turn on the **Push Channel** setting, and optionally require authorization for the channel.\n * * Then, instantiate an Epicenter Channel Manager.\n * * Next, get the channel with the scope you want (user, world, group, data).\n * * Finally, use the channel's `subscribe()` and `publish()` methods to subscribe to topics or publish data to topics.\n *\n * Here's an example of those last three steps (instantiate, get channel, subscribe):\n *\n * var cm = new F.manager.ChannelManager();\n * var gc = cm.getGroupChannel();\n * gc.subscribe('', function(data) { console.log(data); });\n * gc.publish('', { message: 'a new message to the group' });\n *\n * For a more detailed example, see a [complete publish and subscribe example](../../../rest_apis/multiplayer/channel/#epijs-example).\n *\n * For details on what data is published automatically to which channels, see [Automatic Publishing of Events](../../../rest_apis/multiplayer/channel/#publish-message-auto).\n *\n * #### Creating an Epicenter Channel Manager\n *\n * The Epicenter Channel Manager is a wrapper around the (more generic) [Channel Manager](../channel-manager/), to instantiate it with Epicenter-specific defaults. If you are interested in including a notification or chat feature in your project, using an Epicenter Channel Manager is the easiest way to get started.\n *\n * You'll need to include the `epicenter-multiplayer-dependencies.js` library in addition to the `epicenter.js` library in your project to use the Epicenter Channel Manager. See [Including Epicenter.js](../../#include).\n *\n * The parameters for instantiating an Epicenter Channel Manager include:\n *\n * * `options` Object with details about the Epicenter project for this Epicenter Channel Manager instance.\n * * `options.account` The Epicenter account id (**Team ID** for team projects, **User ID** for personal projects).\n * * `options.project` Epicenter project id.\n * * `options.userName` Epicenter userName used for authentication.\n * * `options.userId` Epicenter user id used for authentication. Optional; `options.userName` is preferred.\n * * `options.token` Epicenter token used for authentication. (You can retrieve this using `authManager.getToken()` from the [Authorization Manager](../auth-manager/).)\n * * `options.allowAllChannels` If not included or if set to `false`, all channel paths are validated; if your project requires [Push Channel Authorization](../../../updating_your_settings/), you should use this option. If you want to allow other channel paths, set to `true`; this is not common.\n */\n\nvar ChannelManager = require('./channel-manager');\nvar ConfigService = require('../service/configuration-service');\nvar classFrom = require('../util/inherit');\nvar SessionManager = require('../store/session-manager');\n\nvar validTypes = {\n project: true,\n group: true,\n world: true,\n user: true,\n data: true,\n general: true,\n chat: true\n};\nvar getFromSessionOrError = function (value, sessionKeyName, settings) {\n if (!value) {\n if (settings && settings[sessionKeyName]) {\n value = settings[sessionKeyName];\n } else {\n throw new Error(sessionKeyName + ' not found. Please log-in again, or specify ' + sessionKeyName + ' explicitly');\n }\n }\n return value;\n};\n\nvar isPresenceData = function (payload) {\n return payload.data && payload.data.type === 'user' && payload.data.user;\n};\n\nvar __super = ChannelManager.prototype;\nvar EpicenterChannelManager = classFrom(ChannelManager, {\n constructor: function (options) {\n this.sessionManager = new SessionManager(options);\n var defaultCometOptions = this.sessionManager.getMergedOptions(options);\n\n var urlConfig = new ConfigService(defaultCometOptions).get('server');\n if (!defaultCometOptions.url) {\n defaultCometOptions.url = urlConfig.getAPIPath('channel');\n }\n\n if (defaultCometOptions.handshake === undefined) {\n var userName = defaultCometOptions.userName;\n var userId = defaultCometOptions.userId;\n var token = defaultCometOptions.token;\n if ((userName || userId) && token) {\n var userProp = userName ? 'userName' : 'userId';\n var ext = {\n authorization: 'Bearer ' + token\n };\n ext[userProp] = userName ? userName : userId;\n\n defaultCometOptions.handshake = {\n ext: ext\n };\n }\n }\n\n this.options = defaultCometOptions;\n return __super.constructor.call(this, defaultCometOptions);\n },\n\n /**\n * Creates and returns a channel, that is, an instance of a [Channel Service](../channel-service/).\n *\n * This method enforces Epicenter-specific channel naming: all channels requested must be in the form `/{type}/{account id}/{project id}/{...}`, where `type` is one of `run`, `data`, `user`, `world`, or `chat`.\n *\n * **Example**\n *\n * var cm = new F.manager.EpicenterChannelManager();\n * var channel = cm.getChannel('/group/acme/supply-chain-game/');\n *\n * channel.subscribe('topic', callback);\n * channel.publish('topic', { myData: 100 });\n *\n * **Parameters**\n * @param {Object|String} options (Optional) If string, assumed to be the base channel url. If object, assumed to be configuration options for the constructor.\n * @return {Channel} Channel instance\n */\n getChannel: function (options) {\n if (options && typeof options !== 'object') {\n options = {\n base: options\n };\n }\n var channelOpts = $.extend({}, this.options, options);\n var base = channelOpts.base;\n if (!base) {\n throw new Error('No base topic was provided');\n }\n\n if (!channelOpts.allowAllChannels) {\n var baseParts = base.split('/');\n var channelType = baseParts[1];\n if (baseParts.length < 4) { //eslint-disable-line\n throw new Error('Invalid channel base name, it must be in the form /{type}/{account id}/{project id}/{...}');\n }\n if (!validTypes[channelType]) {\n throw new Error('Invalid channel type');\n }\n }\n return __super.getChannel.apply(this, arguments);\n },\n\n /**\n * Create and return a publish/subscribe channel (from the underlying [Channel Manager](../channel-manager/)) for the given [group](../../../glossary/#groups). The group must exist in the account (team) and project provided.\n *\n * There are no notifications from Epicenter on this channel; all messages are user-originated.\n *\n * **Example**\n *\n * var cm = new F.manager.ChannelManager();\n * var gc = cm.getGroupChannel();\n * gc.subscribe('broadcasts', callback);\n *\n * **Return Value**\n *\n * * *Channel* Returns the channel (an instance of the [Channel Service](../channel-service/)).\n *\n * **Parameters**\n *\n * @param {String} groupName (Optional) Group to broadcast to. If not provided, picks up group from current session if end user is logged in.\n * @return {Channel} Channel instance\n */\n getGroupChannel: function (groupName) {\n var session = this.sessionManager.getMergedOptions(this.options);\n groupName = getFromSessionOrError(groupName, 'groupName', session);\n var account = getFromSessionOrError('', 'account', session);\n var project = getFromSessionOrError('', 'project', session);\n\n var baseTopic = ['/group', account, project, groupName].join('/');\n var channel = __super.getChannel.call(this, { base: baseTopic });\n var oldsubs = channel.subscribe;\n channel.subscribe = function (topic, callback, context, options) {\n var callbackWithoutPresenceData = function (payload) {\n if (!isPresenceData(payload)) {\n callback.call(context, payload);\n }\n };\n return oldsubs.call(channel, topic, callbackWithoutPresenceData, context, options);\n };\n return channel;\n },\n\n /**\n * Create and return a publish/subscribe channel (from the underlying [Channel Manager](../channel-manager/)) for the given [world](../../../glossary/#world).\n *\n * This is typically used together with the [World Manager](../world-manager).\n *\n * **Example**\n *\n * var cm = new F.manager.ChannelManager();\n * var worldManager = new F.manager.WorldManager({\n * account: 'acme-simulations',\n * project: 'supply-chain-game',\n * group: 'team1',\n * run: { model: 'model.eqn' }\n * });\n * worldManager.getCurrentWorld().then(function (worldObject, worldAdapter) {\n * var worldChannel = cm.getWorldChannel(worldObject);\n * worldChannel.subscribe('', function (data) {\n * console.log(data);\n * });\n * });\n *\n * **Return Value**\n *\n * * *Channel* Returns the channel (an instance of the [Channel Service](../channel-service/)).\n *\n * **Parameters**\n *\n * @param {String|Object} world The world object or id.\n * @param {String} groupName (Optional) Group the world exists in. If not provided, picks up group from current session if end user is logged in.\n * @return {Channel} Channel instance\n */\n getWorldChannel: function (world, groupName) {\n var worldid = ($.isPlainObject(world) && world.id) ? world.id : world;\n if (!worldid) {\n throw new Error('Please specify a world id');\n }\n var session = this.sessionManager.getMergedOptions(this.options);\n\n groupName = getFromSessionOrError(groupName, 'groupName', session);\n var account = getFromSessionOrError('', 'account', session);\n var project = getFromSessionOrError('', 'project', session);\n\n var baseTopic = ['/world', account, project, groupName, worldid].join('/');\n return __super.getChannel.call(this, { base: baseTopic });\n },\n\n /**\n * Create and return a publish/subscribe channel (from the underlying [Channel Manager](../channel-manager/)) for the current [end user](../../../glossary/#users) in that user's current [world](../../../glossary/#world).\n *\n * This is typically used together with the [World Manager](../world-manager). Note that this channel only gets notifications for worlds currently in memory. (See more background on [persistence](../../../run_persistence).)\n *\n * **Example**\n *\n * var cm = new F.manager.ChannelManager();\n * var worldManager = new F.manager.WorldManager({\n * account: 'acme-simulations',\n * project: 'supply-chain-game',\n * group: 'team1',\n * run: { model: 'model.eqn' }\n * });\n * worldManager.getCurrentWorld().then(function (worldObject, worldAdapter) {\n * var userChannel = cm.getUserChannel(worldObject);\n * userChannel.subscribe('', function (data) {\n * console.log(data);\n * });\n * });\n *\n *\n * **Return Value**\n *\n * * *Channel* Returns the channel (an instance of the [Channel Service](../channel-service/)).\n *\n * **Parameters**\n *\n * @param {String|Object} world World object or id.\n * @param {String|Object} user (Optional) User object or id. If not provided, picks up user id from current session if end user is logged in.\n * @param {String} groupName (Optional) Group the world exists in. If not provided, picks up group from current session if end user is logged in.\n * @return {Channel} Channel instance\n */\n getUserChannel: function (world, user, groupName) {\n var worldid = ($.isPlainObject(world) && world.id) ? world.id : world;\n if (!worldid) {\n throw new Error('Please specify a world id');\n }\n var session = this.sessionManager.getMergedOptions(this.options);\n\n var userid = ($.isPlainObject(user) && user.id) ? user.id : user;\n userid = getFromSessionOrError(userid, 'userId', session);\n groupName = getFromSessionOrError(groupName, 'groupName', session);\n\n var account = getFromSessionOrError('', 'account', session);\n var project = getFromSessionOrError('', 'project', session);\n\n var baseTopic = ['/user', account, project, groupName, worldid, userid].join('/');\n return __super.getChannel.call(this, { base: baseTopic });\n },\n\n /**\n * Create and return a publish/subscribe channel (from the underlying [Channel Manager](../channel-manager/)) that automatically tracks the presence of an [end user](../../../glossary/#users), that is, whether the end user is currently online in this group. Notifications are automatically sent when the end user comes online, and when the end user goes offline (not present for more than 2 minutes). Useful in multiplayer games for letting each end user know whether other users in their group are also online.\n *\n * Note that the presence channel is tracking all end users in a group. In particular, if the project additionally splits each group into [worlds](../world-manager/), this channel continues to show notifications for all end users in the group (not restricted by worlds).\n *\n * **Example**\n *\n * var cm = new F.manager.ChannelManager();\n * var pc = cm.getPresenceChannel();\n * pc.subscribe('', function (data) {\n * // 'data' is the entire message object to the channel;\n * // parse for information of interest\n * if (data.data.subType === 'disconnect') {\n * console.log('user ', data.data.user.userName, 'disconnected at ', data.data.date);\n * }\n * if (data.data.subType === 'connect') {\n * console.log('user ', data.data.user.userName, 'connected at ', data.data.date);\n * }\n * });\n *\n *\n * **Return Value**\n *\n * * *Channel* Returns the channel (an instance of the [Channel Service](../channel-service/)).\n *\n * **Parameters**\n *\n * @param {String} groupName (Optional) Group the end user is in. If not provided, picks up group from current session if end user is logged in.\n * @return {Channel} Channel instance\n */\n getPresenceChannel: function (groupName) {\n var session = this.sessionManager.getMergedOptions(this.options);\n groupName = getFromSessionOrError(groupName, 'groupName', session);\n var account = getFromSessionOrError('', 'account', session);\n var project = getFromSessionOrError('', 'project', session);\n\n var baseTopic = ['/group', account, project, groupName].join('/');\n var channel = __super.getChannel.call(this, { base: baseTopic });\n var oldsubs = channel.subscribe;\n channel.subscribe = function (topic, callback, context, options) {\n var callbackWithOnlyPresenceData = function (payload) {\n if (isPresenceData(payload)) {\n callback.call(context, payload);\n }\n };\n return oldsubs.call(channel, topic, callbackWithOnlyPresenceData, context, options);\n };\n return channel;\n },\n\n /**\n * Create and return a publish/subscribe channel (from the underlying [Channel Manager](../channel-manager/)) for the given collection. (The collection name is specified in the `root` argument when the [Data Service](../data-api-service/) is instantiated.) Must be one of the collections in this account (team) and project.\n *\n * There are automatic notifications from Epicenter on this channel when data is created, updated, or deleted in this collection. See more on [automatic messages to the data channel](../../../rest_apis/multiplayer/channel/#data-messages).\n *\n * **Example**\n *\n * var cm = new F.manager.ChannelManager();\n * var dc = cm.getDataChannel('survey-responses');\n * dc.subscribe('', function(data, meta) {\n * console.log(data);\n *\n * // meta.date is time of change,\n * // meta.subType is the kind of change: new, update, or delete\n * // meta.path is the full path to the changed data\n * console.log(meta);\n * });\n *\n * **Return Value**\n *\n * * *Channel* Returns the channel (an instance of the [Channel Service](../channel-service/)).\n *\n * **Parameters**\n *\n * @param {String} collection Name of collection whose automatic notifications you want to receive.\n * @return {Channel} Channel instance\n */\n getDataChannel: function (collection) {\n if (!collection) {\n throw new Error('Please specify a collection to listen on.');\n }\n\n var session = this.sessionManager.getMergedOptions(this.options);\n var account = getFromSessionOrError('', 'account', session);\n var project = getFromSessionOrError('', 'project', session);\n var baseTopic = ['/data', account, project, collection].join('/');\n var channel = __super.getChannel.call(this, { base: baseTopic });\n\n //TODO: Fix after Epicenter bug is resolved\n var oldsubs = channel.subscribe;\n channel.subscribe = function (topic, callback, context, options) {\n var callbackWithCleanData = function (payload) {\n var meta = {\n path: payload.channel,\n subType: payload.data.subType,\n date: payload.data.date,\n dataPath: payload.data.data.path,\n };\n var actualData = payload.data.data;\n if (actualData.data !== undefined) { //Delete notifications are one data-level behind of course\n actualData = actualData.data;\n }\n\n callback.call(context, actualData, meta);\n };\n return oldsubs.call(channel, topic, callbackWithCleanData, context, options);\n };\n\n return channel;\n }\n});\n\nmodule.exports = EpicenterChannelManager;\n","'use strict';\n\nmodule.exports = {\n EPI_SESSION_KEY: 'epicenterjs.session',\n STRATEGY_SESSION_KEY: 'epicenter-scenario'\n};","/**\n* ## Run Manager\n*\n* The Run Manager gives you access to runs for your project. This allows you to read and update variables, call operations, etc. Additionally, the Run Manager gives you control over run creation depending on run states. Specifically, you can select [run creation strategies (rules)](../strategies/) for which runs end users of your project work with when they log in to your project.\n*\n* There are many ways to create new runs, including the Epicenter.js [Run Service](../run-api-service/) and the RESFTful [Run API](../../../rest_apis/aggregate_run_api). However, for some projects it makes more sense to pick up where the user left off, using an existing run. And in some projects, whether to create a new run or use an existing one is conditional, for example based on characteristics of the existing run or your own knowledge about the model. The Run Manager provides this level of control: your call to `getRun()`, rather than always returning a new run, returns a run based on the strategy you've specified.\n*\n*\n* ### Using the Run Manager to create and access runs\n*\n* To use the Run Manager, instantiate it by passing in:\n*\n* * `run`: (required) Run object. Must contain:\n* * `account`: Epicenter account id (**Team ID** for team projects, **User ID** for personal projects).\n* * `project`: Epicenter project id.\n* * `model`: The name of your primary model file. (See more on [Writing your Model](../../../writing_your_model/).)\n* * `scope`: (optional) Scope object for the run, for example `scope.group` with value of the name of the group.\n* * `server`: (optional) An object with one field, `host`. The value of `host` is the string `api.forio.com`, the URI of the Forio server. This is automatically set, but you can pass it explicitly if desired. It is most commonly used for clarity when you are [hosting an Epicenter project on your own server](../../../how_to/self_hosting/).\n* * `files`: (optional) If and only if you are using a Vensim model and you have additional data to pass in to your model, you can optionally pass a `files` object with the names of the files, for example: `\"files\": {\"data\": \"myExtraData.xls\"}`. (See more on [Using External Data in Vensim](../../../model_code/vensim/vensim_example_xls/).)\n*\n* * `strategy`: (optional) Run creation strategy for when to create a new run and when to reuse an end user's existing run. This is *optional*; by default, the Run Manager selects `reuse-per-session`, or `reuse-last-initialized` if you also pass in an initial operation. See [below](#using-the-run-manager-to-access-and-register-strategies) for more information on strategies.\n*\n* * `strategyOptions`: (optional) Additional options passed directly to the [run creation strategy](../strategies/).\n*\n* * `sessionKey`: (optional) Name of browser cookie in which to store run information, including run id. Many conditional strategies, including the provided strategies, rely on this browser cookie to store the run id and help make the decision of whether to create a new run or use an existing one. The name of this cookie defaults to `epicenter-scenario` and can be set with the `sessionKey` parameter. This can also be a function which returns a string, if you'd like to control this at runtime.\n*\n*\n* After instantiating a Run Manager, make a call to `getRun()` whenever you need to access a run for this end user. The `RunManager.run` contains the instantiated [Run Service](../run-api-service/). The Run Service allows you to access variables, call operations, etc.\n*\n* **Example**\n*\n* var rm = new F.manager.RunManager({\n* run: {\n* account: 'acme-simulations',\n* project: 'supply-chain-game',\n* model: 'supply-chain-model.jl',\n* server: { host: 'api.forio.com' }\n* },\n* strategy: 'reuse-never',\n* sessionKey: 'epicenter-session'\n* });\n* rm.getRun()\n* .then(function(run) {\n* // the return value of getRun() is a run object\n* var thisRunId = run.id;\n* // the RunManager.run also contains the instantiated Run Service,\n* // so any Run Service method is valid here\n* rm.run.do('runModel');\n* })\n*\n*\n* ### Using the Run Manager to access and register strategies\n*\n* The `strategy` for a Run Manager describes when to create a new run and when to reuse an end user's existing run. The Run Manager is responsible for passing a strategy everything it might need to determine the 'correct' run, that is, how to find the best existing run and how to decide when to create a new run.\n*\n* There are several common strategies provided as part of Epicenter.js, which you can list by accessing `F.manager.RunManager.strategies`. You can also create your own strategies, and register them to use with Run Managers. See [Run Manager Strategies](../strategies/) for details.\n* \n*/\n\n'use strict';\nvar strategies = require('./run-strategies');\nvar specialOperations = require('./special-operations');\n\nvar RunService = require('../service/run-api-service');\nvar SessionManager = require('../store/session-manager');\n\nvar util = require('../util/object-util');\nvar keyNames = require('./key-names');\n\nfunction patchRunService(service, manager) {\n if (service.patched) {\n return service;\n }\n\n var orig = service.do;\n service.do = function (operation, params, options) {\n var reservedOps = Object.keys(specialOperations);\n if (reservedOps.indexOf(operation) === -1) {\n return orig.apply(service, arguments);\n } else {\n return specialOperations[operation].call(service, params, options, manager);\n }\n };\n\n service.patched = true;\n\n return service;\n}\n\nfunction sessionKeyFromOptions(options, runService) {\n var config = runService.getCurrentConfig();\n var sessionKey = $.isFunction(options.sessionKey) ? options.sessionKey(config) : options.sessionKey;\n return sessionKey;\n}\n\nfunction setRunInSession(sessionKey, run, sessionManager) {\n if (sessionKey) {\n delete run.variables;\n sessionManager.getStore().set(sessionKey, JSON.stringify(run));\n }\n}\n\nvar defaults = {\n sessionKey: function (config) { \n var baseKey = keyNames.STRATEGY_SESSION_KEY;\n var key = ['account', 'project', 'model'].reduce(function (accum, key) {\n return config[key] ? accum + '-' + config[key] : accum; \n }, baseKey);\n return key;\n }\n};\n\nfunction RunManager(options) {\n this.options = $.extend(true, {}, defaults, options);\n\n if (this.options.run instanceof RunService) {\n this.run = this.options.run;\n } else if (!util.isEmpty(this.options.run)) {\n this.run = new RunService(this.options.run);\n } else {\n throw new Error('No run options passed to RunManager');\n }\n patchRunService(this.run, this);\n\n this.strategy = strategies.getBestStrategy(this.options);\n this.sessionManager = new SessionManager(this.options);\n}\n\nRunManager.prototype = {\n /**\n * Returns the run object for the 'correct' run. The correct run is defined by the strategy. \n *\n * For example, if the strategy is `reuse-never`, the call\n * to `getRun()` always returns a newly created run; if the strategy is `reuse-per-session`,\n * `getRun()` returns the run currently referenced in the browser cookie, and if there is none, creates a new run. \n * See [Run Manager Strategies](../strategies/) for more on strategies.\n *\n * **Example**\n *\n * rm.getRun().then(function (run) {\n * // use the run object\n * var thisRunId = run.id;\n *\n * // use the Run Service object\n * rm.run.do('runModel');\n * });\n *\n * rm.getRun(['sample_int']).then(function (run) {\n * // an object whose fields are the name : value pairs of the variables passed to getRun()\n * console.log(run.variables);\n * // the value of sample_int\n * console.log(run.variables.sample_int); \n * });\n *\n * @param {Array} variables (Optional) The run object is populated with the provided model variables, if provided. Note: `getRun()` does not throw an error if you try to get a variable which doesn't exist. Instead, the variables list is empty, and any errors are logged to the console.\n * @param {Object} options (Optional) Configuration options; passed on to [RunService#create](../run-api-service/#create) if the strategy does create a new run.\n * @return {$promise} Promise to complete the call.\n */\n getRun: function (variables, options) {\n var me = this;\n var sessionStore = this.sessionManager.getStore();\n\n var sessionContents = sessionStore.get(sessionKeyFromOptions(this.options, me.run));\n var runSession = JSON.parse(sessionContents || '{}');\n \n if (runSession.runId) {\n //EpiJS < 2.2 used runId as key, so maintain comptaibility. Remove at some future date (Summer `17?)\n runSession.id = runSession.runId;\n }\n\n var authSession = this.sessionManager.getSession();\n if (this.strategy.requiresAuth && util.isEmpty(authSession)) {\n console.error('No user-session available', this.options.strategy, 'requires authentication.');\n return $.Deferred().reject('No user-session available').promise();\n }\n return this.strategy\n .getRun(this.run, authSession, runSession, options).then(function (run) {\n if (run && run.id) {\n me.run.updateConfig({ filter: run.id });\n var sessionKey = sessionKeyFromOptions(me.options, me.run);\n setRunInSession(sessionKey, run, me.sessionManager);\n\n if (variables && variables.length) {\n return me.run.variables().query(variables).then(function (results) {\n run.variables = results;\n return run;\n }).catch(function (err) {\n run.variables = {};\n console.error(err);\n return run;\n });\n }\n }\n return run;\n });\n },\n\n /**\n * Returns the run object for a 'reset' run. The definition of a reset is defined by the strategy, but typically means forcing the creation of a new run. For example, `reset()` for the default strategies `reuse-per-session` and `reuse-last-initialized` both create new runs.\n *\n * **Example**\n *\n * rm.reset().then(function (run) {\n * // use the (new) run object\n * var thisRunId = run.id;\n *\n * // use the Run Service object\n * rm.run.do('runModel');\n * });\n *\n * **Parameters**\n * @param {Object} options (Optional) Configuration options; passed on to [RunService#create](../run-api-service/#create).\n * @return {Promise}\n */\n reset: function (options) {\n var me = this;\n var authSession = this.sessionManager.getSession();\n if (this.strategy.requiresAuth && util.isEmpty(authSession)) {\n console.error('No user-session available', this.options.strategy, 'requires authentication.');\n return $.Deferred().reject('No user-session available').promise();\n }\n return this.strategy.reset(this.run, authSession, options).then(function (run) {\n if (run && run.id) {\n me.run.updateConfig({ filter: run.id });\n var sessionKey = sessionKeyFromOptions(me.options, me.run);\n setRunInSession(sessionKey, run.id, me.sessionManager);\n }\n return run;\n });\n }\n};\n\nRunManager.strategies = strategies;\nmodule.exports = RunManager;\n","'use strict';\n\nvar Base = require('./none-strategy');\nvar classFrom = require('../../util/inherit');\n\n/**\n* ## Conditional Creation Strategy\n*\n* This strategy will try to get the run stored in the cookie and\n* evaluate if it needs to create a new run by calling the `condition` function.\n*/\n\nvar Strategy = classFrom(Base, {\n constructor: function Strategy(condition) {\n if (condition == null) { //eslint-disable-line\n throw new Error('Conditional strategy needs a condition to create a run');\n }\n this.condition = typeof condition !== 'function' ? function () { return condition; } : condition;\n },\n\n /**\n * Gets a new 'correct' run, or updates the existing one (the definition of 'correct' depends on strategy implementation).\n * @param {RunService} runService A Run Service instance for the current run, as determined by the Run Manager.\n * @param {Object} userSession Information about the current user session. See [AuthManager#getCurrentUserSessionInfo](../auth-manager/#getcurrentusersessioninfo) for format.\n * @param {Object} options (Optional) See [RunService#create](../run-api-service/#create) for supported options.\n * @return {Promise} \n */\n reset: function (runService, userSession, options) {\n var group = userSession && userSession.groupName;\n var opt = $.extend({\n scope: { group: group }\n }, runService.getCurrentConfig());\n\n return runService\n .create(opt, options)\n .then(function (run) {\n run.freshlyCreated = true;\n return run;\n });\n },\n\n /**\n * Gets the 'correct' run (the definition of 'correct' depends on strategy implementation).\n * @param {RunService} runService A Run Service instance for the current run, as determined by the Run Manager.\n * @param {Object} userSession Information about the current user session. See [AuthManager#getCurrentUserSessionInfo](../auth-manager/#getcurrentusersessioninfo) for format.\n * @param {Object} runSession The Run Manager stores the 'last accessed' run in a cookie and passes it back here.\n * @param {Object} options (Optional) See [RunService#create](../run-api-service/#create) for supported options.\n * @return {Promise} \n */\n getRun: function (runService, userSession, runSession, options) {\n var me = this;\n if (runSession && runSession.id) {\n return this.loadAndCheck(runService, userSession, runSession, options).catch(function () {\n return me.reset(runService, userSession, options); //if it got the wrong cookie for e.g.\n });\n } else {\n return this.reset(runService, userSession, options);\n }\n },\n\n loadAndCheck: function (runService, userSession, runSession, options) {\n var shouldCreate = false;\n var me = this;\n\n return runService\n .load(runSession.id, null, {\n success: function (run, msg, headers) {\n shouldCreate = me.condition(run, headers, userSession, runSession);\n }\n })\n .then(function (run) {\n if (shouldCreate) {\n return me.reset(runService, userSession, options);\n }\n return run;\n });\n }\n});\n\nmodule.exports = Strategy;\n","/**\n * The `new-if-initialized` strategy creates a new run if the current one is in memory or has its `initialized` field set to `true`. The `initialized` field in the run record is automatically set to `true` at run creation, but can be changed.\n * \n * This strategy is useful if your project is structured such that immediately after a run is created, the model is executed completely (for example, a Vensim model is stepped to the end). It is similar to the `new-if-missing` strategy, except that it checks a field of the run record.\n * \n * Specifically, the strategy is:\n *\n * * Check the `sessionKey` cookie. \n * * This cookie is set by the [Run Manager](../run-manager/) and configurable through its options.\n * * If the cookie exists, check whether the run is in memory or only persisted in the database. Additionally, check whether the run's `initialized` field is `true`. \n * * If the run is in memory, create a new run.\n * * If the run's `initialized` field is `true`, create a new run.\n * * Otherwise, use the existing run.\n * * If the cookie does not exist, create a new run for this end user.\n * \n * @deprecated Consider using `reuse-last-initialized` instead\n */\n\n'use strict';\nvar classFrom = require('../../../util/inherit');\nvar ConditionalStrategy = require('../conditional-creation-strategy');\n\nvar __super = ConditionalStrategy.prototype;\n\nvar Strategy = classFrom(ConditionalStrategy, {\n constructor: function (options) {\n __super.constructor.call(this, this.createIf, options);\n console.warn('This strategy is deprecated; all runs now default to being initialized by default making this redundant. Consider using `reuse-last-initialized` instead.');\n },\n\n createIf: function (run, headers) {\n return headers.getResponseHeader('pragma') === 'persistent' || run.initialized;\n }\n});\n\nmodule.exports = Strategy;\n","/**\n * The `new-if-persisted` strategy creates a new run when the current one becomes persisted (end user is idle for a set period), but otherwise uses the current one. \n * \n * Using this strategy means that when end users navigate between pages in your project, or refresh their browsers, they will still be working with the same run. \n * \n * However, if they are idle for longer than your project's **Model Session Timeout** (configured in your project's [Settings](../../../updating_your_settings/)), then their run is persisted; the next time they interact with the project, they will get a new run. (See more background on [Run Persistence](../../../run_persistence/).)\n * \n * This strategy is useful for multi-page projects where end users play through a simulation in one sitting, stepping through the model sequentially (for example, a Vensim model that uses the `step` operation) or calling specific functions until the model is \"complete.\" However, you will need to guarantee that your end users will remain engaged with the project from beginning to end — or at least, that if they are idle for longer than the **Model Session Timeout**, it is okay for them to start the project from scratch (with an uninitialized model). \n * \n * Specifically, the strategy is:\n *\n * * Check the `sessionKey` cookie.\n * * This cookie is set by the [Run Manager](../run-manager/) and configurable through its options.\n * * If the cookie exists, check whether the run is in memory or only persisted in the database. \n * * If the run is in memory, use the run.\n * * If the run is only persisted (and not still in memory), create a new run for this end user.\n * * If the cookie does not exist, create a new run for this end user.\n *\n * @deprecated The run-service now sets a header to automatically bring back runs into memory\n */\n\n'use strict';\nvar classFrom = require('../../../util/inherit');\nvar ConditionalStrategy = require('../conditional-creation-strategy');\n\nvar __super = ConditionalStrategy.prototype;\n\nvar Strategy = classFrom(ConditionalStrategy, {\n constructor: function (options) {\n __super.constructor.call(this, this.createIf, options);\n console.warn('This strategy is deprecated; the run-service now sets a header to automatically bring back runs into memory');\n },\n\n createIf: function (run, headers) {\n return headers.getResponseHeader('pragma') === 'persistent';\n }\n});\n\nmodule.exports = Strategy;\n","/**\n * ### Working with Run Strategies\n *\n * You can access a list of available strategies using `F.manager.RunManager.strategies.list`. You can also ask for a particular strategy by name.\n *\n * If you decide to [create your own run strategy](#create-your-own), you can register your strategy. Registering your strategy means that:\n *\n * * You can pass the strategy by name to a Run Manager (as opposed to passing the strategy function): `new F.manager.RunManager({ strategy: 'mynewname'})`.\n * * You can pass configuration options to your strategy.\n * * You can specify whether or not your strategy requires authorization (a valid user session) to work.\n */\n\n\nvar list = {\n 'conditional-creation': require('./conditional-creation-strategy'),\n 'new-if-initialized': require('./deprecated/new-if-initialized-strategy'), //deprecated\n 'new-if-persisted': require('./deprecated/new-if-persisted-strategy'), //deprecated\n\n none: require('./none-strategy'),\n\n multiplayer: require('./multiplayer-strategy'),\n 'reuse-never': require('./reuse-never'),\n 'reuse-per-session': require('./reuse-per-session'),\n 'reuse-across-sessions': require('./reuse-across-sessions'),\n 'reuse-last-initialized': require('./reuse-last-initialized'),\n};\n\n//Add back older aliases\nlist['always-new'] = list['reuse-never'];\nlist['new-if-missing'] = list['reuse-per-session'];\nlist['persistent-single-player'] = list['reuse-across-sessions'];\n\n\nmodule.exports = {\n /**\n * List of available strategies. Within this object, each key is the strategy name and the associated value is the strategy constructor.\n * @type {Object} \n */\n list: list,\n\n /**\n * Gets strategy by name.\n *\n * **Example**\n *\n * var reuseStrat = F.manager.RunManager.strategies.byName('reuse-across-sessions');\n * // shows strategy function\n * console.log('reuseStrat = ', reuseStrat);\n * // create a new run manager using this strategy\n * var rm = new F.manager.RunManager({strategy: reuseStrat, run: { model: 'model.vmf'} });\n *\n * **Parameters**\n * @param {String} strategyName Name of strategy to get.\n * @return {Function} Strategy function.\n */\n byName: function (strategyName) {\n return list[strategyName];\n },\n\n getBestStrategy: function (options) {\n var strategy = options.strategy;\n if (!strategy) {\n if (options.strategyOptions && options.strategyOptions.initOperation) {\n strategy = 'reuse-last-initialized';\n } else {\n strategy = 'reuse-per-session';\n }\n }\n\n if (strategy.getRun) {\n return strategy;\n }\n var StrategyCtor = typeof strategy === 'function' ? strategy : this.byName(strategy);\n if (!StrategyCtor) {\n throw new Error('Specified run creation strategy was invalid:', strategy);\n }\n\n var strategyInstance = new StrategyCtor(options);\n if (!strategyInstance.getRun || !strategyInstance.reset) {\n throw new Error('All strategies should implement a `getRun` and `reset` interface', options.strategy);\n }\n strategyInstance.requiresAuth = StrategyCtor.requiresAuth;\n\n return strategyInstance;\n },\n\n /**\n * Adds a new strategy.\n *\n * **Example**\n *\n * // this \"favorite run\" strategy always returns the same run, no matter what\n * // (not a useful strategy, except as an example)\n * F.manager.RunManager.strategies.register(\n * 'favRun', \n * function() { \n * return { getRun: function() { return '0000015a4cd1700209cd0a7d207f44bac289'; },\n * reset: function() { return '0000015a4cd1700209cd0a7d207f44bac289'; } \n * } \n * }, \n * { requiresAuth: true }\n * );\n * \n * var rm = new F.manager.RunManager({strategy: 'favRun', run: { model: 'model.vmf'} });\n *\n * **Parameters**\n * @param {String} name Name for strategy. This string can then be passed to a Run Manager as `new F.manager.RunManager({ strategy: 'mynewname'})`.\n * @param {Function} strategy The strategy constructor. Will be called with `new` on Run Manager initialization.\n * @param {Object} options Options for strategy.\n * @param {Boolean} options.requiresAuth Specify if the strategy requires a valid user session to work.\n */\n register: function (name, strategy, options) {\n strategy.options = options;\n list[name] = strategy;\n }\n};","/**\n * The `multiplayer` strategy is for use with [multiplayer worlds](../../../glossary/#world). It checks the current world for this end user, and always returns the current run for that world. This is equivalent to calling `getCurrentWorldForUser()` and then `getCurrentRunId()` from the [World API Adapater](../world-api-adapter/). If you use the [World Manager](../world-manager/), you are automatically using this strategy.\n * \n * Using this strategy means that end users in projects with multiplayer worlds always see the most current world and run. This ensures that they are in sync with the other end users sharing their world and run. In turn, this allows for competitive or collaborative multiplayer projects.\n */\n'use strict';\n\nvar classFrom = require('../../util/inherit');\n\nvar IdentityStrategy = require('./none-strategy');\nvar WorldApiAdapter = require('../../service/world-api-adapter');\n\nvar defaults = {};\n\nvar Strategy = classFrom(IdentityStrategy, {\n constructor: function (options) {\n this.options = $.extend(true, {}, defaults, options);\n this.worldApi = new WorldApiAdapter(this.options.run);\n },\n\n reset: function (runService, session, options) {\n var curUserId = session.userId;\n var curGroupName = session.groupName;\n\n return this.worldApi\n .getCurrentWorldForUser(curUserId, curGroupName)\n .then(function (world) {\n return this.worldApi.newRunForWorld(world.id, options).then(function (runid) {\n return {\n id: runid\n };\n });\n }.bind(this));\n },\n\n getRun: function (runService, session) {\n var curUserId = session.userId;\n var curGroupName = session.groupName;\n var worldApi = this.worldApi;\n var model = this.options.model;\n var me = this;\n var dtd = $.Deferred();\n\n if (!curUserId) {\n return dtd.reject({ statusCode: 400, error: 'We need an authenticated user to join a multiplayer world. (ERR: no userId in session)' }, session).promise();\n }\n\n var loadRunFromWorld = function (world) {\n if (!world) {\n return dtd.reject({ statusCode: 404, error: 'The user is not in any world.' }, { options: me.options, session: session });\n }\n return worldApi.getCurrentRunId({ model: model, filter: world.id })\n .then(function (id) {\n return runService.load(id);\n })\n .then(dtd.resolve)\n .fail(dtd.reject);\n };\n\n var serverError = function (error) {\n // is this possible?\n return dtd.reject(error, session, me.options);\n };\n\n this.worldApi\n .getCurrentWorldForUser(curUserId, curGroupName)\n .then(loadRunFromWorld)\n .fail(serverError);\n\n return dtd.promise();\n },\n\n});\n\nmodule.exports = Strategy;\n","/**\n * The `none` strategy never returns a run or tries to create a new run. It simply returns the contents of the current [Run Service instance](../run-api-service/).\n * \n * This strategy is useful if you want to manually decide how to create your own runs and don't want any automatic assistance.\n */\n\n'use strict';\n\nvar classFrom = require('../../util/inherit');\nvar Base = {};\n\n// Interface that all strategies need to implement\nmodule.exports = classFrom(Base, {\n constructor: function (options) {\n\n },\n\n reset: function () {\n // return a newly created run\n return $.Deferred().resolve().promise();\n },\n\n getRun: function (runService) {\n // return a usable run\n return $.Deferred().resolve(runService).promise();\n }\n});\n","/**\n * The `reuse-across-sessions` strategy returns the latest (most recent) run for this user, whether it is in memory or not. If there are no runs for this user, it creates a new one.\n *\n * This strategy is useful if end users are using your project for an extended period of time, possibly over several sessions. This is most common in cases where a user of your project executes the model step by step (as opposed to a project where the model is executed completely, for example, a Vensim model that is immediately stepped to the end).\n *\n * Specifically, the strategy is:\n * \n * * Check if there are any runs for this end user.\n * * If there are no runs (either in memory or in the database), create a new one.\n * * If there are runs, take the latest (most recent) one.\n *\n * @name persistent-single-player\n */\n\n'use strict';\n\nvar classFrom = require('../../util/inherit');\nvar IdentityStrategy = require('./none-strategy');\nvar injectFiltersFromSession = require('../strategy-utils').injectFiltersFromSession;\nvar injectScopeFromSession = require('../strategy-utils').injectScopeFromSession;\n\nvar defaults = {\n /**\n * (Optional) Additional criteria to use while selecting the last run\n * @type {Object}\n */\n filter: {},\n};\n\nvar Strategy = classFrom(IdentityStrategy, {\n constructor: function Strategy(options) {\n var strategyOptions = options ? options.strategyOptions : {};\n this.options = $.extend(true, {}, defaults, strategyOptions);\n },\n\n reset: function (runService, userSession, options) {\n var opt = injectScopeFromSession(runService.getCurrentConfig(), userSession);\n return runService\n .create(opt, options)\n .then(function (run) {\n run.freshlyCreated = true;\n return run;\n });\n },\n\n getRun: function (runService, userSession, runSession, options) {\n var filter = injectFiltersFromSession(this.options.filter, userSession);\n var me = this;\n return runService.query(filter, { \n startrecord: 0,\n endrecord: 0,\n sort: 'created', \n direction: 'desc'\n }).then(function (runs) {\n if (!runs.length) {\n return me.reset(runService, userSession, options);\n }\n return runs[0];\n });\n }\n});\n\nmodule.exports = Strategy;\n","/**\n * The `reuse-last-initialized` strategy looks for the most recent run that matches particular criteria; if it cannot find one, it creates a new run and immediately executes a set of \"initialization\" operations. \n *\n * This strategy is useful if you have a time-based model and always want the run you're operating on to start at a particular step. For example:\n *\n * var rm = new F.manager.RunManager({\n * strategy: 'reuse-last-initialized',\n * strategyOptions: {\n * initOperation: [{ step: 10 }]\n * }\n * });\n * \n * This strategy is also useful if you have a custom initialization function in your model, and want to make sure it's always executed for new runs.\n *\n * Specifically, the strategy is:\n *\n * * Look for the most recent run that matches the (optional) `flag` criteria\n * * If there are no runs that match the `flag` criteria, create a new run. Immediately \"initialize\" this new run by:\n * * Calling the model operation(s) specified in the `initOperation` array.\n * * Optionally, setting a `flag` in the run.\n *\n */\n\n'use strict';\nvar classFrom = require('../../util/inherit');\nvar injectFiltersFromSession = require('../strategy-utils').injectFiltersFromSession;\nvar injectScopeFromSession = require('../strategy-utils').injectScopeFromSession;\n\nvar Base = {};\n\nvar defaults = {\n /**\n * Operations to execute in the model for initialization to be considered complete.\n * @type {Array} Can be in any of the formats [Run Service's `serial()`](../run-api-service/#serial) supports.\n */\n initOperation: [],\n\n /**\n * (Optional) Flag to set in run after initialization operations are run. You typically would not override this unless you needed to set additional properties as well.\n * @type {Object}\n */\n flag: null,\n};\nmodule.exports = classFrom(Base, {\n constructor: function (options) {\n var strategyOptions = options ? options.strategyOptions : {};\n this.options = $.extend(true, {}, defaults, strategyOptions);\n if (!this.options.initOperation || !this.options.initOperation.length) {\n throw new Error('Specifying an init function is required for this strategy');\n }\n if (!this.options.flag) {\n this.options.flag = {\n isInitComplete: true\n };\n }\n },\n\n reset: function (runService, userSession, options) {\n var opt = injectScopeFromSession(runService.getCurrentConfig(), userSession);\n var me = this;\n return runService.create(opt, options).then(function (createResponse) {\n return runService.serial([].concat(me.options.initOperation)).then(function () {\n return createResponse;\n });\n }).then(function (createResponse) {\n return runService.save(me.options.flag).then(function (patchResponse) {\n return $.extend(true, {}, createResponse, patchResponse);\n });\n });\n },\n\n getRun: function (runService, userSession, runSession, options) {\n var sessionFilter = injectFiltersFromSession(this.options.flag, userSession);\n var runopts = runService.getCurrentConfig();\n var filter = $.extend(true, { trashed: false }, sessionFilter, { model: runopts.model });\n var me = this;\n return runService.query(filter, { \n startrecord: 0,\n endrecord: 0,\n sort: 'created', \n direction: 'desc'\n }).then(function (runs) {\n if (!runs.length) {\n return me.reset(runService, userSession, options);\n }\n return runs[0];\n });\n }\n});","/**\n * The `reuse-never` strategy always creates a new run for this end user irrespective of current state. This is equivalent to calling `F.service.Run.create()` from the [Run Service](../run-api-service/) every time. \n * \n * This strategy means that every time your end users refresh their browsers, they get a new run. \n * \n * This strategy can be useful for basic, single-page projects. This strategy is also useful for prototyping or project development: it creates a new run each time you refresh the page, and you can easily check the outputs of the model. However, typically you will use one of the other strategies for a production project.\n *\n * @name always-new\n */\n\n'use strict';\n\nvar classFrom = require('../../util/inherit');\nvar ConditionalStrategy = require('./conditional-creation-strategy');\n\nvar __super = ConditionalStrategy.prototype;\n\nvar Strategy = classFrom(ConditionalStrategy, {\n constructor: function (options) {\n __super.constructor.call(this, this.createIf, options);\n },\n\n createIf: function (run, headers) {\n // always create a new run!\n return true;\n }\n});\n\nmodule.exports = Strategy;\n","/**\n * The `reuse-per-session` strategy creates a new run when the current one is not in the browser cookie.\n * \n * Using this strategy means that when end users navigate between pages in your project, or refresh their browsers, they will still be working with the same run. However, if end users log out and return to the project at a later date, a new run is created.\n *\n * This strategy is useful if your project is structured such that immediately after a run is created, the model is executed completely (for example, a Vensim model that is stepped to the end as soon as it is created). In contrast, if end users play with your project for an extended period of time, executing the model step by step, the `reuse-across-sessions` strategy is probably a better choice (it allows end users to pick up where they left off, rather than starting from scratch each browser session).\n * \n * Specifically, the strategy is:\n *\n * * Check the `sessionKey` cookie.\n * * This cookie is set by the [Run Manager](../run-manager/) and configurable through its options. \n * * If the cookie exists, use the run id stored there. \n * * If the cookie does not exist, create a new run for this end user.\n *\n * @name new-if-missing\n */\n\n'use strict';\n\nvar classFrom = require('../../util/inherit');\nvar ConditionalStrategy = require('./conditional-creation-strategy');\n\nvar __super = ConditionalStrategy.prototype;\n\n/*\n* create a new run only if nothing is stored in the cookie\n* this is useful for baseRuns.\n*/\nvar Strategy = classFrom(ConditionalStrategy, {\n constructor: function (options) {\n __super.constructor.call(this, this.createIf, options);\n },\n\n createIf: function (run, headers) {\n // if we are here, it means that the run exists... so we don't need a new one\n return false;\n }\n});\n\nmodule.exports = Strategy;\n","/**\n * ## Saved Runs Manager\n *\n * The Saved Runs Manager is a specific type of [Run Manager](../../run-manager/) which provides access to a list of runs (rather than just one run). It also provides utility functions for dealing with multiple runs (e.g. saving, deleting, listing).\n *\n * An instance of a Saved Runs Manager is included automatically in every instance of a [Scenario Manager](../), and is accessible from the Scenario Manager at `.savedRuns`. See [more information](../#properties) on using `.savedRuns` within the Scenario Manager.\n *\n */\n'use strict';\n\nvar RunService = require('../service/run-api-service');\nvar SessionManager = require('../store/session-manager');\n\nvar injectFiltersFromSession = require('./strategy-utils').injectFiltersFromSession;\n\nvar SavedRunsManager = function (config) {\n var defaults = {\n /**\n * If set, will only pull runs from current group. Defaults to `true`.\n * @type {Boolean}\n */\n scopeByGroup: true,\n\n /**\n * If set, will only pull runs from current user. Defaults to `true`.\n *\n * For multiplayer run comparison projects, set this to false so that all end users in a group can view the shared set of saved runs.\n * @type {Boolean}\n */\n scopeByUser: true,\n };\n\n this.sessionManager = new SessionManager();\n\n var options = $.extend(true, {}, defaults, config);\n if (options.run) {\n if (options.run instanceof RunService) {\n this.runService = options.run;\n } else {\n this.runService = new RunService(options.run);\n }\n this.options = options;\n } else {\n throw new Error('No run options passed to SavedRunsManager');\n }\n};\n\nSavedRunsManager.prototype = {\n /**\n * Marks a run as saved. \n *\n * Note that while any run can be saved, only runs which also match the configuration options `scopeByGroup` and `scopeByUser` are returned by the `getRuns()` method.\n *\n * **Example**\n *\n * var sm = new F.manager.ScenarioManager();\n * sm.savedRuns.save('0000015a4cd1700209cd0a7d207f44bac289');\n *\n * @param {String|RunService} run Run to save. Pass in either the run id, as a string, or the [Run Service](../../run-api-service/).\n * @param {Object} otherFields (Optional) Any other meta-data to save with the run.\n * @return {Promise}\n */\n save: function (run, otherFields) {\n var param = $.extend(true, {}, otherFields, { saved: true, trashed: false });\n return this.mark(run, param);\n },\n /**\n * Marks a run as removed; the inverse of marking as saved.\n *\n * **Example**\n *\n * var sm = new F.manager.ScenarioManager();\n * sm.savedRuns.remove('0000015a4cd1700209cd0a7d207f44bac289');\n *\n * @param {String|RunService} run Run to remove. Pass in either the run id, as a string, or the [Run Service](../../run-api-service/).\n * @param {Object} otherFields (Optional) any other meta-data to save with the run.\n * @return {Promise}\n */\n remove: function (run, otherFields) {\n var param = $.extend(true, {}, otherFields, { saved: false, trashed: true });\n return this.mark(run, param);\n },\n\n\n /**\n * Sets additional fields on a run. This is a convenience method for [RunService#save](../../run-api-service/#save).\n *\n * **Example**\n *\n * var sm = new F.manager.ScenarioManager();\n * sm.savedRuns.mark('0000015a4cd1700209cd0a7d207f44bac289', \n * { 'myRunName': 'sample policy decisions' });\n *\n * @param {String|RunService} run Run to operate on. Pass in either the run id, as a string, or the [Run Service](../../run-api-service/).\n * @param {Object} toMark Fields to set, as name : value pairs.\n * @return {Promise}\n */\n mark: function (run, toMark) {\n var rs;\n var existingOptions = this.runService.getCurrentConfig();\n if (run instanceof RunService) {\n rs = run;\n } else if (run && (typeof run === 'string')) {\n rs = new RunService($.extend(true, {}, existingOptions, { id: run, autoRestore: false }));\n } else if ($.isArray(run)) {\n var me = this;\n var proms = run.map(function (r) {\n return me.mark(r, toMark);\n });\n return $.when.apply(null, proms);\n } else {\n throw new Error('Invalid run object provided');\n }\n return rs.save(toMark);\n },\n\n /**\n * Returns a list of saved runs.\n *\n * **Example**\n *\n * var sm = new F.manager.ScenarioManager();\n * sm.savedRuns.getRuns().then(function (runs) {\n * for (var i=0; i\n * //\n * $('#fileupload').on('change', function (e) {\n * var file = e.target.files[0];\n * var data = new FormData();\n * data.append('file', file, file.name);\n * fa.create(file.name, data);\n * });\n */\n\n'use strict';\n\nvar ConfigService = require('./configuration-service');\nvar TransportFactory = require('../transport/http-transport-factory');\nvar SessionManager = require('../store/session-manager');\n\nmodule.exports = function (config) {\n var defaults = {\n /**\n * For projects that require authentication, pass in the user access token (defaults to empty string). If the user is already logged in to Epicenter, the user access token is already set in a cookie and automatically loaded from there. (See [more background on access tokens](../../../project_access/)).\n * @see [Authentication API Service](../auth-api-service/) for getting tokens.\n * @type {String}\n */\n token: undefined,\n\n /**\n * The account id. In the Epicenter UI, this is the **Team ID** (for team projects) or **User ID** (for personal projects). Defaults to undefined.\n * @type {String}\n */\n account: undefined,\n\n /**\n * The project id. Defaults to undefined.\n * @type {String}\n */\n project: undefined,\n\n /**\n * The folder type. One of `model` | `static` | `node`.\n * @type {String}\n */\n folderType: 'static',\n\n\n /**\n * Options to pass on to the underlying transport layer. All jquery.ajax options at http://api.jquery.com/jQuery.ajax/ are available. Defaults to empty object.\n * @type {Object}\n */\n transport: {}\n };\n\n this.sessionManager = new SessionManager();\n var serviceOptions = this.sessionManager.getMergedOptions(defaults, config);\n var urlConfig = new ConfigService(serviceOptions).get('server');\n if (serviceOptions.account) {\n urlConfig.accountPath = serviceOptions.account;\n }\n if (serviceOptions.project) {\n urlConfig.projectPath = serviceOptions.project;\n }\n\n var httpOptions = $.extend(true, {}, serviceOptions.transport, {\n url: urlConfig.getAPIPath('file')\n });\n\n if (serviceOptions.token) {\n httpOptions.headers = {\n Authorization: 'Bearer ' + serviceOptions.token\n };\n }\n var http = new TransportFactory(httpOptions);\n\n function uploadBody(fileName, contents) {\n var boundary = '---------------------------7da24f2e50046';\n\n return {\n body: '--' + boundary + '\\r\\n' +\n 'Content-Disposition: form-data; name=\"file\";' +\n 'filename=\"' + fileName + '\"\\r\\n' +\n 'Content-type: text/html\\r\\n\\r\\n' +\n contents + '\\r\\n' +\n '--' + boundary + '--',\n boundary: boundary\n };\n }\n\n function uploadFileOptions(filePath, contents, options) {\n filePath = filePath.split('/');\n var fileName = filePath.pop();\n filePath = filePath.join('/');\n var path = serviceOptions.folderType + '/' + filePath;\n\n var extraParams = {};\n if (contents instanceof FormData) {\n extraParams = {\n data: contents,\n processData: false,\n contentType: false,\n };\n } else {\n var upload = uploadBody(fileName, contents);\n extraParams = {\n data: upload.body,\n contentType: 'multipart/form-data; boundary=' + upload.boundary\n };\n }\n\n return $.extend(true, {}, serviceOptions, options, {\n url: urlConfig.getAPIPath('file') + path,\n }, extraParams);\n }\n\n var publicAsyncAPI = {\n /**\n * Get a directory listing, or contents of a file.\n * @param {String} filePath Path to the file\n * @param {Object} options (Optional) Overrides for configuration options.\n * @return {Promise}\n */\n getContents: function (filePath, options) {\n var path = serviceOptions.folderType + '/' + filePath;\n var httpOptions = $.extend(true, {}, serviceOptions, options, {\n url: urlConfig.getAPIPath('file') + path\n });\n return http.get('', httpOptions);\n },\n\n /**\n * Replaces the file at the given file path.\n * @param {String} filePath Path to the file\n * @param {String | FormData } contents Contents to write to file\n * @param {Object} options (Optional) Overrides for configuration options\n * @return {Promise}\n */\n replace: function (filePath, contents, options) {\n var httpOptions = uploadFileOptions(filePath, contents, options);\n return http.put(httpOptions.data, httpOptions);\n },\n\n /**\n * Creates a file in the given file path.\n * @param {String} filePath Path to the file\n * @param {String | FormData } contents Contents to write to file\n * @param {Boolean} replaceExisting Replace file if it already exists; defaults to false\n * @param {Object} options (Optional) Overrides for configuration options\n * @return {Promise}\n */\n create: function (filePath, contents, replaceExisting, options) {\n var httpOptions = uploadFileOptions(filePath, contents, options);\n var prom = http.post(httpOptions.data, httpOptions);\n var me = this;\n if (replaceExisting === true) {\n prom = prom.then(null, function (xhr) {\n var conflictStatus = 409;\n if (xhr.status === conflictStatus) {\n return me.replace(filePath, contents, options);\n }\n });\n }\n return prom;\n },\n\n /**\n * Removes the file.\n * @param {String} filePath Path to the file\n * @param {Object} options (Optional) Overrides for configuration options\n * @return {Promise}\n */\n remove: function (filePath, options) {\n var path = serviceOptions.folderType + '/' + filePath;\n var httpOptions = $.extend(true, {}, serviceOptions, options, {\n url: urlConfig.getAPIPath('file') + path\n });\n return http.delete(null, httpOptions);\n },\n\n /**\n * Renames the file.\n * @param {String} filePath Path to the file\n * @param {String} newName New name of file\n * @param {Object} options (Optional) Overrides for configuration options\n * @return {Promise}\n */\n rename: function (filePath, newName, options) {\n var path = serviceOptions.folderType + '/' + filePath;\n var httpOptions = $.extend(true, {}, serviceOptions, options, {\n url: urlConfig.getAPIPath('file') + path\n });\n return http.patch({ name: newName }, httpOptions);\n }\n };\n\n $.extend(this, publicAsyncAPI);\n};\n","/**\n * ## Asset API Adapter\n *\n * The Asset API Adapter allows you to store assets -- resources or files of any kind -- used by a project with a scope that is specific to project, group, or end user.\n *\n * Assets are used with [team projects](../../../project_admin/#team). One common use case is having end users in a [group](../../../glossary/#groups) or in a [multiplayer world](../../../glossary/#world) upload data -- videos created during game play, profile pictures for customizing their experience, etc. -- as part of playing through the project.\n *\n * Resources created using the Asset Adapter are scoped:\n *\n * * Project assets are writable only by [team members](../../../glossary/#team), that is, Epicenter authors.\n * * Group assets are writable by anyone with access to the project that is part of that particular [group](../../../glossary/#groups). This includes all [team members](../../../glossary/#team) (Epicenter authors) and any [end users](../../../glossary/#users) who are members of the group -- both facilitators and standard end users.\n * * User assets are writable by the specific end user, and by the facilitator of the group.\n * * All assets are readable by anyone with the exact URI.\n *\n * To use the Asset Adapter, instantiate it and then access the methods provided. Instantiating requires the account id (**Team ID** in the Epicenter user interface) and project id (**Project ID**). The group name is required for assets with a group scope, and the group name and userId are required for assets with a user scope. If not included, they are taken from the logged in user's session information if needed.\n *\n * When creating an asset, you can pass in text (encoded data) to the `create()` call. Alternatively, you can make the `create()` call as part of an HTML form and pass in a file uploaded via the form.\n *\n * // instantiate the Asset Adapter\n * var aa = new F.service.Asset({\n * account: 'acme-simulations',\n * project: 'supply-chain-game',\n * group: 'team1',\n * userId: '12345'\n * });\n *\n * // create a new asset using encoded text\n * aa.create('test.txt', {\n * encoding: 'BASE_64',\n * data: 'VGhpcyBpcyBhIHRlc3QgZmlsZS4=',\n * contentType: 'text/plain'\n * }, { scope: 'user' });\n *\n * // alternatively, create a new asset using a file uploaded through a form\n * // this sample code goes with an html form that looks like this:\n * //\n * //
\n * // \n * // \n * // \n * //
\n * //\n * $('#upload-file').on('submit', function (e) {\n * e.preventDefault();\n * var filename = $('#filename').val();\n * var data = new FormData();\n * var inputControl = $('#file')[0];\n * data.append('file', inputControl.files[0], filename);\n *\n * aa.create(filename, data, { scope: 'user' });\n * });\n *\n */\n\n'use strict';\n\nvar ConfigService = require('./configuration-service');\nvar TransportFactory = require('../transport/http-transport-factory');\nvar _pick = require('../util/object-util')._pick;\nvar SessionManager = require('../store/session-manager');\n\nvar apiEndpoint = 'asset';\n\nmodule.exports = function (config) {\n var defaults = {\n /**\n * For projects that require authentication, pass in the user access token (defaults to empty string). If the user is already logged in to Epicenter, the user access token is already set in a cookie and automatically loaded from there. (See [more background on access tokens](../../../project_access/)).\n * @see [Authentication API Service](../auth-api-service/) for getting tokens.\n * @type {String}\n */\n token: undefined,\n /**\n * The account id. In the Epicenter UI, this is the **Team ID** (for team projects). If left undefined, taken from the URL.\n * @type {String}\n */\n account: undefined,\n /**\n * The project id. If left undefined, taken from the URL.\n * @type {String}\n */\n project: undefined,\n /**\n * The group name. Defaults to session's `groupName`.\n * @type {String}\n */\n group: undefined,\n /**\n * The user id. Defaults to session's `userId`.\n * @type {String}\n */\n userId: undefined,\n /**\n * The scope for the asset. Valid values are: `user`, `group`, and `project`. See above for the required permissions to write to each scope. Defaults to `user`, meaning the current end user or a facilitator in the end user's group can edit the asset.\n * @type {String}\n */\n scope: 'user',\n /**\n * Determines if a request to list the assets in a scope includes the complete URL for each asset (`true`), or only the file names of the assets (`false`). Defaults to `true`.\n * @type {boolean}\n */\n fullUrl: true,\n /**\n * The transport object contains the options passed to the XHR request.\n * @type {object}\n */\n transport: {\n processData: false\n }\n };\n this.sessionManager = new SessionManager();\n var serviceOptions = this.sessionManager.getMergedOptions(defaults, config);\n var urlConfig = new ConfigService(serviceOptions).get('server');\n\n if (!serviceOptions.account) {\n serviceOptions.account = urlConfig.accountPath;\n }\n\n if (!serviceOptions.project) {\n serviceOptions.project = urlConfig.projectPath;\n }\n\n var transportOptions = $.extend(true, {}, serviceOptions.transport, {\n url: urlConfig.getAPIPath(apiEndpoint)\n });\n\n if (serviceOptions.token) {\n transportOptions.headers = {\n Authorization: 'Bearer ' + serviceOptions.token\n };\n }\n\n var http = new TransportFactory(transportOptions);\n\n var assetApiParams = ['encoding', 'data', 'contentType'];\n var scopeConfig = {\n user: ['scope', 'account', 'project', 'group', 'userId'],\n group: ['scope', 'account', 'project', 'group'],\n project: ['scope', 'account', 'project'],\n };\n\n var validateFilename = function (filename) {\n if (!filename) {\n throw new Error('filename is needed.');\n }\n };\n\n var validateUrlParams = function (options) {\n var partKeys = scopeConfig[options.scope];\n if (!partKeys) {\n throw new Error('scope parameter is needed.');\n }\n\n $.each(partKeys, function () {\n if (!options[this]) {\n throw new Error(this + ' parameter is needed.');\n }\n });\n };\n\n var buildUrl = function (filename, options) {\n validateUrlParams(options);\n var partKeys = scopeConfig[options.scope];\n var parts = $.map(partKeys, function (key) {\n return options[key];\n });\n if (filename) {\n // This prevents adding a trailing / in the URL as the Asset API\n // does not work correctly with it\n filename = '/' + filename;\n }\n return urlConfig.getAPIPath(apiEndpoint) + parts.join('/') + filename;\n };\n\n // Private function, all requests follow a more or less same approach to\n // use the Asset API and the difference is the HTTP verb\n //\n // @param {string} method` (Required) HTTP verb\n // @param {string} filename` (Required) Name of the file to delete/replace/create\n // @param {object} params` (Optional) Body parameters to send to the Asset API\n // @param {object} options` (Optional) Options object to override global options.\n var upload = function (method, filename, params, options) {\n validateFilename(filename);\n // make sure the parameter is clean\n method = method.toLowerCase();\n var contentType = params instanceof FormData === true ? false : 'application/json';\n if (contentType === 'application/json') {\n // whitelist the fields that we actually can send to the api\n params = _pick(params, assetApiParams);\n } else { // else we're sending form data which goes directly in request body\n // For multipart/form-data uploads the filename is not set in the URL,\n // it's getting picked by the FormData field filename.\n filename = method === 'post' || method === 'put' ? '' : filename;\n }\n var urlOptions = $.extend({}, serviceOptions, options);\n var url = buildUrl(filename, urlOptions);\n var createOptions = $.extend(true, {}, urlOptions, { url: url, contentType: contentType });\n\n return http[method](params, createOptions);\n };\n\n var publicAPI = {\n /**\n * Creates a file in the Asset API. The server returns an error (status code `409`, conflict) if the file already exists, so\n * check first with a `list()` or a `get()`.\n *\n * **Example**\n *\n * var aa = new F.service.Asset({\n * account: 'acme-simulations',\n * project: 'supply-chain-game',\n * group: 'team1',\n * userId: ''\n * });\n *\n * // create a new asset using encoded text\n * aa.create('test.txt', {\n * encoding: 'BASE_64',\n * data: 'VGhpcyBpcyBhIHRlc3QgZmlsZS4=',\n * contentType: 'text/plain'\n * }, { scope: 'user' });\n *\n * // alternatively, create a new asset using a file uploaded through a form\n * // this sample code goes with an html form that looks like this:\n * //\n * //
\n * // \n * // \n * // \n * //
\n * //\n * $('#upload-file').on('submit', function (e) {\n * e.preventDefault();\n * var filename = $('#filename').val();\n * var data = new FormData();\n * var inputControl = $('#file')[0];\n * data.append('file', inputControl.files[0], filename);\n *\n * aa.create(filename, data, { scope: 'user' });\n * });\n *\n *\n * **Parameters**\n * @param {string} filename (Required) Name of the file to create.\n * @param {object} params (Optional) Body parameters to send to the Asset API. Required if the `options.transport.contentType` is `application/json`, otherwise ignored.\n * @param {string} params.encoding Either `HEX` or `BASE_64`. Required if `options.transport.contentType` is `application/json`.\n * @param {string} params.data The encoded data for the file. Required if `options.transport.contentType` is `application/json`.\n * @param {string} params.contentType The mime type of the file. Optional.\n * @param {object} options (Optional) Options object to override global options.\n * @return {Promise}\n */\n create: function (filename, params, options) {\n return upload('post', filename, params, options);\n },\n\n /**\n * Gets a file from the Asset API, fetching the asset content. (To get a list\n * of the assets in a scope, use `list()`.)\n *\n * **Parameters**\n * @param {string} filename (Required) Name of the file to retrieve.\n * @param {object} options (Optional) Options object to override global options.\n * @return {Promise}\n */\n get: function (filename, options) {\n var getServiceOptions = _pick(serviceOptions, ['scope', 'account', 'project', 'group', 'userId']);\n var urlOptions = $.extend({}, getServiceOptions, options);\n var url = buildUrl(filename, urlOptions);\n var getOptions = $.extend(true, {}, urlOptions, { url: url });\n\n return http.get({}, getOptions);\n },\n\n /**\n * Gets the list of the assets in a scope.\n *\n * **Example**\n *\n * aa.list({ fullUrl: true }).then(function(fileList){\n * console.log('array of files = ', fileList);\n * });\n *\n * **Parameters**\n * @param {object} options (Optional) Options object to override global options.\n * @param {string} options.scope (Optional) The scope (`user`, `group`, `project`).\n * @param {boolean} options.fullUrl (Optional) Determines if the list of assets in a scope includes the complete URL for each asset (`true`), or only the file names of the assets (`false`).\n * @return {Promise}\n */\n list: function (options) {\n var dtd = $.Deferred();\n var me = this;\n var urlOptions = $.extend({}, serviceOptions, options);\n var url = buildUrl('', urlOptions);\n var getOptions = $.extend(true, {}, urlOptions, { url: url });\n var fullUrl = getOptions.fullUrl;\n\n if (!fullUrl) {\n return http.get({}, getOptions);\n }\n\n http.get({}, getOptions)\n .then(function (files) {\n var fullPathFiles = $.map(files, function (file) {\n return buildUrl(file, urlOptions);\n });\n dtd.resolveWith(me, [fullPathFiles]);\n })\n .fail(dtd.reject);\n\n return dtd.promise();\n },\n\n /**\n * Replaces an existing file in the Asset API.\n *\n * **Example**\n *\n * // replace an asset using encoded text\n * aa.replace('test.txt', {\n * encoding: 'BASE_64',\n * data: 'VGhpcyBpcyBhIHNlY29uZCB0ZXN0IGZpbGUu',\n * contentType: 'text/plain'\n * }, { scope: 'user' });\n *\n * // alternatively, replace an asset using a file uploaded through a form\n * // this sample code goes with an html form that looks like this:\n * //\n * //
\n * // \n * // \n * // \n * //
\n * //\n * $('#replace-file').on('submit', function (e) {\n * e.preventDefault();\n * var filename = $('#replace-filename').val();\n * var data = new FormData();\n * var inputControl = $('#file')[0];\n * data.append('file', inputControl.files[0], filename);\n *\n * aa.replace(filename, data, { scope: 'user' });\n * });\n *\n * **Parameters**\n * @param {string} filename (Required) Name of the file being replaced.\n * @param {object} params (Optional) Body parameters to send to the Asset API. Required if the `options.transport.contentType` is `application/json`, otherwise ignored.\n * @param {string} params.encoding Either `HEX` or `BASE_64`. Required if `options.transport.contentType` is `application/json`.\n * @param {string} params.data The encoded data for the file. Required if `options.transport.contentType` is `application/json`.\n * @param {string} params.contentType The mime type of the file. Optional.\n * @param {object} options (Optional) Options object to override global options.\n * @return {Promise}\n */\n replace: function (filename, params, options) {\n return upload('put', filename, params, options);\n },\n\n /**\n * Deletes a file from the Asset API.\n *\n * **Example**\n *\n * aa.delete(sampleFileName);\n *\n * **Parameters**\n * @param {string} filename (Required) Name of the file to delete.\n * @param {object} options (Optional) Options object to override global options.\n * @return {Promise}\n */\n delete: function (filename, options) {\n return upload('delete', filename, {}, options);\n },\n\n assetUrl: function (filename, options) {\n var urlOptions = $.extend({}, serviceOptions, options);\n return buildUrl(filename, urlOptions);\n }\n };\n $.extend(this, publicAPI);\n};\n","/**\n *\n * ## Authentication API Service\n *\n * The Authentication API Service provides a method for logging in, which creates and returns a user access token.\n *\n * User access tokens are required for each call to Epicenter. (See [Project Access](../../../project_access/) for more information.)\n *\n * If you need additional functionality -- such as tracking session information, easily retrieving the user token, or getting the groups to which an end user belongs -- consider using the [Authorization Manager](../auth-manager/) instead.\n *\n * var auth = new F.service.Auth();\n * auth.login({ userName: 'jsmith@acmesimulations.com',\n * password: 'passw0rd' });\n */\n\n'use strict';\n\nvar ConfigService = require('./configuration-service');\nvar TransportFactory = require('../transport/http-transport-factory');\n\nmodule.exports = function (config) {\n var defaults = {\n /**\n * Email or username to use for logging in. Defaults to empty string.\n * @type {String}\n */\n userName: '',\n\n /**\n * Password for specified `userName`. Defaults to empty string.\n * @type {String}\n */\n password: '',\n\n /**\n * The account id for this `userName`. In the Epicenter UI, this is the **Team ID** (for team projects) or the **User ID** (for personal projects). Required if the `userName` is for an [end user](../../../glossary/#users). Defaults to empty string.\n * @type {String}\n */\n account: '',\n\n /**\n * Options to pass on to the underlying transport layer. All jquery.ajax options at http://api.jquery.com/jQuery.ajax/ are available. Defaults to empty object.\n * @type {Object}\n */\n transport: {}\n };\n var serviceOptions = $.extend({}, defaults, config);\n var urlConfig = new ConfigService(serviceOptions).get('server');\n\n var transportOptions = $.extend(true, {}, serviceOptions.transport, {\n url: urlConfig.getAPIPath('authentication')\n });\n var http = new TransportFactory(transportOptions);\n\n var publicAPI = {\n\n /**\n * Logs user in, returning the user access token.\n *\n * If no `userName` or `password` were provided in the initial configuration options, they are required in the `options` here. If no `account` was provided in the initial configuration options and the `userName` is for an [end user](../../../glossary/#users), the `account` is required as well.\n *\n * **Example**\n *\n * auth.login({\n * userName: 'jsmith',\n * password: 'passw0rd',\n * account: 'acme-simulations' })\n * .then(function (token) {\n * console.log(\"user access token is: \", token.access_token);\n * });\n *\n * **Parameters**\n * @param {Object} options (Optional) Overrides for configuration options.\n * @return {Promise}\n */\n login: function (options) {\n var httpOptions = $.extend(true, { success: $.noop }, serviceOptions, options);\n if (!httpOptions.userName || !httpOptions.password) {\n var resp = { status: 401, statusMessage: 'No username or password specified.' };\n if (options.error) {\n options.error.call(this, resp);\n }\n\n return $.Deferred().reject(resp).promise();\n }\n\n var postParams = {\n userName: httpOptions.userName,\n password: httpOptions.password,\n };\n if (httpOptions.account) {\n //pass in null for account under options if you don't want it to be sent\n postParams.account = httpOptions.account;\n }\n\n return http.post(postParams, httpOptions);\n },\n\n // (replace with /* */ comment block, to make visible in docs, once this is more than a noop)\n //\n // Logs user out from specified accounts.\n //\n // Epicenter logout is not implemented yet, so for now this is a dummy promise that gets automatically resolved.\n //\n // **Example**\n //\n // auth.logout();\n //\n // **Parameters**\n // @param {Object} `options` (Optional) Overrides for configuration options.\n //\n logout: function (options) {\n var dtd = $.Deferred();\n dtd.resolve();\n return dtd.promise();\n }\n };\n\n $.extend(this, publicAPI);\n};\n","/**\n * ## Channel Service\n *\n * The Epicenter platform provides a push channel, which allows you to publish and subscribe to messages within a [project](../../../glossary/#projects), [group](../../../glossary/#groups), or [multiplayer world](../../../glossary/#world). There are two main use cases for the channel: event notifications and chat messages.\n *\n * If you are developing with Epicenter.js, you should use the [Epicenter Channel Manager](../epicenter-channel-manager/) directly. The Epicenter Channel Manager documentation also has more [background](../epicenter-channel-manager/#background) information on channels and their use.\n *\n * The Channel Service is a building block for this functionality. It creates a publish-subscribe object, allowing you to publish messages, subscribe to messages, or unsubscribe from messages for a given 'topic' on a `$.cometd` transport instance.\n *\n * You'll need to include the `epicenter-multiplayer-dependencies.js` library in addition to the `epicenter.js` library in your project to use the Channel Service. See [Including Epicenter.js](../../#include).\n *\n * To use the Channel Service, instantiate it, then make calls to any of the methods you need.\n *\n * var cs = new F.service.Channel();\n * cs.publish('/acme-simulations/supply-chain-game/fall-seminar/run/variables', { price: 50 });\n *\n * If you are working through the [Epicenter Channel Manager](../epicenter-channel-manager/), when you ask to \"get\" a particular channel, you are really asking for an instance of the Channel Service with a topic already set, for example to the appropriate group or world:\n *\n * var cm = new F.manager.ChannelManager();\n * var gc = cm.getGroupChannel();\n * // because we used an Epicenter Channel Manager to get the group channel,\n * // subscribe() and publish() here default to the base topic for the group\n * gc.subscribe('', function(data) { console.log(data); });\n * gc.publish('', { message: 'a new message to the group' });\n *\n * The parameters for instantiating a Channel Service include:\n *\n * * `options` The options object to configure the Channel Service.\n * * `options.base` The base topic. This is added as a prefix to all further topics you publish or subscribe to while working with this Channel Service.\n * * `options.topicResolver` A function that processes all 'topics' passed into the `publish` and `subscribe` methods. This is useful if you want to implement your own serialize functions for converting custom objects to topic names. Returns a String. By default, it just echoes the topic.\n * * `options.transport` The instance of `$.cometd` to hook onto. See http://docs.cometd.org/reference/javascript.html for additional background on cometd.\n */\n\n'use strict';\nvar Channel = function (options) {\n var defaults = {\n\n /**\n * The base topic. This is added as a prefix to all further topics you publish or subscribe to while working with this Channel Service.\n * @type {string}\n */\n base: '',\n\n /**\n * A function that processes all 'topics' passed into the `publish` and `subscribe` methods. This is useful if you want to implement your own serialize functions for converting custom objects to topic names. By default, it just echoes the topic.\n *\n * **Parameters**\n *\n * * `topic` Topic to parse.\n *\n * **Return Value**\n *\n * * *String*: This function should return a string topic.\n *\n * @type {function}\n * @param {String} topic topic to resolve\n * @return {String}\n */\n topicResolver: function (topic) {\n return topic;\n },\n\n /**\n * The instance of `$.cometd` to hook onto.\n * @type {object}\n */\n transport: null\n };\n this.channelOptions = $.extend(true, {}, defaults, options);\n};\n\nvar makeName = function (channelName, topic) {\n //Replace trailing/double slashes\n var newName = (channelName ? (channelName + '/' + topic) : topic).replace(/\\/\\//g, '/').replace(/\\/$/, '');\n return newName;\n};\n\n\nChannel.prototype = $.extend(Channel.prototype, {\n\n // future functionality:\n // // Set the context for the callback\n // cs.subscribe('run', function () { this.innerHTML = 'Triggered'}, document.body);\n //\n // // Control the order of operations by setting the `priority`\n // cs.subscribe('run', cb, this, {priority: 9});\n //\n // // Only execute the callback, `cb`, if the value of the `price` variable is 50\n // cs.subscribe('run/variables/price', cb, this, {priority: 30, value: 50});\n //\n // // Only execute the callback, `cb`, if the value of the `price` variable is greater than 50\n // subscribe('run/variables/price', cb, this, {priority: 30, value: '>50'});\n //\n // // Only execute the callback, `cb`, if the value of the `price` variable is even\n // subscribe('run/variables/price', cb, this, {priority: 30, value: function (val) {return val % 2 === 0}});\n\n\n /**\n * Subscribe to changes on a topic.\n *\n * The topic should include the full path of the account id (**Team ID** for team projects), project id, and group name. (In most cases, it is simpler to use the [Epicenter Channel Manager](../epicenter-channel-manager/) instead, in which case this is configured for you.)\n *\n * **Examples**\n *\n * var cb = function(val) { console.log(val.data); };\n *\n * // Subscribe to changes on a top-level 'run' topic\n * cs.subscribe('/acme-simulations/supply-chain-game/fall-seminar/run', cb);\n *\n * // Subscribe to changes on children of the 'run' topic. Note this will also be triggered for changes to run.x.y.z.\n * cs.subscribe('/acme-simulations/supply-chain-game/fall-seminar/run/*', cb);\n *\n * // Subscribe to changes on both the top-level 'run' topic and its children\n * cs.subscribe(['/acme-simulations/supply-chain-game/fall-seminar/run',\n * '/acme-simulations/supply-chain-game/fall-seminar/run/*'], cb);\n *\n * // Subscribe to changes on a particular variable\n * subscribe('/acme-simulations/supply-chain-game/fall-seminar/run/variables/price', cb);\n *\n *\n * **Return Value**\n *\n * * *String* Returns a token you can later use to unsubscribe.\n *\n * **Parameters**\n * @param {String|Array} topic List of topics to listen for changes on.\n * @param {Function} callback Callback function to execute. Callback is called with signature `(evt, payload, metadata)`.\n * @param {Object} context Context in which the `callback` is executed.\n * @param {Object} options (Optional) Overrides for configuration options.\n * @param {Number} options.priority Used to control order of operations. Defaults to 0. Can be any +ve or -ve number.\n * @param {String|Number|Function} options.value The `callback` is only triggered if this condition matches. See examples for details.\n * @return {string} Subscription ID\n */\n subscribe: function (topic, callback, context, options) {\n\n var topics = [].concat(topic);\n var me = this;\n var subscriptionIds = [];\n var opts = me.channelOptions;\n\n opts.transport.batch(function () {\n $.each(topics, function (index, topic) {\n topic = makeName(opts.base, opts.topicResolver(topic));\n subscriptionIds.push(opts.transport.subscribe(topic, callback));\n });\n });\n return (subscriptionIds[1] ? subscriptionIds : subscriptionIds[0]);\n },\n\n /**\n * Publish data to a topic.\n *\n * **Examples**\n *\n * // Send data to all subscribers of the 'run' topic\n * cs.publish('/acme-simulations/supply-chain-game/fall-seminar/run', { completed: false });\n *\n * // Send data to all subscribers of the 'run/variables' topic\n * cs.publish('/acme-simulations/supply-chain-game/fall-seminar/run/variables', { price: 50 });\n *\n * **Parameters**\n *\n * @param {String} topic Topic to publish to.\n * @param {*} data Data to publish to topic.\n * @return {Array | Object} Responses to published data\n *\n */\n publish: function (topic, data) {\n var topics = [].concat(topic);\n var me = this;\n var returnObjs = [];\n var opts = me.channelOptions;\n\n\n opts.transport.batch(function () {\n $.each(topics, function (index, topic) {\n topic = makeName(opts.base, opts.topicResolver(topic));\n if (topic.charAt(topic.length - 1) === '*') {\n topic = topic.replace(/\\*+$/, '');\n console.warn('You can cannot publish to channels with wildcards. Publishing to ', topic, 'instead');\n }\n returnObjs.push(opts.transport.publish(topic, data));\n });\n });\n return (returnObjs[1] ? returnObjs : returnObjs[0]);\n },\n\n /**\n * Unsubscribe from changes to a topic.\n *\n * **Example**\n *\n * cs.unsubscribe('sampleToken');\n *\n * **Parameters**\n * @param {String} token The token for topic is returned when you initially subscribe. Pass it here to unsubscribe from that topic.\n * @return {Object} reference to current instance\n */\n unsubscribe: function (token) {\n this.channelOptions.transport.unsubscribe(token);\n return this;\n },\n\n /**\n * Start listening for events on this instance. Signature is same as for jQuery Events: http://api.jquery.com/on/.\n *\n * Supported events are: `connect`, `disconnect`, `subscribe`, `unsubscribe`, `publish`, `error`.\n *\n * **Parameters**\n *\n * @param {string} event The event type. See more detail at jQuery Events: http://api.jquery.com/on/.\n */\n on: function (event) {\n $(this).on.apply($(this), arguments);\n },\n\n /**\n * Stop listening for events on this instance. Signature is same as for jQuery Events: http://api.jquery.com/off/.\n *\n * **Parameters**\n *\n * @param {string} event The event type. See more detail at jQuery Events: http://api.jquery.com/off/.\n */\n off: function (event) {\n $(this).off.apply($(this), arguments);\n },\n\n /**\n * Trigger events and execute handlers. Signature is same as for jQuery Events: http://api.jquery.com/trigger/.\n *\n * **Parameters**\n *\n * @param {string} event The event type. See more detail at jQuery Events: http://api.jquery.com/trigger/.\n */\n trigger: function (event) {\n $(this).trigger.apply($(this), arguments);\n }\n\n});\n\nmodule.exports = Channel;\n","/**\n * @class ConfigurationService\n *\n * All services take in a configuration settings object to configure themselves. A JS hash {} is a valid configuration object, but optionally you can use the configuration service to toggle configs based on the environment\n *\n * @example\n * var cs = require('configuration-service')({\n * dev: { //environment\n port: 3000,\n host: 'localhost',\n },\n prod: {\n port: 8080,\n host: 'api.forio.com',\n logLevel: 'none'\n },\n logLevel: 'DEBUG' //global\n * });\n *\n * cs.get('logLevel'); //returns 'DEBUG'\n *\n * cs.setEnv('dev');\n * cs.get('logLevel'); //returns 'DEBUG'\n *\n * cs.setEnv('prod');\n * cs.get('logLevel'); //returns 'none'\n *\n */\n\n'use strict';\nvar urlService = require('./url-config-service');\n\nmodule.exports = function (config) {\n //TODO: Environments\n var defaults = {\n logLevel: 'NONE'\n };\n var serviceOptions = $.extend({}, defaults, config);\n serviceOptions.server = urlService(serviceOptions.server);\n\n return {\n\n data: serviceOptions,\n\n /**\n * Set the environment key to get configuration options from\n * @param { string} env\n */\n setEnv: function (env) {\n\n },\n\n /**\n * Get configuration.\n * @param { string} property optional\n * @return {*} Value of property if specified, the entire config object otherwise\n */\n get: function (property) {\n return serviceOptions[property];\n },\n\n /**\n * Set configuration.\n * @param { string|Object} key if a key is provided, set a key to that value. Otherwise merge object with current config\n * @param {*} value value for provided key\n */\n set: function (key, value) {\n serviceOptions[key] = value;\n }\n };\n};\n\n","/**\n * ## Data API Service\n *\n * The Data API Service allows you to create, access, and manipulate data related to any of your projects. Data are organized in collections. Each collection contains a document; each element of this top-level document is a JSON object. (See additional information on the underlying [Data API](../../../rest_apis/data_api/).)\n *\n * All API calls take in an \"options\" object as the last parameter. The options can be used to extend/override the Data API Service defaults. In particular, there are three required parameters when you instantiate the Data Service:\n *\n * * `account`: Epicenter account id (**Team ID** for team projects, **User ID** for personal projects).\n * * `project`: Epicenter project id.\n * * `root`: The the name of the collection. If you have multiple collections within each of your projects, you can also pass the collection name as an option for each call.\n *\n * var ds = new F.service.Data({\n * account: 'acme-simulations',\n * project: 'supply-chain-game',\n * root: 'survey-responses',\n * server: { host: 'api.forio.com' }\n * });\n * ds.saveAs('user1',\n * { 'question1': 2, 'question2': 10,\n * 'question3': false, 'question4': 'sometimes' } );\n * ds.saveAs('user2',\n * { 'question1': 3, 'question2': 8,\n * 'question3': true, 'question4': 'always' } );\n * ds.query('',{ 'question2': { '$gt': 9} });\n *\n * Note that in addition to the `account`, `project`, and `root`, the Data Service parameters optionally include a `server` object, whose `host` field contains the URI of the Forio server. This is automatically set, but you can pass it explicitly if desired. It is most commonly used for clarity when you are [hosting an Epicenter project on your own server](../../../how_to/self_hosting/).\n */\n\n'use strict';\n\nvar ConfigService = require('./configuration-service');\nvar qutil = require('../util/query-util');\nvar TransportFactory = require('../transport/http-transport-factory');\nvar SessionManager = require('../store/session-manager');\n\nmodule.exports = function (config) {\n var defaults = {\n /**\n * Name of collection. Required. Defaults to `/`, that is, the root level of your project at `forio.com/app/your-account-id/your-project-id/`, but must be set to a collection name.\n * @type {String}\n */\n root: '/',\n\n /**\n * The account id. In the Epicenter UI, this is the **Team ID** (for team projects) or **User ID** (for personal projects). Defaults to empty string. If left undefined, taken from the URL.\n * @type {String}\n */\n account: undefined,\n\n /**\n * The project id. Defaults to empty string. If left undefined, taken from the URL.\n * @type {String}\n */\n project: undefined,\n\n /**\n * For operations that require authentication, pass in the user access token (defaults to empty string). If the user is already logged in to Epicenter, the user access token is already set in a cookie and automatically loaded from there. (See [more background on access tokens](../../../project_access/)).\n * @see [Authentication API Service](../auth-api-service/) for getting tokens.\n * @type {String}\n */\n token: undefined,\n\n //Options to pass on to the underlying transport layer\n transport: {}\n };\n this.sessionManager = new SessionManager();\n var serviceOptions = this.sessionManager.getMergedOptions(defaults, config);\n\n var urlConfig = new ConfigService(serviceOptions).get('server');\n if (serviceOptions.account) {\n urlConfig.accountPath = serviceOptions.account;\n }\n if (serviceOptions.project) {\n urlConfig.projectPath = serviceOptions.project;\n }\n\n var getURL = function (key, root) {\n if (!root) {\n root = serviceOptions.root;\n }\n var url = urlConfig.getAPIPath('data') + qutil.addTrailingSlash(root);\n if (key) {\n url += qutil.addTrailingSlash(key);\n }\n return url;\n };\n\n var httpOptions = $.extend(true, {}, serviceOptions.transport, {\n url: getURL\n });\n if (serviceOptions.token) {\n httpOptions.headers = {\n Authorization: 'Bearer ' + serviceOptions.token\n };\n }\n var http = new TransportFactory(httpOptions);\n\n var publicAPI = {\n\n /**\n * Search for data within a collection.\n *\n * Searching using comparison or logical operators (as opposed to exact matches) requires MongoDB syntax. See the underlying [Data API](../../../rest_apis/data_api/#searching) for additional details.\n *\n * **Examples**\n *\n * // request all data associated with document 'user1'\n * ds.query('user1');\n *\n * // exact matching:\n * // request all documents in collection where 'question2' is 9\n * ds.query('', { 'question2': 9});\n *\n * // comparison operators:\n * // request all documents in collection\n * // where 'question2' is greater than 9\n * ds.query('', { 'question2': { '$gt': 9} });\n *\n * // logical operators:\n * // request all documents in collection\n * // where 'question2' is less than 10, and 'question3' is false\n * ds.query('', { '$and': [ { 'question2': { '$lt':10} }, { 'question3': false }] });\n *\n * // regular expresssions: use any Perl-compatible regular expressions\n * // request all documents in collection\n * // where 'question5' contains the string '.*day'\n * ds.query('', { 'question5': { '$regex': '.*day' } });\n *\n * **Parameters**\n * @param {String} key The name of the document to search. Pass the empty string ('') to search the entire collection.\n * @param {Object} query The query object. For exact matching, this object contains the field name and field value to match. For matching based on comparison, this object contains the field name and the comparison expression. For matching based on logical operators, this object contains an expression using MongoDB syntax. See the underlying [Data API](../../../rest_apis/data_api/#searching) for additional examples.\n * @param {Object} outputModifier (Optional) Available fields include: `startrecord`, `endrecord`, `sort`, and `direction` (`asc` or `desc`).\n * @param {Object} options (Optional) Overrides for configuration options.\n * @return {Promise} \n */\n query: function (key, query, outputModifier, options) {\n var params = $.extend(true, { q: query }, outputModifier);\n var httpOptions = $.extend(true, {}, serviceOptions, options);\n httpOptions.url = getURL(key, httpOptions.root);\n return http.get(params, httpOptions);\n },\n\n /**\n * Save data in an anonymous document within the collection.\n *\n * The `root` of the collection must be specified. By default the `root` is taken from the Data Service configuration options; you can also pass the `root` to the `save` call explicitly by overriding the options (third parameter).\n *\n * (Additional background: Documents are top-level elements within a collection. Collections must be unique within this account (team or personal account) and project and are set with the `root` field in the `option` parameter. See the underlying [Data API](../../../rest_apis/data_api/) for more information. The `save` method is making a `POST` request.)\n *\n * **Example**\n *\n * // Create a new document, with one element, at the default root level\n * ds.save('question1', 'yes');\n *\n * // Create a new document, with two elements, at the default root level\n * ds.save({ question1:'yes', question2: 32 });\n *\n * // Create a new document, with two elements, at `/students/`\n * ds.save({ name:'John', className: 'CS101' }, { root: 'students' });\n *\n * **Parameters**\n *\n * @param {String|Object} key If `key` is a string, it is the id of the element to save (create) in this document. If `key` is an object, the object is the data to save (create) in this document. In both cases, the id for the document is generated automatically.\n * @param {Object} value (Optional) The data to save. If `key` is a string, this is the value to save. If `key` is an object, the value(s) to save are already part of `key` and this argument is not required.\n * @param {Object} options (Optional) Overrides for configuration options. If you want to override the default `root` of the collection, do so here.\n * @return {Promise} \n */\n save: function (key, value, options) {\n var attrs;\n if (typeof key === 'object') {\n attrs = key;\n options = value;\n } else {\n (attrs = {})[key] = value;\n }\n var httpOptions = $.extend(true, {}, serviceOptions, options);\n httpOptions.url = getURL('', httpOptions.root);\n\n return http.post(attrs, httpOptions);\n },\n\n /**\n * Save (create or replace) data in a named document or element within the collection. \n * \n * The `root` of the collection must be specified. By default the `root` is taken from the Data Service configuration options; you can also pass the `root` to the `saveAs` call explicitly by overriding the options (third parameter).\n *\n * Optionally, the named document or element can include path information, so that you are saving just part of the document.\n *\n * (Additional background: Documents are top-level elements within a collection. Collections must be unique within this account (team or personal account) and project and are set with the `root` field in the `option` parameter. See the underlying [Data API](../../../rest_apis/data_api/) for more information. The `saveAs` method is making a `PUT` request.)\n *\n * **Example**\n *\n * // Create (or replace) the `user1` document at the default root level.\n * // Note that this replaces any existing content in the `user1` document.\n * ds.saveAs('user1',\n * { 'question1': 2, 'question2': 10,\n * 'question3': false, 'question4': 'sometimes' } );\n *\n * // Create (or replace) the `student1` document at the `students` root, \n * // that is, the data at `/students/student1/`.\n * // Note that this replaces any existing content in the `/students/student1/` document.\n * // However, this will keep existing content in other paths of this collection.\n * // For example, the data at `/students/student2/` is unchanged by this call.\n * ds.saveAs('student1',\n * { firstName: 'john', lastName: 'smith' },\n * { root: 'students' });\n *\n * // Create (or replace) the `mgmt100/groupB` document at the `myclasses` root,\n * // that is, the data at `/myclasses/mgmt100/groupB/`.\n * // Note that this replaces any existing content in the `/myclasses/mgmt100/groupB/` document.\n * // However, this will keep existing content in other paths of this collection.\n * // For example, the data at `/myclasses/mgmt100/groupA/` is unchanged by this call.\n * ds.saveAs('mgmt100/groupB',\n * { scenarioYear: '2015' },\n * { root: 'myclasses' });\n *\n * **Parameters**\n *\n * @param {String} key Id of the document.\n * @param {Object} value (Optional) The data to save, in key:value pairs.\n * @param {Object} options (Optional) Overrides for configuration options. If you want to override the default `root` of the collection, do so here.\n * @return {Promise} \n */\n saveAs: function (key, value, options) {\n var httpOptions = $.extend(true, {}, serviceOptions, options);\n httpOptions.url = getURL(key, httpOptions.root);\n\n return http.put(value, httpOptions);\n },\n\n /**\n * Get data for a specific document or field.\n *\n * **Example**\n *\n * ds.load('user1');\n * ds.load('user1/question3');\n *\n * **Parameters**\n * @param {String|Object} key The id of the data to return. Can be the id of a document, or a path to data within that document.\n * @param {Object} outputModifier (Optional) Available fields include: `startrecord`, `endrecord`, `sort`, and `direction` (`asc` or `desc`).\n * @param {Object} options Overrides for configuration options.\n * @return {Promise} \n */\n load: function (key, outputModifier, options) {\n var httpOptions = $.extend(true, {}, serviceOptions, options);\n httpOptions.url = getURL(key, httpOptions.root);\n return http.get(outputModifier, httpOptions);\n },\n\n /**\n * Removes data from collection. Only documents (top-level elements in each collection) can be deleted.\n *\n * **Example**\n *\n * ds.remove('user1');\n *\n *\n * **Parameters**\n *\n * @param {String|Array} keys The id of the document to remove from this collection, or an array of such ids.\n * @param {Object} options (Optional) Overrides for configuration options.\n * @return {Promise} \n */\n remove: function (keys, options) {\n var httpOptions = $.extend(true, {}, serviceOptions, options);\n var params;\n if ($.isArray(keys)) {\n params = { id: keys };\n } else {\n params = '';\n httpOptions.url = getURL(keys, httpOptions.root);\n }\n return http.delete(params, httpOptions);\n }\n\n // Epicenter doesn't allow nuking collections\n // /**\n // * Removes collection being referenced\n // * @return null\n // */\n // destroy: function (options) {\n // return this.remove('', options);\n // }\n };\n\n $.extend(this, publicAPI);\n};\n","/**\n *\n * ## Group API Adapter\n *\n * The Group API Adapter provides methods to look up, create, change or remove information about groups in a project. It is based on query capabilities of the underlying RESTful [Group API](../../../rest_apis/user_management/group/).\n *\n * This is only needed for Authenticated projects, that is, team projects with [end users and groups](../../../groups_and_end_users/).\n *\n * var ma = new F.service.Group({ token: 'user-or-project-access-token' });\n * ma.getGroupsForProject({ account: 'acme', project: 'sample' });\n */\n\n'use strict';\n\nvar serviceUtils = require('./service-utils');\nvar TransportFactory = require('../transport/http-transport-factory');\nvar objectAssign = require('object-assign');\n\nvar apiEndpoint = 'group/local';\n\nvar GroupService = function (config) {\n var defaults = {\n /**\n * Epicenter account name. Defaults to undefined.\n * @type {string}\n */\n account: undefined,\n\n /**\n * Epicenter project name. Defaults to undefined.\n * @type {string}\n */\n project: undefined,\n\n /**\n * Options to pass on to the underlying transport layer. All jquery.ajax options at http://api.jquery.com/jQuery.ajax/ are available. Defaults to empty object.\n * @type {object}\n */\n transport: {}\n };\n var serviceOptions = serviceUtils.getDefaultOptions(defaults, config, { apiEndpoint: apiEndpoint });\n var transportOptions = serviceOptions.transport;\n delete serviceOptions.transport;\n var http = new TransportFactory(transportOptions, serviceOptions);\n var publicAPI = {\n /**\n * Gets information for a group or multiple groups.\n * @param {Object} params object with query parameters\n * @patam {string} params.q partial match for name, organization or event.\n * @patam {string} params.account Epicenter's Team ID\n * @patam {string} params.project Epicenter's Project ID\n * @patam {string} params.name Epicenter's Group Name\n * @param {Object} options (Optional) Overrides for configuration options.\n * @return {Promise}\n */\n getGroups: function (params, options) {\n //groupID is part of the URL\n //q, account and project are part of the query string\n var finalOpts = objectAssign({}, serviceOptions, options);\n var finalParams;\n if (typeof params === 'string') {\n finalOpts.url = serviceUtils.getApiUrl(apiEndpoint + '/' + params, finalOpts);\n } else {\n finalParams = params;\n }\n return http.get(finalParams, finalOpts);\n }\n };\n objectAssign(this, publicAPI);\n};\n\nmodule.exports = GroupService;\n","/**\n *\n * ## Introspection API Service\n *\n * The Introspection API Service allows you to view a list of the variables and operations in a model. Typically used in conjunction with the [Run API Service](../run-api-service/).\n *\n * The Introspection API Service is not available for Forio SimLang.\n *\n * var intro = new F.service.Introspect({\n * account: 'acme-simulations',\n * project: 'supply-chain-game'\n * });\n * intro.byModel('supply-chain.py').then(function(data){ ... });\n * intro.byRunID('2b4d8f71-5c34-435a-8c16-9de674ab72e6').then(function(data){ ... });\n *\n */\n\n'use strict';\n\nvar ConfigService = require('./configuration-service');\nvar TransportFactory = require('../transport/http-transport-factory');\nvar SessionManager = require('../store/session-manager');\n\nvar apiEndpoint = 'model/introspect';\n\nmodule.exports = function (config) {\n var defaults = {\n /**\n * For projects that require authentication, pass in the user access token (defaults to empty string). If the user is already logged in to Epicenter, the user access token is already set in a cookie and automatically loaded from there. (See [more background on access tokens](../../../project_access/)).\n * @see [Authentication API Service](../auth-api-service/) for getting tokens.\n * @type {String}\n */\n token: undefined,\n\n /**\n * The account id. In the Epicenter UI, this is the **Team ID** (for team projects) or **User ID** (for personal projects). Defaults to empty string. If left undefined, taken from the URL.\n * @type {String}\n */\n account: undefined,\n\n /**\n * The project id. Defaults to empty string. If left undefined, taken from the URL.\n * @type {String}\n */\n project: undefined,\n\n };\n\n var sessionManager = new SessionManager();\n var serviceOptions = sessionManager.getMergedOptions(defaults, config);\n\n var urlConfig = new ConfigService(serviceOptions).get('server');\n if (serviceOptions.account) {\n urlConfig.accountPath = serviceOptions.account;\n }\n if (serviceOptions.project) {\n urlConfig.projectPath = serviceOptions.project;\n }\n\n var transportOptions = $.extend(true, {}, serviceOptions.transport, {\n url: urlConfig.getAPIPath(apiEndpoint)\n });\n if (serviceOptions.token) {\n transportOptions.headers = {\n Authorization: 'Bearer ' + serviceOptions.token\n };\n }\n var http = new TransportFactory(transportOptions);\n\n var publicAPI = {\n /**\n * Get the available variables and operations for a given model file.\n *\n * Note: This does not work for any model which requires additional parameters, such as `files`.\n *\n * **Example**\n *\n * intro.byModel('abc.vmf')\n * .then(function(data) {\n * // data contains an object with available functions (used with operations API) and available variables (used with variables API)\n * console.log(data.functions);\n * console.log(data.variables);\n * });\n *\n * **Parameters**\n * @param {String} modelFile Name of the model file to introspect.\n * @param {Object} options (Optional) Overrides for configuration options.\n * @return {Promise} \n */\n byModel: function (modelFile, options) {\n var opts = $.extend(true, {}, serviceOptions, options);\n if (!opts.account || !opts.project) {\n throw new Error('Account and project are required when using introspect#byModel');\n }\n if (!modelFile) {\n throw new Error('modelFile is required when using introspect#byModel');\n }\n var url = { url: urlConfig.getAPIPath(apiEndpoint) + [opts.account, opts.project, modelFile].join('/') };\n var httpOptions = $.extend(true, {}, serviceOptions, options, url);\n return http.get('', httpOptions);\n },\n\n /**\n * Get the available variables and operations for a given model file.\n *\n * Note: This does not work for any model which requires additional parameters such as `files`.\n *\n * **Example**\n *\n * intro.byRunID('2b4d8f71-5c34-435a-8c16-9de674ab72e6')\n * .then(function(data) {\n * // data contains an object with available functions (used with operations API) and available variables (used with variables API)\n * console.log(data.functions);\n * console.log(data.variables);\n * });\n *\n * **Parameters**\n * @param {String} runID Id of the run to introspect.\n * @param {Object} options (Optional) Overrides for configuration options.\n * @return {Promise} \n */\n byRunID: function (runID, options) {\n if (!runID) {\n throw new Error('runID is required when using introspect#byModel');\n }\n var url = { url: urlConfig.getAPIPath(apiEndpoint) + runID };\n var httpOptions = $.extend(true, {}, serviceOptions, options, url);\n return http.get('', httpOptions);\n }\n };\n $.extend(this, publicAPI);\n};\n","/**\n *\n * ## Member API Adapter\n *\n * The Member API Adapter provides methods to look up information about end users for your project and how they are divided across groups. It is based on query capabilities of the underlying RESTful [Member API](../../../rest_apis/user_management/member/).\n *\n * This is only needed for Authenticated projects, that is, team projects with [end users and groups](../../../groups_and_end_users/). For example, if some of your end users are facilitators, or if your end users should be treated differently based on which group they are in, use the Member API to find that information.\n *\n * var ma = new F.service.Member({ token: 'user-or-project-access-token' });\n * ma.getGroupsForUser({ userId: 'b6b313a3-ab84-479c-baea-206f6bff337' });\n * ma.getGroupDetails({ groupId: '00b53308-9833-47f2-b21e-1278c07d53b8' });\n */\n\n'use strict';\n\nvar ConfigService = require('./configuration-service');\nvar TransportFactory = require('../transport/http-transport-factory');\nvar SessionManager = require('../store/session-manager');\nvar _pick = require('../util/object-util')._pick;\nvar apiEndpoint = 'member/local';\n\nmodule.exports = function (config) {\n var defaults = {\n /**\n * Epicenter user id. Defaults to a blank string.\n * @type {string}\n */\n userId: undefined,\n\n /**\n * Epicenter group id. Defaults to a blank string. Note that this is the group *id*, not the group *name*.\n * @type {string}\n */\n groupId: undefined,\n\n /**\n * Options to pass on to the underlying transport layer. All jquery.ajax options at http://api.jquery.com/jQuery.ajax/ are available. Defaults to empty object.\n * @type {object}\n */\n transport: {}\n };\n this.sessionManager = new SessionManager();\n var serviceOptions = this.sessionManager.getMergedOptions(defaults, config);\n var urlConfig = new ConfigService(serviceOptions).get('server');\n\n var transportOptions = $.extend(true, {}, serviceOptions.transport, {\n url: urlConfig.getAPIPath(apiEndpoint)\n });\n\n if (serviceOptions.token) {\n transportOptions.headers = {\n Authorization: 'Bearer ' + serviceOptions.token\n };\n }\n var http = new TransportFactory(transportOptions, serviceOptions);\n\n var getFinalParams = function (params) {\n if (typeof params === 'object') {\n return $.extend(true, serviceOptions, params);\n }\n return serviceOptions;\n };\n\n var patchUserActiveField = function (params, active, options) {\n var httpOptions = $.extend(true, serviceOptions, options, {\n url: urlConfig.getAPIPath(apiEndpoint) + params.groupId + '/' + params.userId\n });\n\n return http.patch({ active: active }, httpOptions);\n };\n\n var publicAPI = {\n\n /**\n * Retrieve details about all of the group memberships for one end user. The membership details are returned in an array, with one element (group record) for each group to which the end user belongs.\n *\n * In the membership array, each group record includes the group id, project id, account (team) id, and an array of members. However, only the user whose userId is included in the call is listed in the members array (regardless of whether there are other members in this group).\n *\n * **Example**\n *\n * var ma = new F.service.Member({ token: 'user-or-project-access-token' });\n * ma.getGroupsForUser('42836d4b-5b61-4fe4-80eb-3136e956ee5c')\n * .then(function(memberships){\n * for (var i=0; i 1,\n * // where variables.price has been persisted (recorded)\n * // in the model.\n * rs.query({\n * 'saved': 'true',\n * '.price': '>1'\n * },\n * {\n * startrecord: 2,\n * endrecord: 5\n * });\n *\n * **Parameters**\n * @param {Object} qs Query object. Each key can be a property of the run or the name of variable that has been saved in the run (prefaced by `variables.`). Each value can be a literal value, or a comparison operator and value. (See [more on filtering](../../../rest_apis/aggregate_run_api/#filters) allowed in the underlying Run API.) Querying for variables is available for runs [in memory](../../../run_persistence/#runs-in-memory) and for runs [in the database](../../../run_persistence/#runs-in-memory) if the variables are persisted (e.g. that have been `record`ed in your model or marked for saving in your [model context file](../../../model_code/context/)).\n * @param {Object} outputModifier (Optional) Available fields include: `startrecord`, `endrecord`, `sort`, and `direction` (`asc` or `desc`).\n * @param {Object} options (Optional) Overrides for configuration options.\n * @return {Promise}\n */\n query: function (qs, outputModifier, options) {\n var httpOptions = $.extend(true, {}, serviceOptions, { url: urlConfig.getFilterURL(qs) }, options);\n httpOptions = urlConfig.addAutoRestoreHeader(httpOptions);\n\n return http.splitGet(outputModifier, httpOptions).then(function (r) {\n return ($.isPlainObject(r) && Object.keys(r).length === 0) ? [] : r;\n });\n },\n\n /**\n * Returns particular runs, based on conditions specified in the `qs` object.\n *\n * Similar to `.query()`.\n *\n * **Parameters**\n * @param {Object} filter Filter object. Each key can be a property of the run or the name of variable that has been saved in the run (prefaced by `variables.`). Each value can be a literal value, or a comparison operator and value. (See [more on filtering](../../../rest_apis/aggregate_run_api/#filters) allowed in the underlying Run API.) Filtering for variables is available for runs [in memory](../../../run_persistence/#runs-in-memory) and for runs [in the database](../../../run_persistence/#runs-in-memory) if the variables are persisted (e.g. that have been `record`ed in your model or marked for saving in your [model context file](../../../model_code/context/)).\n * @param {Object} outputModifier (Optional) Available fields include: `startrecord`, `endrecord`, `sort`, and `direction` (`asc` or `desc`).\n * @param {Object} options (Optional) Overrides for configuration options.\n * @return {Promise}\n */\n filter: function (filter, outputModifier, options) {\n if ($.isPlainObject(serviceOptions.filter)) {\n $.extend(serviceOptions.filter, filter);\n } else {\n serviceOptions.filter = filter;\n }\n var httpOptions = $.extend(true, {}, serviceOptions, options);\n httpOptions = urlConfig.addAutoRestoreHeader(httpOptions);\n return http.splitGet(outputModifier, httpOptions).then(function (r) {\n return ($.isPlainObject(r) && Object.keys(r).length === 0) ? [] : r;\n });\n },\n\n /**\n * Get data for a specific run. This includes standard run data such as the account, model, project, and created and last modified dates. To request specific model variables or run record variables, pass them as part of the `filters` parameter.\n *\n * Note that if the run is [in memory](../../../run_persistence/#runs-in-memory), any model variables are available; if the run is [in the database](../../../run_persistence/#runs-in-db), only model variables that have been persisted — that is, `record`ed or saved in your model — are available.\n *\n * **Example**\n *\n * rs.load('bb589677-d476-4971-a68e-0c58d191e450', { include: ['.price', '.sales'] });\n *\n * **Parameters**\n * @param {String} runID The run id.\n * @param {Object} filters (Optional) Object containing filters and operation modifiers. Use key `include` to list model variables that you want to include in the response. Other available fields include: `startrecord`, `endrecord`, `sort`, and `direction` (`asc` or `desc`).\n * @param {Object} options (Optional) Overrides for configuration options.\n * @return {Promise}\n */\n load: function (runID, filters, options) {\n if (runID) {\n serviceOptions.filter = runID; //shouldn't be able to over-ride\n }\n var httpOptions = $.extend(true, {}, serviceOptions, options);\n httpOptions = urlConfig.addAutoRestoreHeader(httpOptions);\n return http.get(filters, httpOptions);\n },\n\n /**\n * Removes specified runid from memory\n *\n * **Example**\n *\n * rs.removeFromMemory('bb589677-d476-4971-a68e-0c58d191e450');\n *\n * See [details on run persistence](../../../run_persistence/#runs-in-memory)\n * @param {String} [runID] id of run to remove\n * @param {Object} [filters] (Optional) Object containing filters and operation modifiers. Use key `include` to list model variables that you want to include in the response. Other available fields include: `startrecord`, `endrecord`, `sort`, and `direction` (`asc` or `desc`).\n * @param {Object} [options] (Optional) Overrides for configuration options.\n * @return {Promise}\n */\n removeFromMemory: function (runID, filters, options) {\n var httpOptions = $.extend(true, {}, serviceOptions, options);\n if (runID) {\n httpOptions.url = urlConfig.getAPIPath('run') + runID;\n }\n return http.delete({}, httpOptions);\n },\n\n /**\n * Save attributes (data, model variables) of the run.\n *\n * **Examples**\n *\n * // add 'completed' field to run record\n * rs.save({ completed: true });\n *\n * // update 'saved' field of run record, and update values of model variables for this run\n * rs.save({ saved: true, variables: { a: 23, b: 23 } });\n *\n * // update 'saved' field of run record for a particular run\n * rs.save({ saved: true }, { id: '0000015bf2a04995880df6b868d23eb3d229' });\n *\n * **Parameters**\n * @param {Object} attributes The run data and variables to save.\n * @param {Object} attributes.variables Model variables must be included in a `variables` field within the `attributes` object. (Otherwise they are treated as run data and added to the run record directly.)\n * @param {Object} options (Optional) Overrides for configuration options.\n * @return {Promise}\n */\n save: function (attributes, options) {\n var httpOptions = $.extend(true, {}, serviceOptions, options);\n setFilterOrThrowError(httpOptions);\n return http.patch(attributes, httpOptions);\n },\n\n /**\n * Call an operation from the model.\n *\n * Depending on the language in which you have written your model, the operation (function or method) may need to be exposed (e.g. `export` for a Julia model) in the model file in order to be called through the API. See [Writing your Model](../../../writing_your_model/)).\n *\n * The `params` argument is normally an array of arguments to the `operation`. In the special case where `operation` only takes one argument, you are not required to put that argument into an array.\n *\n * Note that you can combine the `operation` and `params` arguments into a single object if you prefer, as in the last example.\n *\n * **Examples**\n *\n * // operation \"solve\" takes no arguments\n * rs.do('solve');\n * // operation \"echo\" takes one argument, a string\n * rs.do('echo', ['hello']);\n * // operation \"echo\" takes one argument, a string\n * rs.do('echo', 'hello');\n * // operation \"sumArray\" takes one argument, an array\n * rs.do('sumArray', [[4,2,1]]);\n * // operation \"add\" takes two arguments, both integers\n * rs.do({ name:'add', params:[2,4] });\n * // call operation \"solve\" on a different run \n * rs.do('solve', { id: '0000015bf2a04995880df6b868d23eb3d229' });\n *\n * **Parameters**\n * @param {String} operation Name of operation.\n * @param {Array} params (Optional) Any parameters the operation takes, passed as an array. In the special case where `operation` only takes one argument, you are not required to put that argument into an array, and can just pass it directly.\n * @param {Object} options (Optional) Overrides for configuration options.\n * @return {Promise}\n */\n do: function (operation, params, options) {\n // console.log('do', operation, params);\n var opsArgs;\n var postOptions;\n if (options) {\n opsArgs = params;\n postOptions = options;\n } else if ($.isPlainObject(params)) {\n opsArgs = null;\n postOptions = params;\n } else {\n opsArgs = params;\n }\n var result = rutil.normalizeOperations(operation, opsArgs);\n var httpOptions = $.extend(true, {}, serviceOptions, postOptions);\n\n setFilterOrThrowError(httpOptions);\n\n var prms = (result.args[0].length && (result.args[0] !== null && result.args[0] !== undefined)) ? result.args[0] : [];\n return http.post({ arguments: prms }, $.extend(true, {}, httpOptions, {\n url: urlConfig.getFilterURL() + 'operations/' + result.ops[0] + '/'\n }));\n },\n\n /**\n * Call several operations from the model, sequentially.\n *\n * Depending on the language in which you have written your model, the operation (function or method) may need to be exposed (e.g. `export` for a Julia model) in the model file in order to be called through the API. See [Writing your Model](../../../writing_your_model/)).\n *\n * **Examples**\n *\n * // operations \"initialize\" and \"solve\" do not take any arguments\n * rs.serial(['initialize', 'solve']);\n * // operations \"init\" and \"reset\" take two arguments each\n * rs.serial([ { name: 'init', params: [1,2] },\n * { name: 'reset', params: [2,3] }]);\n * // operation \"init\" takes two arguments,\n * // operation \"runmodel\" takes none\n * rs.serial([ { name: 'init', params: [1,2] },\n * { name: 'runmodel', params: [] }]);\n *\n * **Parameters**\n * @param {Array} operations If none of the operations take parameters, pass an array of the operation names (strings). If any of the operations do take parameters, pass an array of objects, each of which contains an operation name and its own (possibly empty) array of parameters.\n * @param {*} params Parameters to pass to operations.\n * @param {Object} options (Optional) Overrides for configuration options.\n * @return {Promise} The parameter to the callback is an array. Each array element is an object containing the results of one operation.\n */\n serial: function (operations, params, options) {\n var opParams = rutil.normalizeOperations(operations, params);\n var ops = opParams.ops;\n var args = opParams.args;\n var me = this;\n\n var $d = $.Deferred();\n var postOptions = $.extend(true, {}, serviceOptions, options);\n\n var responses = [];\n var doSingleOp = function () {\n var op = ops.shift();\n var arg = args.shift();\n\n me.do(op, arg, {\n success: function (result) {\n responses.push(result);\n if (ops.length) {\n doSingleOp();\n } else {\n $d.resolve(responses);\n postOptions.success(responses, me);\n }\n },\n error: function (err) {\n responses.push(err);\n $d.reject(responses);\n postOptions.error(responses, me);\n }\n });\n };\n\n doSingleOp();\n\n return $d.promise();\n },\n\n /**\n * Call several operations from the model, executing them in parallel.\n *\n * Depending on the language in which you have written your model, the operation (function or method) may need to be exposed (e.g. `export` for a Julia model) in the model file in order to be called through the API. See [Writing your Model](../../../writing_your_model/)).\n *\n * **Example**\n *\n * // operations \"solve\" and \"reset\" do not take any arguments\n * rs.parallel(['solve', 'reset']);\n * // operations \"add\" and \"subtract\" take two arguments each\n * rs.parallel([ { name: 'add', params: [1,2] },\n * { name: 'subtract', params:[2,3] }]);\n * // operations \"add\" and \"subtract\" take two arguments each\n * rs.parallel({ add: [1,2], subtract: [2,4] });\n *\n * **Parameters**\n * @param {Array|Object} operations If none of the operations take parameters, pass an array of the operation names (as strings). If any of the operations do take parameters, you have two options. You can pass an array of objects, each of which contains an operation name and its own (possibly empty) array of parameters. Alternatively, you can pass a single object with the operation name and a (possibly empty) array of parameters.\n * @param {*} params Parameters to pass to operations.\n * @param {Object} options (Optional) Overrides for configuration options.\n * @return {Promise} The parameter to the callback is an array. Each array element is an object containing the results of one operation.\n */\n parallel: function (operations, params, options) {\n var $d = $.Deferred();\n\n var opParams = rutil.normalizeOperations(operations, params);\n var ops = opParams.ops;\n var args = opParams.args;\n var postOptions = $.extend(true, {}, serviceOptions, options);\n\n var queue = [];\n for (var i = 0; i < ops.length; i++) {\n queue.push(\n this.do(ops[i], args[i])\n );\n }\n\n var me = this;\n $.when.apply(this, queue)\n .then(function () {\n var args = Array.prototype.slice.call(arguments);\n var actualResponse = args.map(function (a) {\n return a[0];\n });\n $d.resolve(actualResponse);\n postOptions.success(actualResponse, me);\n })\n .fail(function () {\n var args = Array.prototype.slice.call(arguments);\n var actualResponse = args.map(function (a) {\n return a[0];\n });\n $d.reject(actualResponse);\n postOptions.error(actualResponse, me);\n });\n\n return $d.promise();\n },\n\n /**\n * Shortcut to using the [Introspection API Service](../introspection-api-service/). Allows you to view a list of the variables and operations in a model.\n *\n * **Example**\n *\n * rs.introspect({ runID: 'cbf85437-b539-4977-a1fc-23515cf071bb' }).then(function (data) {\n * console.log(data.functions);\n * console.log(data.variables);\n * });\n *\n * **Parameters**\n * @param {Object} options Options can either be of the form `{ runID: }` or `{ model: }`. Note that the `runID` is optional if the Run Service is already associated with a particular run (because `id` was passed in when the Run Service was initialized). If provided, the `runID` overrides the `id` currently associated with the Run Service.\n * @param {Object} introspectionConfig (Optional) Service options for Introspection Service\n * @return {Promise}\n */\n introspect: function (options, introspectionConfig) {\n var introspection = new IntrospectionService($.extend(true, {}, serviceOptions, introspectionConfig));\n if (options) {\n if (options.runID) {\n return introspection.byRunID(options.runID);\n } else if (options.model) {\n return introspection.byModel(options.model);\n }\n } else if (serviceOptions.id) {\n return introspection.byRunID(serviceOptions.id);\n } else {\n throw new Error('Please specify either the model or runid to introspect');\n }\n }\n };\n\n var publicSyncAPI = {\n getCurrentConfig: function () {\n return serviceOptions;\n },\n updateConfig: function (config) {\n if (config && config.id) {\n config.filter = config.id;\n } else if (config && config.filter) {\n config.id = config.filter;\n }\n serviceOptions = $.extend(true, {}, serviceOptions, config);\n urlConfig = updateURLConfig(serviceOptions);\n this.urlConfig = urlConfig;\n updateHTTPConfig(serviceOptions, urlConfig);\n },\n /**\n * Returns a Variables Service instance. Use the variables instance to load, save, and query for specific model variables. See the [Variable API Service](../variables-api-service/) for more information.\n *\n * **Example**\n *\n * var vs = rs.variables();\n * vs.save({ sample_int: 4 });\n *\n * **Parameters**\n * @param {Object} config (Optional) Overrides for configuration options.\n * @return {Object} variablesService Instance\n */\n variables: function (config) {\n var vs = new VariablesService($.extend(true, {}, serviceOptions, config, {\n runService: this\n }));\n return vs;\n }\n };\n\n $.extend(this, publicAsyncAPI);\n $.extend(this, publicSyncAPI);\n};\n","'use strict';\n\nvar ConfigService = require('./configuration-service');\nvar SessionManager = require('../store/session-manager');\nvar objectAssign = require('object-assign');\n\nvar serviceUtils = {\n /*\n * Gets the default options for a api service.\n * It will merge:\n * - The Session options (Using the Session Manager)\n * - The Authorization Header from the token option\n * - The full url from the endpoint option\n * With the supplied overrides and defaults\n *\n */\n getDefaultOptions: function (defaults) {\n var rest = Array.prototype.slice.call(arguments, 1);\n var sessionManager = new SessionManager();\n var serviceOptions = sessionManager.getMergedOptions.apply(sessionManager, [defaults].concat(rest));\n\n serviceOptions.transport = objectAssign({}, serviceOptions.transport, {\n url: this.getApiUrl(serviceOptions.apiEndpoint, serviceOptions)\n });\n\n if (serviceOptions.token) {\n serviceOptions.transport.headers = {\n Authorization: 'Bearer ' + serviceOptions.token\n };\n }\n return serviceOptions;\n },\n\n getApiUrl: function (apiEndpoint, serviceOptions) {\n var urlConfig = new ConfigService(serviceOptions).get('server');\n return urlConfig.getAPIPath(apiEndpoint);\n }\n};\n\nmodule.exports = serviceUtils;","'use strict';\n/**\n * ## State API Adapter\n *\n * The State API Adapter allows you to view the history of a run, and to replay or clone runs. \n *\n * The State API Adapter brings existing, persisted run data from the database back into memory, using the same run id (`replay`) or a new run id (`clone`). Runs must be in memory in order for you to update variables or call operations on them.\n *\n * Specifically, the State API Adapter works by \"re-running\" the run (user interactions) from the creation of the run up to the time it was last persisted in the database. This process uses the current version of the run's model. Therefore, if the model has changed since the original run was created, the retrieved run will use the new model — and may end up having different values or behavior as a result. Use with care!\n *\n * To use the State API Adapter, instantiate it and then call its methods:\n *\n * var sa = new F.service.State();\n * sa.replay({runId: '1842bb5c-83ad-4ba8-a955-bd13cc2fdb4f'});\n *\n * The constructor takes an optional `options` parameter in which you can specify the `account` and `project` if they are not already available in the current context.\n *\n */\n\nvar ConfigService = require('./configuration-service');\nvar TransportFactory = require('../transport/http-transport-factory');\nvar _pick = require('../util/object-util')._pick;\nvar SessionManager = require('../store/session-manager');\nvar apiEndpoint = 'model/state';\n\nmodule.exports = function (config) {\n\n var defaults = {\n\n };\n\n this.sessionManager = new SessionManager();\n var serviceOptions = this.sessionManager.getMergedOptions(defaults, config);\n var urlConfig = new ConfigService(serviceOptions).get('server');\n\n var transportOptions = $.extend(true, {}, serviceOptions.transport, {\n url: urlConfig.getAPIPath(apiEndpoint)\n });\n\n if (serviceOptions.token) {\n transportOptions.headers = {\n Authorization: 'Bearer ' + serviceOptions.token\n };\n }\n\n var http = new TransportFactory(transportOptions);\n var parseRunIdOrError = function (params) {\n if ($.isPlainObject(params) && params.runId) {\n return params.runId;\n } else {\n throw new Error('Please pass in a run id');\n }\n };\n\n var publicAPI = {\n\n /**\n * View the history of a run.\n * \n * **Example**\n *\n * var sa = new F.service.State();\n * sa.load('0000015a06bb58613b28b57365677ec89ec5').then(function(history) {\n * console.log('history = ', history);\n * });\n *\n * **Parameters**\n * @param {string} runId The id of the run.\n * @param {object} options (Optional) Overrides for configuration options.\n * @return {Promise}\n */\n load: function (runId, options) {\n var httpParams = $.extend(true, {},\n serviceOptions,\n options,\n { url: urlConfig.getAPIPath(apiEndpoint) + runId }\n );\n return http.get('', httpParams);\n },\n\n /**\n * Replay a run. After this call, the run, with its original run id, is now available [in memory](../../../run_persistence/#runs-in-memory). (It continues to be persisted into the Epicenter database at regular intervals.)\n *\n * **Example**\n *\n * var sa = new F.service.State();\n * sa.replay({runId: '1842bb5c-83ad-4ba8-a955-bd13cc2fdb4f', stopBefore: 'calculateScore'});\n *\n * **Parameters**\n * @param {object} params Parameters object.\n * @param {string} params.runId The id of the run to bring back to memory.\n * @param {string} params.stopBefore (Optional) The run is advanced only up to the first occurrence of this method.\n * @param {array} params.exclude (Optional) Array of methods to exclude when advancing the run.\n * @param {object} options (Optional) Overrides for configuration options.\n * @return {Promise}\n */\n replay: function (params, options) {\n var runId = parseRunIdOrError(params);\n\n var replayOptions = $.extend(true, {},\n serviceOptions,\n options,\n { url: urlConfig.getAPIPath(apiEndpoint) + runId }\n );\n\n params = $.extend(true, { action: 'replay' }, _pick(params, ['stopBefore', 'exclude']));\n\n return http.post(params, replayOptions);\n },\n\n /**\n * Clone a given run and return a new run in the same state as the given run.\n *\n * The new run id is now available [in memory](../../../run_persistence/#runs-in-memory). The new run includes a copy of all of the data from the original run, EXCEPT:\n *\n * * The `saved` field in the new run record is not copied from the original run record. It defaults to `false`.\n * * The `initialized` field in the new run record is not copied from the original run record. It defaults to `false` but may change to `true` as the new run is advanced. For example, if there has been a call to the `step` function (for Vensim models), the `initialized` field is set to `true`.\n * * The `created` field in the new run record is the date and time at which the clone was created (not the time that the original run was created.)\n *\n * The original run remains only [in the database](../../../run_persistence/#runs-in-db).\n *\n * **Example**\n *\n * var sa = new F.service.State();\n * sa.clone({runId: '1842bb5c-83ad-4ba8-a955-bd13cc2fdb4f', stopBefore: 'calculateScore', exclude: ['interimCalculation'] });\n *\n * **Parameters**\n * @param {object} params Parameters object.\n * @param {string} params.runId The id of the run to clone from memory.\n * @param {string} params.stopBefore (Optional) The newly cloned run is advanced only up to the first occurrence of this method.\n * @param {array} params.exclude (Optional) Array of methods to exclude when advancing the newly cloned run.\n * @param {object} options (Optional) Overrides for configuration options.\n * @return {Promise}\n */\n clone: function (params, options) {\n var runId = parseRunIdOrError(params);\n\n var replayOptions = $.extend(true, {},\n serviceOptions,\n options,\n { url: urlConfig.getAPIPath(apiEndpoint) + runId }\n );\n\n params = $.extend(true, { action: 'clone' }, _pick(params, ['stopBefore', 'exclude']));\n\n return http.post(params, replayOptions);\n }\n };\n\n $.extend(this, publicAPI);\n};\n","'use strict';\n\nvar epiVersion = require('../api-version.json');\n\n//TODO: urlutils to get host, since no window on node\nvar defaults = {\n host: window.location.host,\n pathname: window.location.pathname\n};\n\nfunction getLocalHost(existingFn, host) {\n var localHostFn;\n if (existingFn !== undefined) {\n if (!$.isFunction(existingFn)) {\n localHostFn = function () { return existingFn; };\n } else {\n localHostFn = existingFn;\n }\n } else {\n localHostFn = function () {\n var isLocal = !host || //phantomjs\n host === '127.0.0.1' || \n host.indexOf('local.') === 0 || \n host.indexOf('localhost') === 0;\n return isLocal;\n };\n }\n return localHostFn;\n}\n\nvar UrlConfigService = function (config) {\n var envConf = UrlConfigService.defaults;\n\n if (!config) {\n config = {};\n }\n // console.log(this.defaults);\n var overrides = $.extend({}, envConf, config);\n var options = $.extend({}, defaults, overrides);\n\n overrides.isLocalhost = options.isLocalhost = getLocalHost(options.isLocalhost, options.host);\n \n // console.log(isLocalhost(), '___________');\n var actingHost = config && config.host;\n if (!actingHost && options.isLocalhost()) {\n actingHost = 'forio.com';\n } else {\n actingHost = options.host;\n }\n\n var API_PROTOCOL = 'https';\n var HOST_API_MAPPING = {\n 'forio.com': 'api.forio.com',\n 'foriodev.com': 'api.epicenter.foriodev.com'\n };\n\n var publicExports = {\n protocol: API_PROTOCOL,\n\n api: '',\n\n //TODO: this should really be called 'apihost', but can't because that would break too many things\n host: (function () {\n var apiHost = (HOST_API_MAPPING[actingHost]) ? HOST_API_MAPPING[actingHost] : actingHost;\n // console.log(actingHost, config, apiHost);\n return apiHost;\n }()),\n\n isCustomDomain: (function () {\n var path = options.pathname.split('\\/');\n var pathHasApp = path && path[1] === 'app';\n return (!options.isLocalhost() && !pathHasApp);\n }()),\n\n appPath: (function () {\n var path = options.pathname.split('\\/');\n\n return path && path[1] || '';\n }()),\n\n accountPath: (function () {\n var accnt = '';\n var path = options.pathname.split('\\/');\n if (path && path[1] === 'app') {\n accnt = path[2];\n }\n return accnt;\n }()),\n\n projectPath: (function () {\n var prj = '';\n var path = options.pathname.split('\\/');\n if (path && path[1] === 'app') {\n prj = path[3]; //eslint-disable-line no-magic-numbers\n }\n return prj;\n }()),\n\n versionPath: (function () {\n var version = epiVersion.version ? epiVersion.version + '/' : '';\n return version;\n }()),\n\n getAPIPath: function (api) {\n var PROJECT_APIS = ['run', 'data', 'file', 'presence'];\n var apiMapping = {\n channel: 'channel/subscribe'\n };\n var apiEndpoint = apiMapping[api] || api;\n \n if (apiEndpoint === 'config') {\n var actualProtocol = window.location.protocol.replace(':', '');\n var configProtocol = (options.isLocalhost()) ? this.protocol : actualProtocol;\n return configProtocol + '://' + actingHost + '/epicenter/' + this.versionPath + 'config';\n }\n var apiPath = this.protocol + '://' + this.host + '/' + this.versionPath + apiEndpoint + '/';\n\n if ($.inArray(apiEndpoint, PROJECT_APIS) !== -1) {\n apiPath += this.accountPath + '/' + this.projectPath + '/';\n }\n return apiPath;\n }\n };\n\n\n $.extend(publicExports, overrides);\n return publicExports;\n};\n// This data can be set by external scripts, for loading from an env server for eg;\nUrlConfigService.defaults = {};\n\nmodule.exports = UrlConfigService;\n","'use strict';\n/**\n* ## User API Adapter\n*\n* The User API Adapter allows you to retrieve details about end users in your team (account). It is based on the querying capabilities of the underlying RESTful [User API](../../../rest_apis/user_management/user/).\n*\n* To use the User API Adapter, instantiate it and then call its methods.\n*\n* var ua = new F.service.User({\n* account: 'acme-simulations',\n* token: 'user-or-project-access-token'\n* });\n* ua.getById('42836d4b-5b61-4fe4-80eb-3136e956ee5c');\n* ua.get({ userName: 'jsmith' });\n* ua.get({ id: ['42836d4b-5b61-4fe4-80eb-3136e956ee5c',\n* '4ea75631-4c8d-4872-9d80-b4600146478e'] });\n*\n* The constructor takes an optional `options` parameter in which you can specify the `account` and `token` if they are not already available in the current context.\n*/\n\nvar ConfigService = require('./configuration-service');\nvar TransportFactory = require('../transport/http-transport-factory');\nvar SessionManager = require('../store/session-manager');\nvar qutil = require('../util/query-util');\n\nmodule.exports = function (config) {\n var defaults = {\n\n /**\n * The account id. In the Epicenter UI, this is the **Team ID** (for team projects) or **User ID** (for personal projects). Defaults to empty string.\n * @type {String}\n */\n account: undefined,\n\n /**\n * The access token to use when searching for end users. (See [more background on access tokens](../../../project_access/)).\n * @type {String}\n */\n token: undefined,\n\n /**\n * Options to pass on to the underlying transport layer. All jquery.ajax options at http://api.jquery.com/jQuery.ajax/ are available. Defaults to empty object.\n * @type {Object}\n */\n transport: {}\n };\n\n this.sessionManager = new SessionManager();\n var serviceOptions = this.sessionManager.getMergedOptions(defaults, config);\n var urlConfig = new ConfigService(serviceOptions).get('server');\n var transportOptions = $.extend(true, {}, serviceOptions.transport, {\n url: urlConfig.getAPIPath('user')\n });\n\n if (serviceOptions.token) {\n transportOptions.headers = {\n Authorization: 'Bearer ' + serviceOptions.token\n };\n }\n\n var http = new TransportFactory(transportOptions);\n\n var publicAPI = {\n\n /**\n * Retrieve details about particular end users in your team, based on user name or user id.\n *\n * **Example**\n *\n * var ua = new F.service.User({\n * account: 'acme-simulations',\n * token: 'user-or-project-access-token'\n * });\n * ua.get({ userName: 'jsmith' });\n * ua.get({ id: ['42836d4b-5b61-4fe4-80eb-3136e956ee5c',\n * '4ea75631-4c8d-4872-9d80-b4600146478e'] });\n *\n * **Parameters**\n * @param {object} filter Object with field `userName` and value of the username. Alternatively, object with field `id` and value of an array of user ids.\n * @param {object} options (Optional) Overrides for configuration options.\n * @return {Promise}\n */\n\n get: function (filter, options) {\n options = options || {};\n filter = filter || {};\n\n var getOptions = $.extend(true, {},\n serviceOptions,\n options\n );\n\n var toQFilter = function (filter) {\n var res = {};\n\n // API only supports filtering by username for now\n if (filter.userName) {\n res.q = filter.userName;\n }\n\n return res;\n };\n\n var toIdFilters = function (id) {\n if (!id) {\n return '';\n }\n\n id = $.isArray(id) ? id : [id];\n return 'id=' + id.join('&id=');\n };\n\n var getFilters = [\n 'account=' + getOptions.account,\n toIdFilters(filter.id),\n qutil.toQueryFormat(toQFilter(filter))\n ].join('&');\n\n // special case for queries with large number of ids\n // make it as a post with GET semantics\n var threshold = 30;\n if (filter.id && $.isArray(filter.id) && filter.id.length >= threshold) {\n getOptions.url = urlConfig.getAPIPath('user') + '?_method=GET';\n return http.post({ id: filter.id }, getOptions);\n } else {\n return http.get(getFilters, getOptions);\n }\n },\n\n /**\n * Retrieve details about a single end user in your team, based on user id.\n *\n * **Example**\n *\n * var ua = new F.service.User({\n * account: 'acme-simulations',\n * token: 'user-or-project-access-token'\n * });\n * ua.getById('42836d4b-5b61-4fe4-80eb-3136e956ee5c');\n *\n * **Parameters**\n * @param {string} userId The user id for the end user in your team.\n * @param {object} options (Optional) Overrides for configuration options.\n * @return {Promise}\n */\n\n getById: function (userId, options) {\n return publicAPI.get({ id: userId }, options);\n }\n };\n\n $.extend(this, publicAPI);\n};\n\n\n","/**\n *\n * ## Variables API Service\n *\n * Used in conjunction with the [Run API Service](../run-api-service/) to read, write, and search for specific model variables.\n *\n * var rm = new F.manager.RunManager({\n * run: {\n * account: 'acme-simulations',\n * project: 'supply-chain-game',\n * model: 'supply-chain-model.jl'\n * }\n * });\n * rm.getRun()\n * .then(function() {\n * var vs = rm.run.variables();\n * vs.save({sample_int: 4});\n * });\n *\n */\n\n\n 'use strict';\n\n var TransportFactory = require('../transport/http-transport-factory');\n var rutil = require('../util/run-util');\n\n module.exports = function (config) {\n var defaults = {\n /**\n * The runs object to which the variable filters apply. Defaults to null.\n * @type {runService}\n */\n runService: null\n };\n var serviceOptions = $.extend({}, defaults, config);\n\n var getURL = function () {\n //TODO: Replace with getCurrentconfig instead?\n return serviceOptions.runService.urlConfig.getFilterURL() + 'variables/';\n };\n\n var addAutoRestoreHeader = function (options) {\n return serviceOptions.runService.urlConfig.addAutoRestoreHeader(options);\n };\n\n var httpOptions = {\n url: getURL\n };\n if (serviceOptions.token) {\n httpOptions.headers = {\n Authorization: 'Bearer ' + serviceOptions.token\n };\n }\n var http = new TransportFactory(httpOptions);\n http.splitGet = rutil.splitGetFactory(httpOptions);\n\n var publicAPI = {\n\n /**\n * Get values for a variable.\n *\n * **Example**\n *\n * vs.load('sample_int')\n * .then(function(val){\n * // val contains the value of sample_int\n * });\n *\n * **Parameters**\n * @param {String} variable Name of variable to load.\n * @param {Object} outputModifier (Optional) Available fields include: `startrecord`, `endrecord`, `sort`, and `direction` (`asc` or `desc`).\n * @param {Object} options (Optional) Overrides for configuration options.\n * @return {Promise}\n */\n load: function (variable, outputModifier, options) {\n var httpOptions = $.extend(true, {}, serviceOptions, options);\n httpOptions = addAutoRestoreHeader(httpOptions);\n return http.get(outputModifier, $.extend({}, httpOptions, {\n url: getURL() + variable + '/'\n }));\n },\n\n /**\n * Returns particular variables, based on conditions specified in the `query` object.\n *\n * **Example**\n *\n * vs.query(['price', 'sales'])\n * .then(function(val) {\n * // val is an object with the values of the requested variables: val.price, val.sales\n * });\n *\n * vs.query({ include:['price', 'sales'] });\n *\n * **Parameters**\n * @param {Object|Array} query The names of the variables requested.\n * @param {Object} outputModifier (Optional) Available fields include: `startrecord`, `endrecord`, `sort`, and `direction` (`asc` or `desc`).\n * @param {Object} options (Optional) Overrides for configuration options.\n * @return {Promise}\n */\n query: function (query, outputModifier, options) {\n //Query and outputModifier are both querystrings in the url; only calling them out separately here to be consistent with the other calls\n var httpOptions = $.extend(true, {}, serviceOptions, options);\n httpOptions = addAutoRestoreHeader(httpOptions);\n\n if ($.isArray(query)) {\n query = { include: query };\n }\n $.extend(query, outputModifier);\n return http.splitGet(query, httpOptions);\n },\n\n /**\n * Save values to model variables. Overwrites existing values. Note that you can only update model variables if the run is [in memory](../../../run_persistence/#runs-in-memory). (An alternate way to update model variables is to call a method from the model and make sure that the method persists the variables. See `do`, `serial`, and `parallel` in the [Run API Service](../run-api-service/) for calling methods from the model.)\n *\n * **Example**\n *\n * vs.save('price', 4);\n * vs.save({ price: 4, quantity: 5, products: [2,3,4] });\n *\n * **Parameters**\n * @param {Object|String} variable An object composed of the model variables and the values to save. Alternatively, a string with the name of the variable.\n * @param {Object} val (Optional) If passing a string for `variable`, use this argument for the value to save.\n * @param {Object} options (Optional) Overrides for configuration options.\n * @return {Promise}\n */\n save: function (variable, val, options) {\n var attrs;\n if (typeof variable === 'object') {\n attrs = variable;\n options = val;\n } else {\n (attrs = {})[variable] = val;\n }\n var httpOptions = $.extend(true, {}, serviceOptions, options);\n\n return http.patch.call(this, attrs, httpOptions);\n }\n\n // Not Available until underlying API supports PUT. Otherwise save would be PUT and merge would be PATCH\n // *\n // * Save values to the api. Merges arrays, but otherwise same as save\n // * @param {Object|String} variable Object with attributes, or string key\n // * @param {Object} val Optional if prev parameter was a string, set value here\n // * @param {Object} options Overrides for configuration options\n // *\n // * @example\n // * vs.merge({ price: 4, quantity: 5, products: [2,3,4] })\n // * vs.merge('price', 4);\n\n // merge: function (variable, val, options) {\n // var attrs;\n // if (typeof variable === 'object') {\n // attrs = variable;\n // options = val;\n // } else {\n // (attrs = {})[variable] = val;\n // }\n // var httpOptions = $.extend(true, {}, serviceOptions, options);\n\n // return http.patch.call(this, attrs, httpOptions);\n // }\n };\n $.extend(this, publicAPI);\n };\n","/**\n * ## World API Adapter\n *\n * A [run](../../../glossary/#run) is a collection of end user interactions with a project and its model -- including setting variables, making decisions, and calling operations. For building multiplayer simulations you typically want multiple end users to share the same set of interactions, and work within a common state. Epicenter allows you to create \"worlds\" to handle such cases. Only [team projects](../../../glossary/#team) can be multiplayer.\n *\n * The World API Adapter allows you to create, access, and manipulate multiplayer worlds within your Epicenter project. You can use this to add and remove end users from the world, and to create, access, and remove their runs. Because of this, typically the World Adapter is used for facilitator pages in your project. (The related [World Manager](../world-manager/) provides an easy way to access runs and worlds for particular end users, so is typically used in pages that end users will interact with.)\n *\n * As with all the other [API Adapters](../../), all methods take in an \"options\" object as the last parameter. The options can be used to extend/override the World API Service defaults.\n *\n * To use the World Adapter, instantiate it and then access the methods provided. Instantiating requires the account id (**Team ID** in the Epicenter user interface), project id (**Project ID**), and group (**Group Name**).\n *\n * var wa = new F.service.World({\n * account: 'acme-simulations',\n * project: 'supply-chain-game',\n * group: 'team1' });\n * wa.create()\n * .then(function(world) {\n * // call methods, e.g. wa.addUsers()\n * });\n */\n\n'use strict';\n\nvar ConfigService = require('./configuration-service');\n// var qutil = require('../util/query-util');\nvar TransportFactory = require('../transport/http-transport-factory');\nvar SessionManager = require('../store/session-manager');\nvar _pick = require('../util/object-util')._pick;\n\nvar apiBase = 'multiplayer/';\nvar assignmentEndpoint = apiBase + 'assign';\nvar apiEndpoint = apiBase + 'world';\nvar projectEndpoint = apiBase + 'project';\n\nmodule.exports = function (config) {\n var defaults = {\n /**\n * For projects that require authentication, pass in the user access token (defaults to empty string). If the user is already logged in to Epicenter, the user access token is already set in a cookie and automatically loaded from there. (See [more background on access tokens](../../../project_access/)).\n * @see [Authentication API Service](../auth-api-service/) for getting tokens.\n * @type {String}\n */\n token: undefined,\n\n /**\n * The project id. If left undefined, taken from the URL.\n * @type {String}\n */\n project: undefined,\n\n /**\n * The account id. In the Epicenter UI, this is the **Team ID** (for team projects). If left undefined, taken from the URL.\n * @type {String}\n */\n account: undefined,\n\n /**\n * The group name. Defaults to undefined.\n * @type {String}\n */\n group: undefined,\n\n /**\n * The model file to use to create runs in this world. Defaults to undefined.\n * @type {String}\n */\n model: undefined,\n\n /**\n * Criteria by which to filter world. Currently only supports world-ids as filters.\n * @type {String}\n */\n filter: '',\n\n /**\n * Convenience alias for filter\n * @type {String}\n */\n id: '',\n\n /**\n * Options to pass on to the underlying transport layer. All jquery.ajax options at http://api.jquery.com/jQuery.ajax/ are available. Defaults to empty object.\n * @type {Object}\n */\n transport: {},\n\n /**\n * Called when the call completes successfully. Defaults to `$.noop`.\n * @type {function}\n */\n success: $.noop,\n\n /**\n * Called when the call fails. Defaults to `$.noop`.\n * @type {function}\n */\n error: $.noop\n };\n\n this.sessionManager = new SessionManager();\n var serviceOptions = this.sessionManager.getMergedOptions(defaults, config);\n if (serviceOptions.id) {\n serviceOptions.filter = serviceOptions.id;\n }\n\n var urlConfig = new ConfigService(serviceOptions).get('server');\n\n if (!serviceOptions.account) {\n serviceOptions.account = urlConfig.accountPath;\n }\n\n if (!serviceOptions.project) {\n serviceOptions.project = urlConfig.projectPath;\n }\n\n var transportOptions = $.extend(true, {}, serviceOptions.transport, {\n url: urlConfig.getAPIPath(apiEndpoint)\n });\n\n if (serviceOptions.token) {\n transportOptions.headers = {\n Authorization: 'Bearer ' + serviceOptions.token\n };\n }\n\n var http = new TransportFactory(transportOptions);\n\n var setIdFilterOrThrowError = function (options) {\n if (options.id) {\n serviceOptions.filter = options.id;\n }\n if (options.filter) {\n serviceOptions.filter = options.filter;\n }\n if (!serviceOptions.filter) {\n throw new Error('No world id specified to apply operations against. This could happen if the user is not assigned to a world and is trying to work with runs from that world.');\n }\n };\n\n var validateModelOrThrowError = function (options) {\n if (!options.model) {\n throw new Error('No model specified to get the current run');\n }\n };\n\n var publicAPI = {\n\n /**\n * Creates a new World.\n *\n * Using this method is rare. It is more common to create worlds automatically while you `autoAssign()` end users to worlds. (In this case, configuration data for the world, such as the roles, are read from the project-level world configuration information, for example by `getProjectSettings()`.)\n *\n * **Example**\n *\n * var wa = new F.service.World({\n * account: 'acme-simulations',\n * project: 'supply-chain-game',\n * group: 'team1' });\n * wa.create({\n * roles: ['VP Marketing', 'VP Sales', 'VP Engineering']\n * });\n *\n * **Parameters**\n * @param {object} params Parameters to create the world.\n * @param {string} params.group (Optional) The **Group Name** to create this world under. Only end users in this group are eligible to join the world. Optional here; required when instantiating the service (`new F.service.World()`).\n * @param {object} params.roles (Optional) The list of roles (strings) for this world. Some worlds have specific roles that **must** be filled by end users. Listing the roles allows you to autoassign users to worlds and ensure that all roles are filled in each world.\n * @param {object} params.optionalRoles (Optional) The list of optional roles (strings) for this world. Some worlds have specific roles that **may** be filled by end users. Listing the optional roles as part of the world object allows you to autoassign users to worlds and ensure that all roles are filled in each world.\n * @param {integer} params.minUsers (Optional) The minimum number of users for the world. Including this number allows you to autoassign end users to worlds and ensure that the correct number of users are in each world.\n * @param {object} options (Optional) Options object to override global options.\n * @return {Promise}\n */\n create: function (params, options) {\n var createOptions = $.extend(true, {}, serviceOptions, options, { url: urlConfig.getAPIPath(apiEndpoint) });\n var worldApiParams = ['scope', 'files', 'roles', 'optionalRoles', 'minUsers', 'group', 'name'];\n var validParams = _pick(serviceOptions, ['account', 'project', 'group']);\n // whitelist the fields that we actually can send to the api\n params = _pick(params, worldApiParams);\n\n // account and project go in the body, not in the url\n params = $.extend({}, validParams, params);\n\n var oldSuccess = createOptions.success;\n createOptions.success = function (response) {\n serviceOptions.filter = response.id; //all future chained calls to operate on this id\n return oldSuccess.apply(this, arguments);\n };\n\n return http.post(params, createOptions);\n },\n\n /**\n * Updates a World, for example to replace the roles in the world.\n *\n * Typically, you complete world configuration at the project level, rather than at the world level. For example, each world in your project probably has the same roles for end users. And your project is probably either configured so that all end users share the same world (and run), or smaller sets of end users share worlds — but not both. However, this method is available if you need to update the configuration of a particular world.\n *\n * **Example**\n *\n * var wa = new F.service.World({\n * account: 'acme-simulations',\n * project: 'supply-chain-game',\n * group: 'team1' });\n * wa.create()\n * .then(function(world) {\n * wa.update({ roles: ['VP Marketing', 'VP Sales', 'VP Engineering'] });\n * });\n *\n * **Parameters**\n * @param {object} params Parameters to update the world.\n * @param {string} params.name A string identifier for the linked end users, for example, \"name\": \"Our Team\".\n * @param {object} params.roles (Optional) The list of roles (strings) for this world. Some worlds have specific roles that **must** be filled by end users. Listing the roles allows you to autoassign users to worlds and ensure that all roles are filled in each world.\n * @param {object} params.optionalRoles (Optional) The list of optional roles (strings) for this world. Some worlds have specific roles that **may** be filled by end users. Listing the optional roles as part of the world object allows you to autoassign users to worlds and ensure that all roles are filled in each world.\n * @param {integer} params.minUsers (Optional) The minimum number of users for the world. Including this number allows you to autoassign end users to worlds and ensure that the correct number of users are in each world.\n * @param {object} options (Optional) Options object to override global options.\n * @return {Promise}\n */\n update: function (params, options) {\n var whitelist = ['roles', 'optionalRoles', 'minUsers'];\n options = options || {};\n setIdFilterOrThrowError(options);\n\n var updateOptions = $.extend(true, {},\n serviceOptions,\n options,\n { url: urlConfig.getAPIPath(apiEndpoint) + serviceOptions.filter }\n );\n\n params = _pick(params || {}, whitelist);\n\n return http.patch(params, updateOptions);\n },\n\n /**\n * Deletes an existing world.\n *\n * This function optionally takes one argument. If the argument is a string, it is the id of the world to delete. If the argument is an object, it is the override for global options.\n *\n * **Example**\n *\n * var wa = new F.service.World({\n * account: 'acme-simulations',\n * project: 'supply-chain-game',\n * group: 'team1' });\n * wa.create()\n * .then(function(world) {\n * wa.delete();\n * });\n *\n * **Parameters**\n * @param {String|Object} options (Optional) The id of the world to delete, or options object to override global options.\n * @return {Promise}\n */\n delete: function (options) {\n options = (options && (typeof options === 'string')) ? { filter: options } : {};\n setIdFilterOrThrowError(options);\n\n var deleteOptions = $.extend(true, {},\n serviceOptions,\n options,\n { url: urlConfig.getAPIPath(apiEndpoint) + serviceOptions.filter }\n );\n\n return http.delete(null, deleteOptions);\n },\n\n /**\n * Updates the configuration for the current instance of the World API Adapter (including all subsequent function calls, until the configuration is updated again).\n *\n * **Example**\n *\n * var wa = new F.service.World({...}).updateConfig({ filter: '123' }).addUser({ userId: '123' });\n *\n * **Parameters**\n * @param {object} config The configuration object to use in updating existing configuration.\n * @return {Object} reference to current instance\n */\n updateConfig: function (config) {\n $.extend(serviceOptions, config);\n\n return this;\n },\n\n /**\n * Lists all worlds for a given account, project, and group. All three are required, and if not specified as parameters, are read from the service.\n *\n * **Example**\n *\n * var wa = new F.service.World({\n * account: 'acme-simulations',\n * project: 'supply-chain-game',\n * group: 'team1' });\n * wa.create()\n * .then(function(world) {\n * // lists all worlds in group \"team1\"\n * wa.list();\n *\n * // lists all worlds in group \"other-group-name\"\n * wa.list({ group: 'other-group-name' });\n * });\n *\n * **Parameters**\n * @param {object} options (Optional) Options object to override global options.\n * @return {Promise}\n */\n list: function (options) {\n options = options || {};\n\n var getOptions = $.extend(true, {},\n serviceOptions,\n options,\n { url: urlConfig.getAPIPath(apiEndpoint) }\n );\n\n var filters = _pick(getOptions, ['account', 'project', 'group']);\n\n return http.get(filters, getOptions);\n },\n\n /**\n * Gets all worlds that an end user belongs to for a given account (team), project, and group.\n *\n * **Example**\n *\n * var wa = new F.service.World({\n * account: 'acme-simulations',\n * project: 'supply-chain-game',\n * group: 'team1' });\n * wa.create()\n * .then(function(world) {\n * wa.getWorldsForUser('b1c19dda-2d2e-4777-ad5d-3929f17e86d3')\n * });\n *\n * ** Parameters **\n * @param {string} userId The `userId` of the user whose worlds are being retrieved.\n * @param {object} options (Optional) Options object to override global options.\n * @return {Promise}\n */\n getWorldsForUser: function (userId, options) {\n options = options || {};\n\n var getOptions = $.extend(true, {},\n serviceOptions,\n options,\n { url: urlConfig.getAPIPath(apiEndpoint) }\n );\n\n var filters = $.extend(\n _pick(getOptions, ['account', 'project', 'group']),\n { userId: userId }\n );\n\n return http.get(filters, getOptions);\n },\n\n /**\n * Load information for a specific world. All further calls to the world service will use the id provided.\n *\n * **Parameters**\n * @param {String} worldId The id of the world to load.\n * @param {Object} options (Optional) Options object to override global options.\n * @return {Promise}\n */\n load: function (worldId, options) {\n if (worldId) {\n serviceOptions.filter = worldId;\n }\n if (!serviceOptions.filter) {\n throw new Error('Please provide a worldid to load');\n }\n var httpOptions = $.extend(true, {}, serviceOptions, options, { url: urlConfig.getAPIPath(apiEndpoint) + serviceOptions.filter + '/' });\n return http.get('', httpOptions);\n },\n\n /**\n * Adds an end user or list of end users to a given world. The end user must be a member of the `group` that is associated with this world.\n *\n * **Example**\n *\n * var wa = new F.service.World({\n * account: 'acme-simulations',\n * project: 'supply-chain-game',\n * group: 'team1' });\n * wa.create()\n * .then(function(world) {\n * // add one user\n * wa.addUsers('b1c19dda-2d2e-4777-ad5d-3929f17e86d3');\n * wa.addUsers(['b1c19dda-2d2e-4777-ad5d-3929f17e86d3']);\n * wa.addUsers({ userId: 'b1c19dda-2d2e-4777-ad5d-3929f17e86d3', role: 'VP Sales' });\n *\n * // add several users\n * wa.addUsers([\n * { userId: 'a6fe0c1e-f4b8-4f01-9f5f-01ccf4c2ed44',\n * role: 'VP Marketing' },\n * { userId: '8f2604cf-96cd-449f-82fa-e331530734ee',\n * role: 'VP Engineering' }\n * ]);\n *\n * // add one user to a specific world\n * wa.addUsers('b1c19dda-2d2e-4777-ad5d-3929f17e86d3', world.id);\n * wa.addUsers('b1c19dda-2d2e-4777-ad5d-3929f17e86d3', { filter: world.id });\n * });\n *\n * ** Parameters **\n * @param {string|object|array} users User id, array of user ids, object, or array of objects of the users to add to this world.\n * @param {string} users.role The `role` the user should have in the world. It is up to the caller to ensure, if needed, that the `role` passed in is one of the `roles` or `optionalRoles` of this world.\n * @param {string} worldId The world to which the users should be added. If not specified, the filter parameter of the `options` object is used.\n * @param {object} options (Optional) Options object to override global options.\n * @return {Promise}\n */\n addUsers: function (users, worldId, options) {\n\n if (!users) {\n throw new Error('Please provide a list of users to add to the world');\n }\n\n // normalize the list of users to an array of user objects\n users = $.map([].concat(users), function (u) {\n var isObject = $.isPlainObject(u);\n\n if (typeof u !== 'string' && !isObject) {\n throw new Error('Some of the users in the list are not in the valid format: ' + u);\n }\n\n return isObject ? u : { userId: u };\n });\n\n // check if options were passed as the second parameter\n if ($.isPlainObject(worldId) && !options) {\n options = worldId;\n worldId = null;\n }\n\n options = options || {};\n\n // we must have options by now\n if (typeof worldId === 'string') {\n options.filter = worldId;\n }\n\n setIdFilterOrThrowError(options);\n\n var updateOptions = $.extend(true, {},\n serviceOptions,\n options,\n { url: urlConfig.getAPIPath(apiEndpoint) + serviceOptions.filter + '/users' }\n );\n\n return http.post(users, updateOptions);\n },\n\n /**\n * Updates the role of an end user in a given world. (You can only update one end user at a time.)\n *\n * **Example**\n *\n * var wa = new F.service.World({\n * account: 'acme-simulations',\n * project: 'supply-chain-game',\n * group: 'team1' });\n *\n * wa.create().then(function(world) {\n * wa.addUsers('b1c19dda-2d2e-4777-ad5d-3929f17e86d3');\n * wa.updateUser({ userId: 'b1c19dda-2d2e-4777-ad5d-3929f17e86d3', role: 'leader' });\n * });\n *\n * **Parameters**\n * @param {object} user User object with `userId` and the new `role`.\n * @param {object} options (Optional) Options object to override global options.\n * @return {Promise}\n */\n updateUser: function (user, options) {\n options = options || {};\n\n if (!user || !user.userId) {\n throw new Error('You need to pass a userId to update from the world');\n }\n\n setIdFilterOrThrowError(options);\n\n var patchOptions = $.extend(true, {},\n serviceOptions,\n options,\n { url: urlConfig.getAPIPath(apiEndpoint) + serviceOptions.filter + '/users/' + user.userId }\n );\n\n return http.patch(_pick(user, 'role'), patchOptions);\n },\n\n /**\n * Removes an end user from a given world.\n *\n * **Example**\n *\n * var wa = new F.service.World({\n * account: 'acme-simulations',\n * project: 'supply-chain-game',\n * group: 'team1' });\n * wa.create()\n * .then(function(world) {\n * wa.addUsers(['a6fe0c1e-f4b8-4f01-9f5f-01ccf4c2ed44', '8f2604cf-96cd-449f-82fa-e331530734ee']);\n * wa.removeUser('a6fe0c1e-f4b8-4f01-9f5f-01ccf4c2ed44');\n * wa.removeUser({ userId: '8f2604cf-96cd-449f-82fa-e331530734ee' });\n * });\n *\n * ** Parameters **\n * @param {object|string} user The `userId` of the user to remove from the world, or an object containing the `userId` field.\n * @param {object} options (Optional) Options object to override global options.\n * @return {Promise}\n */\n removeUser: function (user, options) {\n options = options || {};\n\n if (typeof user === 'string') {\n user = { userId: user };\n }\n\n if (!user.userId) {\n throw new Error('You need to pass a userId to remove from the world');\n }\n\n setIdFilterOrThrowError(options);\n\n var getOptions = $.extend(true, {},\n serviceOptions,\n options,\n { url: urlConfig.getAPIPath(apiEndpoint) + serviceOptions.filter + '/users/' + user.userId }\n );\n\n return http.delete(null, getOptions);\n },\n\n /**\n * Gets the run id of current run for the given world. If the world does not have a run, creates a new one and returns the run id.\n *\n * Remember that a [run](../../glossary/#run) is a collection of interactions with a project and its model. In the case of multiplayer projects, the run is shared by all end users in the world.\n *\n * **Example**\n *\n * var wa = new F.service.World({\n * account: 'acme-simulations',\n * project: 'supply-chain-game',\n * group: 'team1' });\n * wa.create()\n * .then(function(world) {\n * wa.getCurrentRunId({ model: 'model.py' });\n * });\n *\n * ** Parameters **\n * @param {object} options (Optional) Options object to override global options.\n * @param {object} options.model The model file to use to create a run if needed.\n * @return {Promise}\n */\n getCurrentRunId: function (options) {\n options = options || {};\n\n setIdFilterOrThrowError(options);\n\n var getOptions = $.extend(true, {},\n serviceOptions,\n options,\n { url: urlConfig.getAPIPath(apiEndpoint) + serviceOptions.filter + '/run' }\n );\n\n validateModelOrThrowError(getOptions);\n return http.post(_pick(getOptions, 'model'), getOptions);\n },\n\n /**\n * Gets the current (most recent) world for the given end user in the given group. Brings this most recent world into memory if needed.\n *\n * **Example**\n *\n * var wa = new F.service.World({\n * account: 'acme-simulations',\n * project: 'supply-chain-game',\n * group: 'team1' });\n * wa.getCurrentWorldForUser('8f2604cf-96cd-449f-82fa-e331530734ee')\n * .then(function(world) {\n * // use data from world\n * });\n *\n * ** Parameters **\n * @param {string} userId The `userId` of the user whose current (most recent) world is being retrieved.\n * @param {string} groupName (Optional) The name of the group. If not provided, defaults to the group used to create the service.\n * @return {Promise}\n */\n getCurrentWorldForUser: function (userId, groupName) {\n var dtd = $.Deferred();\n var me = this;\n this.getWorldsForUser(userId, { group: groupName })\n .then(function (worlds) {\n // assume the most recent world as the 'active' world\n worlds.sort(function (a, b) { return new Date(b.lastModified) - new Date(a.lastModified); });\n var currentWorld = worlds[0];\n\n if (currentWorld) {\n serviceOptions.filter = currentWorld.id;\n }\n\n dtd.resolveWith(me, [currentWorld]);\n })\n .fail(dtd.reject);\n\n return dtd.promise();\n },\n\n /**\n * Deletes the current run from the world.\n *\n * (Note that the world id remains part of the run record, indicating that the run was formerly an active run for the world.)\n *\n * **Example**\n *\n * var wa = new F.service.World({\n * account: 'acme-simulations',\n * project: 'supply-chain-game',\n * group: 'team1' });\n *\n * wa.deleteRun('sample-world-id');\n *\n * **Parameters**\n * @param {string} worldId The `worldId` of the world from which the current run is being deleted.\n * @param {object} options (Optional) Options object to override global options.\n * @return {Promise}\n */\n deleteRun: function (worldId, options) {\n options = options || {};\n\n if (worldId) {\n options.filter = worldId;\n }\n\n setIdFilterOrThrowError(options);\n\n var deleteOptions = $.extend(true, {},\n serviceOptions,\n options,\n { url: urlConfig.getAPIPath(apiEndpoint) + serviceOptions.filter + '/run' }\n );\n\n return http.delete(null, deleteOptions);\n },\n\n /**\n * Creates a new run for the world.\n *\n * **Example**\n *\n * var wa = new F.service.World({\n * account: 'acme-simulations',\n * project: 'supply-chain-game',\n * group: 'team1' });\n *\n * wa.getCurrentWorldForUser('8f2604cf-96cd-449f-82fa-e331530734ee')\n * .then(function (world) {\n * wa.newRunForWorld(world.id);\n * });\n *\n * **Parameters**\n * @param {string} worldId worldId in which we create the new run.\n * @param {object} options (Optional) Options object to override global options.\n * @param {object} options.model The model file to use to create a run if needed.\n * @return {Promise}\n */\n newRunForWorld: function (worldId, options) {\n var currentRunOptions = $.extend(true, {},\n serviceOptions,\n options,\n { filter: worldId || serviceOptions.filter }\n );\n var me = this;\n\n validateModelOrThrowError(currentRunOptions);\n\n return this.deleteRun(worldId, options)\n .then(function () {\n return me.getCurrentRunId(currentRunOptions);\n });\n },\n\n /**\n * Assigns end users to worlds, creating new worlds as appropriate, automatically. Assigns all end users in the group, and creates new worlds as needed based on the project-level world configuration (roles, optional roles, and minimum end users per world).\n *\n * **Example**\n *\n * var wa = new F.service.World({\n * account: 'acme-simulations',\n * project: 'supply-chain-game',\n * group: 'team1' });\n *\n * wa.autoAssign();\n *\n * **Parameters**\n * @param {object} options (Optional) Options object to override global options.\n * @param {number} options.maxUsers Sets the maximum number of users in a world.\n * @param {string[]} options.userIds A list of users to be assigned be assigned instead of all end users in the group.\n * @return {Promise}\n *\n */\n autoAssign: function (options) {\n options = options || {};\n\n var opt = $.extend(true, {},\n serviceOptions,\n options,\n { url: urlConfig.getAPIPath(assignmentEndpoint) }\n );\n\n var params = {\n account: opt.account,\n project: opt.project,\n group: opt.group\n };\n\n if (opt.maxUsers) {\n params.maxUsers = opt.maxUsers;\n }\n\n if (opt.userIds) {\n params.userIds = opt.userIds;\n }\n\n return http.post(params, opt);\n },\n\n /**\n * Gets the project's world configuration.\n *\n * Typically, every interaction with your project uses the same configuration of each world. For example, each world in your project probably has the same roles for end users. And your project is probably either configured so that all end users share the same world (and run), or smaller sets of end users share worlds — but not both.\n *\n * (The [Multiplayer Project REST API](../../../rest_apis/multiplayer/multiplayer_project/) allows you to set these project-level world configurations. The World Adapter simply retrieves them, for example so they can be used in auto-assignment of end users to worlds.)\n *\n * **Example**\n *\n * var wa = new F.service.World({\n * account: 'acme-simulations',\n * project: 'supply-chain-game',\n * group: 'team1' });\n *\n * wa.getProjectSettings()\n * .then(function(settings) {\n * console.log(settings.roles);\n * console.log(settings.optionalRoles);\n * });\n *\n * **Parameters**\n * @param {object} options (Optional) Options object to override global options.\n * @return {Promise}\n */\n getProjectSettings: function (options) {\n options = options || {};\n\n var opt = $.extend(true, {},\n serviceOptions,\n options,\n { url: urlConfig.getAPIPath(projectEndpoint) }\n );\n\n opt.url += [opt.account, opt.project].join('/');\n return http.get(null, opt);\n }\n\n };\n\n $.extend(this, publicAPI);\n};\n","/**\n * @class Cookie Storage Service\n *\n * @example\n * var people = require('cookie-store')({ root: 'people' });\n people\n .save({lastName: 'smith' })\n\n */\n\n\n'use strict';\n\n// Thin document.cookie wrapper to allow unit testing\nvar Cookie = function () {\n this.get = function () {\n return document.cookie;\n };\n\n this.set = function (newCookie) {\n document.cookie = newCookie;\n };\n};\n\nmodule.exports = function (config) {\n var host = window.location.hostname;\n var validHost = host.split('.').length > 1;\n var domain = validHost ? '.' + host : null;\n\n var defaults = {\n /**\n * Name of collection\n * @type { string}\n */\n root: '/',\n\n domain: domain,\n cookie: new Cookie()\n };\n this.serviceOptions = $.extend({}, defaults, config);\n\n var publicAPI = {\n // * TBD\n // * Query collection; uses MongoDB syntax\n // * @see \n // *\n // * @param { string} qs Query Filter\n // * @param { string} limiters @see \n // *\n // * @example\n // * cs.query(\n // * { name: 'John', className: 'CSC101' },\n // * {limit: 10}\n // * )\n\n // query: function (qs, limiters) {\n\n // },\n\n /**\n * Save cookie value\n * @param { string|Object} key If given a key save values under it, if given an object directly, save to top-level api\n * @param {Object} value (Optional)\n * @param {Object} options Overrides for service options\n *\n * @return {*} The saved value\n *\n * @example\n * cs.set('person', { firstName: 'john', lastName: 'smith' });\n * cs.set({ name:'smith', age:'32' });\n */\n set: function (key, value, options) {\n var setOptions = $.extend(true, {}, this.serviceOptions, options);\n\n var domain = setOptions.domain;\n var path = setOptions.root;\n var cookie = setOptions.cookie;\n\n cookie.set(encodeURIComponent(key) + '=' +\n encodeURIComponent(value) +\n (domain ? '; domain=' + domain : '') +\n (path ? '; path=' + path : '')\n );\n\n return value;\n },\n\n /**\n * Load cookie value\n * @param { string|Object} key If given a key save values under it, if given an object directly, save to top-level api\n * @return {*} The value stored\n *\n * @example\n * cs.get('person');\n */\n get: function (key) {\n var cookie = this.serviceOptions.cookie;\n var cookieReg = new RegExp('(?:^|;)\\\\s*' + encodeURIComponent(key).replace(/[\\-\\.\\+\\*]/g, '\\\\$&') + '\\\\s*\\\\=\\\\s*([^;]*).*$');\n var res = cookieReg.exec(cookie.get());\n var val = res ? decodeURIComponent(res[1]) : null;\n return val;\n },\n\n /**\n * Removes key from collection\n * @param { string} key key to remove\n * @param {object} options (optional) overrides for service options\n * @return { string} key The key removed\n *\n * @example\n * cs.remove('person');\n */\n remove: function (key, options) {\n var remOptions = $.extend(true, {}, this.serviceOptions, options);\n\n var domain = remOptions.domain;\n var path = remOptions.root;\n var cookie = remOptions.cookie;\n\n cookie.set(encodeURIComponent(key) +\n '=; expires=Thu, 01 Jan 1970 00:00:00 GMT' +\n (domain ? '; domain=' + domain : '') +\n (path ? '; path=' + path : '')\n );\n return key;\n },\n\n /**\n * Removes collection being referenced\n * @return { array} keys All the keys removed\n */\n destroy: function () {\n var cookie = this.serviceOptions.cookie;\n var aKeys = cookie.get().replace(/((?:^|\\s*;)[^\\=]+)(?=;|$)|^\\s*|\\s*(?:\\=[^;]*)?(?:\\1|$)/g, '').split(/\\s*(?:\\=[^;]*)?;\\s*/);\n for (var nIdx = 0; nIdx < aKeys.length; nIdx++) {\n var cookieKey = decodeURIComponent(aKeys[nIdx]);\n this.remove(cookieKey);\n }\n return aKeys;\n }\n };\n\n $.extend(this, publicAPI);\n};\n","'use strict';\n\nvar keyNames = require('../managers/key-names');\nvar StorageFactory = require('./store-factory');\nvar optionUtils = require('../util/option-utils');\n\nvar EPI_SESSION_KEY = keyNames.EPI_SESSION_KEY;\nvar EPI_MANAGER_KEY = 'epicenter.token'; //can't be under key-names, or logout will clear this too\n\nvar defaults = {\n /**\n * Where to store user access tokens for temporary access. Defaults to storing in a cookie in the browser.\n * @type {string}\n */\n store: { synchronous: true }\n};\n\nvar SessionManager = function (managerOptions) {\n managerOptions = managerOptions || {};\n function getBaseOptions(overrides) {\n overrides = overrides || {};\n var libOptions = optionUtils.getOptions();\n var finalOptions = $.extend(true, {}, defaults, libOptions, managerOptions, overrides);\n return finalOptions;\n }\n\n function getStore(overrides) {\n var baseOptions = getBaseOptions(overrides);\n var storeOpts = baseOptions.store || {};\n var isEpicenterDomain = !baseOptions.isLocal && !baseOptions.isCustomDomain;\n if (storeOpts.root === undefined && baseOptions.account && baseOptions.project && isEpicenterDomain) {\n storeOpts.root = '/app/' + baseOptions.account + '/' + baseOptions.project;\n }\n return new StorageFactory(storeOpts);\n }\n\n var publicAPI = {\n saveSession: function (userInfo, options) {\n var serialized = JSON.stringify(userInfo);\n getStore(options).set(EPI_SESSION_KEY, serialized);\n },\n getSession: function (options) {\n var store = getStore(options);\n var finalOpts = store.serviceOptions;\n var serialized = store.get(EPI_SESSION_KEY) || '{}';\n var session = JSON.parse(serialized);\n // If the url contains the project and account\n // validate the account and project in the session\n // and override project, groupName, groupId and isFac\n // Otherwise (i.e. localhost) use the saved session values\n var account = finalOpts.account;\n var project = finalOpts.project;\n if (account && session.account !== account) {\n // This means that the token was not used to login to the same account\n return {};\n }\n if (session.groups && account && project) {\n var group = session.groups[project] || { groupId: '', groupName: '', isFac: false };\n $.extend(session, { project: project }, group);\n }\n return session;\n },\n removeSession: function (options) {\n var store = getStore(options);\n Object.keys(keyNames).forEach(function (cookieKey) {\n var cookieName = keyNames[cookieKey];\n store.remove(cookieName);\n });\n return true;\n },\n getStore: function (options) {\n return getStore(options);\n },\n\n getMergedOptions: function () {\n var args = Array.prototype.slice.call(arguments);\n var overrides = $.extend.apply($, [true, {}].concat(args));\n var baseOptions = getBaseOptions(overrides);\n var session = this.getSession(overrides);\n\n var token = session.auth_token;\n if (!token) {\n var factory = new StorageFactory();\n token = factory.get(EPI_MANAGER_KEY);\n }\n\n var sessionDefaults = {\n /**\n * For projects that require authentication, pass in the user access token (defaults to empty string). If the user is already logged in to Epicenter, the user access token is already set in a cookie and automatically loaded from there. (See [more background on access tokens](../../../project_access/)).\n * @see [Authentication API Service](../auth-api-service/) for getting tokens.\n * @type {String}\n */\n token: token,\n\n /**\n * The account. If left undefined, taken from the cookie session.\n * @type {String}\n */\n account: session.account,\n\n /**\n * The project. If left undefined, taken from the cookie session.\n * @type {String}\n */\n project: session.project,\n\n\n /**\n * The group name. If left undefined, taken from the cookie session.\n * @type {String}\n */\n group: session.groupName,\n /**\n * Alias for group. \n * @type {String}\n */\n groupName: session.groupName, //It's a little weird that it's called groupName in the cookie, but 'group' in all the service options, so normalize for both\n /**\n * The group id. If left undefined, taken from the cookie session.\n * @type {String}\n */\n groupId: session.groupId,\n userId: session.userId,\n userName: session.userName,\n };\n return $.extend(true, sessionDefaults, baseOptions);\n }\n };\n $.extend(this, publicAPI);\n};\n\nmodule.exports = SessionManager;","/**\n Decides type of store to provide\n*/\n\n'use strict';\n// var isNode = false; FIXME: Browserify/minifyify has issues with the next link\n// var store = (isNode) ? require('./session-store') : require('./cookie-store');\nvar store = require('./cookie-store');\n\nmodule.exports = store;\n","'use strict';\n\nvar qutils = require('../util/query-util');\n\nmodule.exports = function (config) {\n\n var defaults = {\n url: '',\n\n contentType: 'application/json',\n headers: {},\n statusCode: {\n 404: $.noop\n },\n\n /**\n * ONLY for strings in the url. All GET & DELETE params are run through this\n * @type {[type] }\n */\n parameterParser: qutils.toQueryFormat,\n\n // To allow epicenter.token and other session cookies to be passed\n // with the requests\n xhrFields: {\n withCredentials: true\n }\n };\n\n var transportOptions = $.extend({}, defaults, config);\n\n var result = function (d) {\n return ($.isFunction(d)) ? d() : d;\n };\n\n var connect = function (method, params, connectOptions) {\n params = result(params);\n params = ($.isPlainObject(params) || $.isArray(params)) ? JSON.stringify(params) : params;\n\n var options = $.extend(true, {}, transportOptions, connectOptions, {\n type: method,\n data: params\n });\n var ALLOWED_TO_BE_FUNCTIONS = ['data', 'url'];\n $.each(options, function (key, value) {\n if ($.isFunction(value) && $.inArray(key, ALLOWED_TO_BE_FUNCTIONS) !== -1) {\n options[key] = value();\n }\n });\n\n if (options.logLevel && options.logLevel === 'DEBUG') {\n console.log(options.url);\n var oldSuccessFn = options.success || $.noop;\n options.success = function (response, ajaxStatus, ajaxReq) {\n console.log(response);\n oldSuccessFn.apply(this, arguments);\n };\n }\n\n var beforeSend = options.beforeSend;\n options.beforeSend = function (xhr, settings) {\n xhr.requestUrl = (connectOptions || {}).url;\n if (beforeSend) {\n beforeSend.apply(this, arguments);\n }\n };\n\n return $.ajax(options);\n };\n\n var publicAPI = {\n get: function (params, ajaxOptions) {\n var options = $.extend({}, transportOptions, ajaxOptions);\n params = options.parameterParser(result(params));\n return connect.call(this, 'GET', params, options);\n },\n splitGet: function () {\n\n },\n post: function () {\n return connect.apply(this, ['post'].concat([].slice.call(arguments)));\n },\n patch: function () {\n return connect.apply(this, ['patch'].concat([].slice.call(arguments)));\n },\n put: function () {\n return connect.apply(this, ['put'].concat([].slice.call(arguments)));\n },\n delete: function (params, ajaxOptions) {\n //DELETE doesn't support body params, but jQuery thinks it does.\n var options = $.extend({}, transportOptions, ajaxOptions);\n params = options.parameterParser(result(params));\n if ($.trim(params)) {\n var delimiter = (result(options.url).indexOf('?') === -1) ? '?' : '&';\n options.url = result(options.url) + delimiter + params;\n }\n return connect.call(this, 'DELETE', null, options);\n },\n head: function () {\n return connect.apply(this, ['head'].concat([].slice.call(arguments)));\n },\n options: function () {\n return connect.apply(this, ['options'].concat([].slice.call(arguments)));\n }\n };\n\n return $.extend(this, publicAPI);\n};\n","'use strict';\n\n// var isNode = false; FIXME: Browserify/minifyify has issues with the next link\n// var transport = (isNode) ? require('./node-http-transport') : require('./ajax-http-transport');\nvar transport = require('./ajax-http-transport');\nmodule.exports = transport;\n","/**\n/* Inherit from a class (using prototype borrowing)\n*/\n'use strict';\n\nfunction inherit(C, P) {\n var F = function () {};\n F.prototype = P.prototype;\n C.prototype = new F();\n C.__super = P.prototype;\n C.prototype.constructor = C;\n}\n\n/**\n* Shallow copy of an object\n* @param {Object} dest object to extend\n* @return {Object} extended object\n*/\nvar extend = function (dest /*, var_args*/) {\n var obj = Array.prototype.slice.call(arguments, 1);\n var current;\n for (var j = 0; j < obj.length; j++) {\n if (!(current = obj[j])) { //eslint-disable-line\n continue;\n }\n\n // do not wrap inner in dest.hasOwnProperty or bad things will happen\n for (var key in current) { //eslint-disable-line\n dest[key] = current[key];\n }\n }\n\n return dest;\n};\n\nmodule.exports = function (base, props, staticProps) {\n var parent = base;\n var child;\n\n child = props && props.hasOwnProperty('constructor') ? props.constructor : function () { return parent.apply(this, arguments); };\n\n // add static properties to the child constructor function\n extend(child, parent, staticProps);\n\n // associate prototype chain\n inherit(child, parent);\n\n // add instance properties\n if (props) {\n extend(child.prototype, props);\n }\n\n // done\n return child;\n};\n","'use strict';\n\nmodule.exports = {\n _pick: function (obj, props) {\n var res = {};\n for (var p in obj) {\n if (props.indexOf(p) !== -1) {\n res[p] = obj[p];\n }\n }\n\n return res;\n },\n isEmpty: function isEmpty(value) {\n return (!value || ($.isPlainObject(value) && Object.keys(value).length === 0));\n }\n};\n","'use strict';\n\nvar ConfigService = require('../service/configuration-service');\n\nvar urlConfig = new ConfigService().get('server');\nvar customDefaults = {};\nvar libDefaults = {\n /**\n * The account id. In the Epicenter UI, this is the **Team ID** (for team projects) or **User ID** (for personal projects). Defaults to empty string. If left undefined, taken from the URL.\n * @type {String}\n */\n account: urlConfig.accountPath || undefined,\n /**\n * The account id. In the Epicenter UI, this is the **Team ID** (for team projects) or **User ID** (for personal projects). Defaults to empty string. If left undefined, taken from the URL.\n * @type {String}\n */\n project: urlConfig.projectPath || undefined,\n isLocal: urlConfig.isLocalhost(),\n isCustomDomain: urlConfig.isCustomDomain,\n store: {}\n};\n\nvar optionUtils = {\n /**\n * Gets the final options by overriding the global options set with\n * optionUtils#setDefaults() and the lib defaults.\n * @param {object} options The final options object.\n * @return {object} Extended object\n */\n getOptions: function (options) {\n return $.extend(true, {}, libDefaults, customDefaults, options);\n },\n /**\n * Sets the global defaults for the optionUtils#getOptions() method.\n * @param {object} defaults The defaults object.\n */\n setDefaults: function (defaults) {\n customDefaults = defaults;\n }\n};\nmodule.exports = optionUtils;\n","/**\n * Utilities for working with query strings\n*/\n'use strict';\n\nmodule.exports = (function () {\n\n return {\n /**\n * Converts to matrix format\n * @param {Object} qs Object to convert to query string\n * @return { string} Matrix-format query parameters\n */\n toMatrixFormat: function (qs) {\n if (qs === null || qs === undefined || qs === '') {\n return ';';\n }\n if (typeof qs === 'string' || qs instanceof String) {\n return qs;\n }\n\n var returnArray = [];\n var OPERATORS = ['<', '>', '!'];\n $.each(qs, function (key, value) {\n if (typeof value !== 'string' || $.inArray($.trim(value).charAt(0), OPERATORS) === -1) {\n value = '=' + value;\n }\n returnArray.push(key + value);\n });\n\n var mtrx = ';' + returnArray.join(';');\n return mtrx;\n },\n\n /**\n * Converts strings/arrays/objects to type 'a=b&b=c'\n * @param { string|Array|Object} qs\n * @return { string}\n */\n toQueryFormat: function (qs) {\n if (qs === null || qs === undefined) {\n return '';\n }\n if (typeof qs === 'string' || qs instanceof String) {\n return qs;\n }\n\n var returnArray = [];\n $.each(qs, function (key, value) {\n if ($.isArray(value)) {\n value = value.join(',');\n }\n if ($.isPlainObject(value)) {\n //Mostly for data api\n value = JSON.stringify(value);\n }\n returnArray.push(key + '=' + value);\n });\n\n var result = returnArray.join('&');\n return result;\n },\n\n /**\n * Converts strings of type 'a=b&b=c' to { a:b, b:c}\n * @param { string} qs\n * @return {object}\n */\n qsToObject: function (qs) {\n if (qs === null || qs === undefined || qs === '') {\n return {};\n }\n\n var qsArray = qs.split('&');\n var returnObj = {};\n $.each(qsArray, function (index, value) {\n var qKey = value.split('=')[0];\n var qVal = value.split('=')[1];\n\n if (qVal.indexOf(',') !== -1) {\n qVal = qVal.split(',');\n }\n\n returnObj[qKey] = qVal;\n });\n\n return returnObj;\n },\n\n /**\n * Normalizes and merges strings of type 'a=b', { b:c} to { a:b, b:c}\n * @param { string|Array|Object} qs1\n * @param { string|Array|Object} qs2\n * @return {Object}\n */\n mergeQS: function (qs1, qs2) {\n var obj1 = this.qsToObject(this.toQueryFormat(qs1));\n var obj2 = this.qsToObject(this.toQueryFormat(qs2));\n return $.extend(true, {}, obj1, obj2);\n },\n\n addTrailingSlash: function (url) {\n if (!url) {\n return '';\n }\n return (url.charAt(url.length - 1) === '/') ? url : (url + '/');\n }\n };\n}());\n\n\n","/**\n * Utilities for working with the run service\n*/\n'use strict';\nvar qutil = require('./query-util');\nvar MAX_URL_LENGTH = 2048;\n\nmodule.exports = (function () {\n return {\n /**\n * normalizes different types of operation inputs\n * @param {Object|Array|String} operations operations to perform\n * @param {Array} args arguments for operation\n * @return {String} operations of the form `{ ops: [], args: [] }`\n */\n normalizeOperations: function (operations, args) {\n if (!args) {\n args = [];\n }\n var returnList = {\n ops: [],\n args: []\n };\n\n var _concat = function (arr) {\n return (arr !== null && arr !== undefined) ? [].concat(arr) : [];\n };\n\n //{ add: [1,2], subtract: [2,4] }\n var _normalizePlainObjects = function (operations, returnList) {\n if (!returnList) {\n returnList = { ops: [], args: [] };\n }\n $.each(operations, function (opn, arg) {\n returnList.ops.push(opn);\n returnList.args.push(_concat(arg));\n });\n return returnList;\n };\n //{ name: 'add', params: [1] }\n var _normalizeStructuredObjects = function (operation, returnList) {\n if (!returnList) {\n returnList = { ops: [], args: [] };\n }\n returnList.ops.push(operation.name);\n returnList.args.push(_concat(operation.params));\n return returnList;\n };\n\n var _normalizeObject = function (operation, returnList) {\n return ((operation.name) ? _normalizeStructuredObjects : _normalizePlainObjects)(operation, returnList);\n };\n\n var _normalizeLiterals = function (operation, args, returnList) {\n if (!returnList) {\n returnList = { ops: [], args: [] };\n }\n returnList.ops.push(operation);\n returnList.args.push(_concat(args));\n return returnList;\n };\n\n\n var _normalizeArrays = function (operations, arg, returnList) {\n if (!returnList) {\n returnList = { ops: [], args: [] };\n }\n $.each(operations, function (index, opn) {\n if ($.isPlainObject(opn)) {\n _normalizeObject(opn, returnList);\n } else {\n _normalizeLiterals(opn, args[index], returnList);\n }\n });\n return returnList;\n };\n\n if ($.isPlainObject(operations)) {\n _normalizeObject(operations, returnList);\n } else if ($.isArray(operations)) {\n _normalizeArrays(operations, args, returnList);\n } else {\n _normalizeLiterals(operations, args, returnList);\n }\n\n return returnList;\n },\n\n splitGetFactory: function (httpOptions) {\n return function (params, options) {\n var http = this; //eslint-disable-line\n var getValue = function (name) {\n var value = options[name] || httpOptions[name];\n if (typeof value === 'function') {\n value = value();\n }\n return value;\n };\n var getFinalUrl = function (params) {\n var url = getValue('url', options);\n var data = params;\n // There is easy (or known) way to get the final URL jquery is going to send so\n // we're replicating it. The process might change at some point but it probably will not.\n // 1. Remove hash\n url = url.replace(/#.*$/, '');\n // 1. Append query string\n var queryParams = qutil.toQueryFormat(data);\n var questionIdx = url.indexOf('?');\n if (queryParams && questionIdx > -1) {\n return url + '&' + queryParams;\n } else if (queryParams) {\n return url + '?' + queryParams;\n }\n return url;\n };\n var url = getFinalUrl(params);\n // We must split the GET in multiple short URL's\n // The only property allowed to be split is \"include\"\n if (params && params.include && encodeURI(url).length > MAX_URL_LENGTH) {\n var dtd = $.Deferred();\n var paramsCopy = $.extend(true, {}, params);\n delete paramsCopy.include;\n var urlNoIncludes = getFinalUrl(paramsCopy);\n var diff = MAX_URL_LENGTH - urlNoIncludes.length;\n var oldSuccess = options.success || httpOptions.success || $.noop;\n var oldError = options.error || httpOptions.error || $.noop;\n // remove the original success and error callbacks\n options.success = $.noop;\n options.error = $.noop;\n\n var include = params.include;\n var currIncludes = [];\n var includeOpts = [currIncludes];\n var currLength = encodeURIComponent('?include=').length;\n var variable = include.pop();\n while (variable) {\n var varLenght = encodeURIComponent(variable).length;\n // Use a greedy approach for now, can be optimized to be solved in a more\n // efficient way\n // + 1 is the comma\n if (currLength + varLenght + 1 < diff) {\n currIncludes.push(variable);\n currLength += varLenght + 1;\n } else {\n currIncludes = [variable];\n includeOpts.push(currIncludes);\n currLength = '?include='.length + varLenght;\n }\n variable = include.pop();\n }\n var reqs = $.map(includeOpts, function (include) {\n var reqParams = $.extend({}, params, { include: include });\n return http.get(reqParams, options);\n });\n $.when.apply($, reqs).then(function () {\n // Each argument are arrays of the arguments of each done request\n // So the first argument of the first array of arguments is the data\n var isValid = arguments[0] && arguments[0][0];\n if (!isValid) {\n // Should never happen...\n oldError();\n return dtd.reject();\n }\n var firstResponse = arguments[0][0];\n var isObject = $.isPlainObject(firstResponse);\n var isRunAPI = (isObject && $.isPlainObject(firstResponse.variables)) || !isObject;\n if (isRunAPI) {\n if (isObject) {\n // aggregate the variables property only\n var aggregateRun = arguments[0][0];\n $.each(arguments, function (idx, args) {\n var run = args[0];\n $.extend(true, aggregateRun.variables, run.variables);\n });\n oldSuccess(aggregateRun, arguments[0][1], arguments[0][2]);\n dtd.resolve(aggregateRun, arguments[0][1], arguments[0][2]);\n } else {\n // array of runs\n // Agregate variables in each run\n var aggregatedRuns = {};\n $.each(arguments, function (idx, args) {\n var runs = args[0];\n if (!$.isArray(runs)) {\n return;\n }\n $.each(runs, function (idxRun, run) {\n if (run.id && !aggregatedRuns[run.id]) {\n run.variables = run.variables || {};\n aggregatedRuns[run.id] = run;\n } else if (run.id) {\n $.extend(true, aggregatedRuns[run.id].variables, run.variables);\n }\n });\n });\n // turn it into an array\n aggregatedRuns = $.map(aggregatedRuns, function (run) { return run; });\n oldSuccess(aggregatedRuns, arguments[0][1], arguments[0][2]);\n dtd.resolve(aggregatedRuns, arguments[0][1], arguments[0][2]);\n }\n } else {\n // is variables API\n // aggregate the response\n var aggregatedVariables = {};\n $.each(arguments, function (idx, args) {\n var vars = args[0];\n $.extend(true, aggregatedVariables, vars);\n });\n oldSuccess(aggregatedVariables, arguments[0][1], arguments[0][2]);\n dtd.resolve(aggregatedVariables, arguments[0][1], arguments[0][2]);\n }\n }, function () {\n oldError.apply(http, arguments);\n dtd.reject.apply(dtd, arguments);\n });\n return dtd.promise();\n } else {\n return http.get(params, options);\n }\n };\n }\n };\n}());\n"]} \ No newline at end of file