diff --git a/dist/epicenter.js b/dist/epicenter.js index 02a122a2..e2abf5c9 100644 --- a/dist/epicenter.js +++ b/dist/epicenter.js @@ -5903,6 +5903,8 @@ var StorageFactory = require('./store-factory'); var optionUtils = require('../util/option-utils'); var EPI_SESSION_KEY = keyNames.EPI_SESSION_KEY; +var EPI_MANAGER_KEY = 'epicenter.token'; //can't be under key-names, or logout will clear this too + var defaults = { /** * Where to store user access tokens for temporary access. Defaults to storing in a cookie in the browser. @@ -5976,13 +5978,19 @@ var SessionManager = function (managerOptions) { var baseOptions = getBaseOptions(overrides); var session = this.getSession(overrides); + var token = session.auth_token; + if (!token) { + var factory = new StorageFactory(); + token = factory.get(EPI_MANAGER_KEY); + } + var sessionDefaults = { /** * 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/)). * @see [Authentication API Service](../auth-api-service/) for getting tokens. * @type {String} */ - token: session.auth_token, + token: token, /** * The account. If left undefined, taken from the cookie session. @@ -6604,4 +6612,4 @@ module.exports = (function () { }()); },{"./query-util":47}]},{},[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/always-new-strategy.js","src/managers/run-strategies/conditional-creation-strategy.js","src/managers/run-strategies/multiplayer-strategy.js","src/managers/run-strategies/new-if-initialized-strategy.js","src/managers/run-strategies/new-if-missing-strategy.js","src/managers/run-strategies/new-if-persisted-strategy.js","src/managers/run-strategies/none-strategy.js","src/managers/run-strategies/persistent-single-player-strategy.js","src/managers/run-strategies/strategies-map.js","src/managers/scenario-manager.js","src/managers/special-operations.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/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;;;;ACtEA;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;;ACjYA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7PA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtYA;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;;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;;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;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;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;;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;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACTA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjDA;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;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvLA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;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;;ACxOA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;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;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5hBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;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;;AC5HA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;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;;ACrKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrvBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;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;;AC5HA;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;;ACdA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;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    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');\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');\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');\n\nF.manager.strategy['always-new'] = require('./managers/run-strategies/always-new-strategy');\nF.manager.strategy['conditional-creation'] = require('./managers/run-strategies/conditional-creation-strategy');\nF.manager.strategy.identity = require('./managers/run-strategies/none-strategy');\nF.manager.strategy['new-if-missing'] = require('./managers/run-strategies/new-if-missing-strategy');\nF.manager.strategy['new-if-missing'] = require('./managers/run-strategies/new-if-missing-strategy');\nF.manager.strategy['new-if-persisted'] = require('./managers/run-strategies/new-if-persisted-strategy');\nF.manager.strategy['new-if-initialized'] = require('./managers/run-strategies/new-if-initialized-strategy');\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\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) {\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 });\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 sessionInfo = {\n                auth_token: token,\n                account: adapterOptions.account,\n                project: project,\n                userId: userInfo.user_id,\n                groups: oldGroups,\n                isTeamMember: isTeamMember\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', 401, data);\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);\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                        handleGroupList(groups);\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     * 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 * 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 * While you can work directly with the Channel Manager through Node.js (for example, `require('manager/channel-manager')`) -- or even work directly with `$.cometd` and Epicenter's underlying [Push Channel API](../../../rest_apis/multiplayer/channel/) -- most often it will be easiest to work with the [Epicenter Channel Manager](../epicenter-channel-manager/). The Epicenter Channel Manager is a wrapper that instantiates a Channel Manager with Epicenter-specific defaults.\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 the channel, 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 channel = cm.getChannel();\n *\n *        channel.subscribe('topic', callback);\n *        channel.publish('topic', { myData: 100 });\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. See 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). There are two main use cases for the channel: event notifications and chat messages.\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 probably 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 * To use the Epicenter Channel Manager: instantiate it, get the channel of the scope you want ([user](../../../glossary/#users), [world](../../../glossary/#world), or [group](../../../glossary/#groups)), 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 *     gc.subscribe('broadcasts', callback);\n *\n * For additional background on Epicenter's push channel, see the introductory notes on the [Push Channel API](../../../rest_apis/multiplayer/channel/) page.\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 classFrom = require('../util/inherit');\nvar urlService = require('../service/url-config-service');\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};\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 urlOpts = urlService(defaultCometOptions.server);\n        if (!defaultCometOptions.url) {\n            //Default epicenter cometd endpoint\n            defaultCometOptions.url = urlOpts.protocol + '://' + urlOpts.host + '/channel/subscribe';\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        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 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 and world. 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 shared world are also online.\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     *         model: 'model.eqn'\n     *     });\n     *     worldManager.getCurrentWorld().then(function (worldObject, worldService) {\n     *         var presenceChannel = cm.getPresenceChannel(worldObject);\n     *         presenceChannel.on('presence', function (evt, notification) {\n     *              console.log(notification.online, notification.userId);\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} userid (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    getPresenceChannel: function (world, userid, 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\n        var session = this.sessionManager.getMergedOptions(this.options);\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].join('/');\n        var channel = __super.getChannel.call(this, { base: baseTopic });\n\n        var lastPingTime = { };\n\n        var PING_INTERVAL = 6000;\n        channel.subscribe('internal-ping-channel', function (notification) {\n            var incomingUserId = notification.data.user;\n            if (!lastPingTime[incomingUserId] && incomingUserId !== userid) {\n                channel.trigger('presence', { userId: incomingUserId, online: true });\n            }\n            lastPingTime[incomingUserId] = (new Date()).valueOf();\n        });\n\n        setInterval(function () {\n            channel.publish('internal-ping-channel', { user: userid });\n\n            $.each(lastPingTime, function (key, value) {\n                var now = (new Date()).valueOf();\n                if (value && value + (PING_INTERVAL * 2) < now) {\n                    lastPingTime[key] = null;\n                    channel.trigger('presence', { userId: key, online: false });\n                }\n            });\n        }, PING_INTERVAL);\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                };\n                var actualData = payload.data.data;\n                if (actualData.data) { //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/), the RESFTful [Run API](../../../rest_apis/aggregate_run_api) and the [Model Run API](../../../rest_apis/other_apis/model_apis/run/). 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. (Note that many of the Epicenter sample projects use a Run Service directly, because generally the sample projects are played in one end user session and don't care about run states or run strategies.)\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 pass a `files` object with the names of the files, for example: `\"files\": {\"data\": \"myExtraData.xls\"}`. (Note that you'll also need to add this same files object to your Vensim [configuration file](../../../model_code/vensim/).) See the [underlying Model Run API](../../../rest_apis/other_apis/model_apis/run/#post-creating-a-new-run-for-this-project) for additional information.\n*\n*   * `strategy`: (optional) Run creation strategy for when to create a new run and when to reuse an end user's existing run. See [Run Manager Strategies](../strategies/) for details. Defaults to `new-if-initialized`.\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.\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: 'always-new',\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\n'use strict';\nvar strategiesMap = require('./run-strategies/strategies-map');\nvar specialOperations = require('./special-operations');\nvar RunService = require('../service/run-api-service');\n\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\n\nvar defaults = {\n    /**\n     * Run creation strategy for when to create a new run and when to reuse an end user's existing run. See [Run Manager Strategies](../strategies/) for details. Defaults to `new-if-initialized`.\n     * @type {String}\n     */\n\n    strategy: 'new-if-initialized'\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 {\n        this.run = new RunService(this.options.run);\n    }\n\n    patchRunService(this.run, this);\n\n    var StrategyCtor = typeof this.options.strategy === 'function' ? this.options.strategy : strategiesMap[this.options.strategy];\n\n    if (!StrategyCtor) {\n        throw new Error('Specified run creation strategy was invalid:', this.options.strategy);\n    }\n\n    this.strategy = new StrategyCtor(this.run, this.options);\n}\n\nRunManager.prototype = {\n    /**\n     * Returns the run object for a 'good' run.\n     *\n     * A good run is defined by the strategy. For example, if the strategy is `always-new`, the call\n     * to `getRun()` always returns a newly created run; if the strategy is `new-if-persisted`,\n     * `getRun()` creates a new run if the previous run is in a persisted state, otherwise\n     * it returns the previous run. 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     * @return {$promise} Promise to complete the call.\n     */\n    getRun: function () {\n        return this.strategy\n                .getRun();\n    },\n\n    /**\n     * Returns the run object for a new run, regardless of strategy: force creation of a new run.\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} runServiceOptions The options object to configure the Run Service. See [Run API Service](../run-api-service/) for more.\n     * @return {Promise}\n     */\n    reset: function (runServiceOptions) {\n        return this.strategy.reset(runServiceOptions);\n    }\n};\n\nmodule.exports = RunManager;\n","/**\n * The `always-new` 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\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 (runService, options) {\n        __super.constructor.call(this, runService, 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","'use strict';\n\nvar Base = require('./none-strategy');\nvar SessionManager = require('../../store/session-manager');\nvar classFrom = require('../../util/inherit');\nvar AuthManager = require('../auth-manager');\n\nvar keyNames = require('../key-names');\n\nvar defaults = {\n    sessionKey: keyNames.STRATEGY_SESSION_KEY,\n    path: ''\n};\n\nfunction setRunInSession(sessionKey, run, sessionManager) {\n    sessionManager.getStore().set(sessionKey, JSON.stringify({ runId: run.id }));\n}\n\n/**\n* Conditional Creation Strategy\n* This strategy will try to get the run stored in the cookie and\n* evaluate if needs to create a new run by calling the 'condition' function\n*/\n\nvar Strategy = classFrom(Base, {\n    constructor: function Strategy(runService, condition, options) {\n        if (condition == null) { //eslint-disable-line\n            //TODO: not sure why this is explicitly ==\n            throw new Error('Conditional strategy needs a condition to create a run');\n        }\n\n        this._auth = new AuthManager();\n        this.run = runService;\n        this.condition = typeof condition !== 'function' ? function () { return condition; } : condition;\n        this.options = $.extend(true, {}, defaults, options);\n        this.sessionManager = new SessionManager(options);\n        this.runOptions = this.options.run;\n    },\n\n    runOptionsWithScope: function () {\n        var userSession = this._auth.getCurrentUserSessionInfo();\n        return $.extend({\n            scope: { group: userSession.groupName }\n        }, this.runOptions);\n    },\n\n    reset: function (runServiceOptions) {\n        var me = this;\n        var opt = this.runOptionsWithScope();\n\n        return this.run\n                .create(opt, runServiceOptions)\n            .then(function (run) {\n                setRunInSession(me.options.sessionKey, run, me.sessionManager);\n                run.freshlyCreated = true;\n                return run;\n            });\n    },\n\n    getRun: function () {\n        var sessionStore = this.sessionManager.getStore();\n        var runSession = JSON.parse(sessionStore.get(this.options.sessionKey));\n        var me = this;\n        if (runSession && runSession.runId) {\n            return this._loadAndCheck(runSession).fail(function () {\n                return me.reset(); //if it got the wrong cookie for e.g.\n            });\n        } else {\n            return this.reset();\n        }\n    },\n\n    _loadAndCheck: function (runSession) {\n        var shouldCreate = false;\n        var me = this;\n\n        return this.run\n            .load(runSession.runId, null, {\n                success: function (run, msg, headers) {\n                    shouldCreate = me.condition(run, headers);\n                }\n            })\n            .then(function (run) {\n                if (shouldCreate) {\n                    var opt = me.runOptionsWithScope();\n                    return me.run.create(opt)\n                    .then(function (run) {\n                        setRunInSession(me.options.sessionKey, run, me.sessionManager);\n                        run.freshlyCreated = true;\n                        return run;\n                    });\n                }\n                return run;\n            });\n    }\n});\n\nmodule.exports = Strategy;\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/).\n * \n * Using this strategy means that end users in projects with multiplayer worlds always see the most current run and world. 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');\nvar AuthManager = require('../auth-manager');\n\nvar defaults = {\n    store: {\n        synchronous: true\n    }\n};\n\nvar Strategy = classFrom(IdentityStrategy, {\n\n    constructor: function (runService, options) {\n        this.runService = runService;\n        this.options = $.extend(true, {}, defaults, options);\n        this._auth = new AuthManager();\n        this._loadRun = this._loadRun.bind(this);\n        this.worldApi = new WorldApiAdapter(this.options.run);\n    },\n\n    reset: function () {\n        var session = this._auth.getCurrentUserSessionInfo();\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);\n            }.bind(this));\n    },\n\n    getRun: function () {\n        var session = this._auth.getCurrentUserSessionInfo();\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\n            return worldApi.getCurrentRunId({ model: model, filter: world.id })\n                .then(me._loadRun)\n                .then(dtd.resolve)\n                .fail(dtd.reject);\n        };\n\n        var serverError = function (error) {\n            // is this possible?\n            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    _loadRun: function (id, options) {\n        return this.runService.load(id, null, options);\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 for Vensim models; it can be set manually for other models.\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\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 (runService, options) {\n        __super.constructor.call(this, runService, this.createIf, options);\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-missing` 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.\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 other words, you care whether you have a run, but as long as you have one, you are certain that this run is the one you are interested in. \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\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 (runService, options) {\n        __super.constructor.call(this, runService, 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 * 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, if they are idle for longer than the **Model Session Timeout**, that 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\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 (runService, options) {\n        __super.constructor.call(this, runService, this.createIf, options);\n    },\n\n    createIf: function (run, headers) {\n        return headers.getResponseHeader('pragma') === 'persistent';\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 * Also, this strategy is necessary if you are working with a multiplayer project and using the [World Manager](../world-manager/) &mdash; or other, similar situations where you do not have direct control over creating the [Run Service](../run-api-service/) instance.\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 (runService, options) {\n        this.runService = runService;\n    },\n\n    reset: function () {\n        // return a newly created run\n        return $.Deferred().resolve().promise();\n    },\n\n    getRun: function () {\n        // return a usable run\n        return $.Deferred().resolve(this.runService).promise();\n    }\n});\n","/**\n * The `persistent-single-player` 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 your project executes your 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). It is useful if end users play with your project for an extended period of time, possibly over several sessions.\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 *         * If the most recent run is currently in the database, bring it back into memory so that the end user can continue working with it. (See more background on [Run Persistence](../../../run_persistence/), or read more on the underlying [State API](../../../rest_apis/other_apis/model_apis/state/) for bringing runs from the database back into memory.) \n */\n\n'use strict';\n\nvar classFrom = require('../../util/inherit');\nvar IdentityStrategy = require('./none-strategy');\nvar StorageFactory = require('../../store/store-factory');\nvar StateApi = require('../../service/state-api-adapter');\nvar AuthManager = require('../auth-manager');\n\nvar keyNames = require('../key-names');\n\nvar defaults = {\n    store: {\n        synchronous: true\n    }\n};\n\nvar Strategy = classFrom(IdentityStrategy, {\n    constructor: function Strategy(runService, options) {\n        this.run = runService;\n        this.options = $.extend(true, {}, defaults, options);\n        this.runOptions = this.options.run;\n        this._store = new StorageFactory(this.options.store);\n        this.stateApi = new StateApi();\n        this._auth = new AuthManager();\n\n        this._loadAndCheck = this._loadAndCheck.bind(this);\n        this._restoreRun = this._restoreRun.bind(this);\n        this._getAllRuns = this._getAllRuns.bind(this);\n        this._loadRun = this._loadRun.bind(this);\n    },\n\n    reset: function (runServiceOptions) {\n        var session = this._auth.getCurrentUserSessionInfo();\n        var opt = $.extend({\n            scope: { group: session.groupName }\n        }, this.runOptions);\n\n        return this.run\n            .create(opt, runServiceOptions)\n            .then(function (run) {\n                run.freshlyCreated = true;\n                return run;\n            });\n    },\n\n    getRun: function () {\n        return this._getAllRuns()\n            .then(this._loadAndCheck);\n    },\n\n    _getAllRuns: function () {\n        var session = JSON.parse(this._store.get(keyNames.EPI_SESSION_KEY) || '{}');\n        return this.run.query({\n            'user.id': session.userId || '0000',\n            'scope.group': session.groupName\n        });\n    },\n\n    _loadAndCheck: function (runs) {\n        if (!runs || !runs.length) {\n            return this.reset();\n        }\n\n        var dateComp = function (a, b) { return new Date(b.date) - new Date(a.date); };\n        var latestRun = runs.sort(dateComp)[0];\n        var me = this;\n        var shouldReplay = false;\n\n        return this.run.load(latestRun.id, null, {\n            success: function (run, msg, headers) {\n                shouldReplay = headers.getResponseHeader('pragma') === 'persistent';\n            }\n        }).then(function (run) {\n            return shouldReplay ? me._restoreRun(run.id) : run;\n        });\n    },\n\n    _restoreRun: function (runId) {\n        var me = this;\n        return this.stateApi.replay({ runId: runId })\n            .then(function (resp) {\n                return me._loadRun(resp.run);\n            });\n    },\n\n    _loadRun: function (id, options) {\n        return this.run.load(id, null, options);\n    }\n\n});\n\nmodule.exports = Strategy;\n","module.exports = {\n    'new-if-initialized': require('./new-if-initialized-strategy'),\n    'new-if-persisted': require('./new-if-persisted-strategy'),\n    'new-if-missing': require('./new-if-missing-strategy'),\n    'always-new': require('./always-new-strategy'),\n    multiplayer: require('./multiplayer-strategy'),\n    'persistent-single-player': require('./persistent-single-player-strategy'),\n    none: require('./none-strategy')\n};\n","'use strict';\nvar RunService = require('../service/run-api-service');\n\nvar defaults = {\n    validFilter: { saved: true }\n};\n\nfunction ScenarioManager(options) {\n    this.options = $.extend(true, {}, defaults, options);\n    this.runService = this.options.run || new RunService(this.options);\n}\n\nScenarioManager.prototype = {\n    getRuns: function (filter) {\n        this.filter = $.extend(true, {}, this.options.validFilter, filter);\n        return this.runService.query(this.filter);\n    },\n\n    loadVariables: function (vars) {\n        return this.runService.query(this.filter, { include: vars });\n    },\n\n    save: function (run, meta) {\n        return this._getService(run).save($.extend(true, {}, { saved: true }, meta));\n    },\n\n    archive: function (run) {\n        return this._getService(run).save({ saved: false });\n    },\n\n    _getService: function (run) {\n        if (typeof run === 'string') {\n            return new RunService($.extend(true, {}, this.options, { filter: run }));\n        }\n\n        if (typeof run === 'object' && run instanceof RunService) {\n            return run;\n        }\n\n        throw new Error('Save method requires a run service or a runId');\n    },\n\n    getRun: function (runId) {\n        return new RunService($.extend(true, {}, this.options, { filter: runId }));\n    }\n};\n\nmodule.exports = ScenarioManager;\n\n","'use strict';\n\n\nmodule.exports = {\n    reset: function (params, options, manager) {\n        return manager.reset(options);\n    }\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(runService, options) {\n        this.runService = runService;\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 () {\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 me.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({model: '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\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 * The Asset API Service (https://forio.com/epicenter/docs/public/api_adapters/generated/asset-api-adapter/) is typically used for all project use cases, and it's unlikely this File Service will be used directly except by Admin tools (e.g. Flow Inspector).\n *\n * Partially implemented.\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        var upload = uploadBody(fileName, contents);\n\n        return $.extend(true, {}, serviceOptions, options, {\n            url: urlConfig.getAPIPath('file') + path,\n            data: upload.body,\n            contentType: 'multipart/form-data; boundary=' + upload.boundary\n        });\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} 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\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} 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 * 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 * Typically, you use the [Epicenter Channel Manager](../epicenter-channel-manager/) to create or retrieve channels, then use the Channel Service `subscribe()` and `publish()` methods to listen to or update data. (For additional background on Epicenter's push channel, see the introductory notes on the [Push Channel API](../../../rest_apis/multiplayer/channel/) page.)\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 * 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 * ## 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 gives you control over run creation depending on run states.\n *\n * However, many of the Epicenter sample projects use a Run Service, because generally the sample projects are played in one end user session and don't care about run states or [run strategies](../strategies/). The Run API Service is also useful for building an interface for a facilitator, because it makes it easy to list data across multiple runs (using the `filter()` and `query()` methods).\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 * 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.\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 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         * Criteria by which to filter runs. Defaults to empty string.\n         * @type {String}\n         */\n        filter: '',\n\n        /**\n         * Convenience alias for filter.\n         * @type {String}\n         */\n        id: '',\n\n        /**\n         * Flag determines if `X-AutoRestore: true` header is sent to Epicenter. 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    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    urlConfig.filter = ';';\n    urlConfig.getFilterURL = function () {\n        var url = urlConfig.getAPIPath('run');\n        var filter = qutil.toMatrixFormat(serviceOptions.filter);\n\n        if (filter) {\n            url += filter + '/';\n        }\n        return url;\n    };\n\n    urlConfig.addAutoRestoreHeader = function (options) {\n        var filter = serviceOptions.filter;\n        // The semicolon separated filter is used when filter is an object\n        var isFilterRunId = filter && $.type(filter) === 'string';\n        if (serviceOptions.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\n    var 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    var http = new TransportFactory(httpOptions);\n    http.splitGet = rutil.splitGetFactory(httpOptions);\n\n    var setFilterOrThrowError = 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 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 `always-new`, 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'];\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 Julia model).\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            serviceOptions.filter = qs; //shouldn't be able to over-ride\n            var httpOptions = $.extend(true, {}, serviceOptions, options);\n            httpOptions = urlConfig.addAutoRestoreHeader(httpOptions);\n\n            return http.splitGet(outputModifier, httpOptions);\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 Julia model).\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);\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, 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 in your Julia 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        /**\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         * **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 a method from the model.\n         *\n         * Depending on the language in which you have written your model, the 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         *      // method \"solve\" takes no arguments\n         *     rs.do('solve');\n         *      // method \"echo\" takes one argument, a string\n         *     rs.do('echo', ['hello']);\n         *      // method \"echo\" takes one argument, a string\n         *     rs.do('echo', 'hello');\n         *      // method \"sumArray\" takes one argument, an array\n         *     rs.do('sumArray', [[4,2,1]]);\n         *      // method \"add\" takes two arguments, both integers\n         *     rs.do({ name:'add', params:[2,4] });\n         *\n         * **Parameters**\n         * @param {String} operation Name of method.\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 methods from the model, sequentially.\n         *\n         * Depending on the language in which you have written your model, the methods 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         *      // methods \"initialize\" and \"solve\" do not take any arguments\n         *     rs.serial(['initialize', 'solve']);\n         *      // methods \"init\" and \"reset\" take two arguments each\n         *     rs.serial([  { name: 'init', params: [1,2] },\n         *                  { name: 'reset', params: [2,3] }]);\n         *      // method \"init\" takes two arguments,\n         *      // method \"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 methods take parameters, pass an array of the method names (strings). If any of the methods do take parameters, pass an array of objects, each of which contains a method 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}\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 doSingleOp = function () {\n                var op = ops.shift();\n                var arg = args.shift();\n\n                me.do(op, arg, {\n                    success: function () {\n                        if (ops.length) {\n                            doSingleOp();\n                        } else {\n                            $d.resolve.apply(this, arguments);\n                            postOptions.success.apply(this, arguments);\n                        }\n                    },\n                    error: function () {\n                        $d.reject.apply(this, arguments);\n                        postOptions.error.apply(this, arguments);\n                    }\n                });\n            };\n\n            doSingleOp();\n\n            return $d.promise();\n        },\n\n        /**\n         * Call several methods from the model, executing them in parallel.\n         *\n         * Depending on the language in which you have written your model, the methods 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         *      // methods \"solve\" and \"reset\" do not take any arguments\n         *     rs.parallel(['solve', 'reset']);\n         *      // methods \"add\" and \"subtract\" take two arguments each\n         *     rs.parallel([ { name: 'add', params: [1,2] },\n         *                   { name: 'subtract', params:[2,3] }]);\n         *      // methods \"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 methods take parameters, pass an array of the method names (as strings). If any of the methods do take parameters, you have two options. You can pass an array of objects, each of which contains a method name and its own (possibly empty) array of parameters. Alternatively, you can pass a single object with the method 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}\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            $.when.apply(this, queue)\n                .then(function () {\n                    $d.resolve.apply(this, arguments);\n                    postOptions.success.apply(this.arguments);\n                })\n                .fail(function () {\n                    $d.reject.apply(this, arguments);\n                    postOptions.error.apply(this.arguments);\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> }`.\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        /**\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 replay or clone runs. It 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        * 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'];\n\n            if (api === '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 + api + '/';\n\n            if ($.inArray(api, 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         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                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        * @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            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 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 session = getStore(options).get(EPI_SESSION_KEY) || '{}';\n            // return JSON.parse(session);\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 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: session.auth_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            };\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};\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         * returns operations of the form `[[op1,op2], [arg1, arg2]]`\n         * @param  {Object|Array|String} operations operations to perform\n         * @param  {Array} args arguments for operation\n         * @return {String}    Matrix-format query parameters\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/always-new-strategy.js","src/managers/run-strategies/conditional-creation-strategy.js","src/managers/run-strategies/multiplayer-strategy.js","src/managers/run-strategies/new-if-initialized-strategy.js","src/managers/run-strategies/new-if-missing-strategy.js","src/managers/run-strategies/new-if-persisted-strategy.js","src/managers/run-strategies/none-strategy.js","src/managers/run-strategies/persistent-single-player-strategy.js","src/managers/run-strategies/strategies-map.js","src/managers/scenario-manager.js","src/managers/special-operations.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/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;;;;ACtEA;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;;ACjYA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7PA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtYA;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;;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;;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;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;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;;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;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACTA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjDA;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;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvLA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;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;;ACxOA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;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;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5hBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;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;;AC5HA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;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;;ACrKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrvBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;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;AACA;;ACpIA;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;;ACdA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;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    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');\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');\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');\n\nF.manager.strategy['always-new'] = require('./managers/run-strategies/always-new-strategy');\nF.manager.strategy['conditional-creation'] = require('./managers/run-strategies/conditional-creation-strategy');\nF.manager.strategy.identity = require('./managers/run-strategies/none-strategy');\nF.manager.strategy['new-if-missing'] = require('./managers/run-strategies/new-if-missing-strategy');\nF.manager.strategy['new-if-missing'] = require('./managers/run-strategies/new-if-missing-strategy');\nF.manager.strategy['new-if-persisted'] = require('./managers/run-strategies/new-if-persisted-strategy');\nF.manager.strategy['new-if-initialized'] = require('./managers/run-strategies/new-if-initialized-strategy');\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\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) {\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 });\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 sessionInfo = {\n                auth_token: token,\n                account: adapterOptions.account,\n                project: project,\n                userId: userInfo.user_id,\n                groups: oldGroups,\n                isTeamMember: isTeamMember\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', 401, data);\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);\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                        handleGroupList(groups);\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     * 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 * 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 * While you can work directly with the Channel Manager through Node.js (for example, `require('manager/channel-manager')`) -- or even work directly with `$.cometd` and Epicenter's underlying [Push Channel API](../../../rest_apis/multiplayer/channel/) -- most often it will be easiest to work with the [Epicenter Channel Manager](../epicenter-channel-manager/). The Epicenter Channel Manager is a wrapper that instantiates a Channel Manager with Epicenter-specific defaults.\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 the channel, 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 channel = cm.getChannel();\n *\n *        channel.subscribe('topic', callback);\n *        channel.publish('topic', { myData: 100 });\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. See 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). There are two main use cases for the channel: event notifications and chat messages.\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 probably 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 * To use the Epicenter Channel Manager: instantiate it, get the channel of the scope you want ([user](../../../glossary/#users), [world](../../../glossary/#world), or [group](../../../glossary/#groups)), 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 *     gc.subscribe('broadcasts', callback);\n *\n * For additional background on Epicenter's push channel, see the introductory notes on the [Push Channel API](../../../rest_apis/multiplayer/channel/) page.\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 classFrom = require('../util/inherit');\nvar urlService = require('../service/url-config-service');\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};\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 urlOpts = urlService(defaultCometOptions.server);\n        if (!defaultCometOptions.url) {\n            //Default epicenter cometd endpoint\n            defaultCometOptions.url = urlOpts.protocol + '://' + urlOpts.host + '/channel/subscribe';\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        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 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 and world. 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 shared world are also online.\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     *         model: 'model.eqn'\n     *     });\n     *     worldManager.getCurrentWorld().then(function (worldObject, worldService) {\n     *         var presenceChannel = cm.getPresenceChannel(worldObject);\n     *         presenceChannel.on('presence', function (evt, notification) {\n     *              console.log(notification.online, notification.userId);\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} userid (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    getPresenceChannel: function (world, userid, 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\n        var session = this.sessionManager.getMergedOptions(this.options);\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].join('/');\n        var channel = __super.getChannel.call(this, { base: baseTopic });\n\n        var lastPingTime = { };\n\n        var PING_INTERVAL = 6000;\n        channel.subscribe('internal-ping-channel', function (notification) {\n            var incomingUserId = notification.data.user;\n            if (!lastPingTime[incomingUserId] && incomingUserId !== userid) {\n                channel.trigger('presence', { userId: incomingUserId, online: true });\n            }\n            lastPingTime[incomingUserId] = (new Date()).valueOf();\n        });\n\n        setInterval(function () {\n            channel.publish('internal-ping-channel', { user: userid });\n\n            $.each(lastPingTime, function (key, value) {\n                var now = (new Date()).valueOf();\n                if (value && value + (PING_INTERVAL * 2) < now) {\n                    lastPingTime[key] = null;\n                    channel.trigger('presence', { userId: key, online: false });\n                }\n            });\n        }, PING_INTERVAL);\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                };\n                var actualData = payload.data.data;\n                if (actualData.data) { //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/), the RESFTful [Run API](../../../rest_apis/aggregate_run_api) and the [Model Run API](../../../rest_apis/other_apis/model_apis/run/). 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. (Note that many of the Epicenter sample projects use a Run Service directly, because generally the sample projects are played in one end user session and don't care about run states or run strategies.)\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 pass a `files` object with the names of the files, for example: `\"files\": {\"data\": \"myExtraData.xls\"}`. (Note that you'll also need to add this same files object to your Vensim [configuration file](../../../model_code/vensim/).) See the [underlying Model Run API](../../../rest_apis/other_apis/model_apis/run/#post-creating-a-new-run-for-this-project) for additional information.\n*\n*   * `strategy`: (optional) Run creation strategy for when to create a new run and when to reuse an end user's existing run. See [Run Manager Strategies](../strategies/) for details. Defaults to `new-if-initialized`.\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.\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: 'always-new',\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\n'use strict';\nvar strategiesMap = require('./run-strategies/strategies-map');\nvar specialOperations = require('./special-operations');\nvar RunService = require('../service/run-api-service');\n\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\n\nvar defaults = {\n    /**\n     * Run creation strategy for when to create a new run and when to reuse an end user's existing run. See [Run Manager Strategies](../strategies/) for details. Defaults to `new-if-initialized`.\n     * @type {String}\n     */\n\n    strategy: 'new-if-initialized'\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 {\n        this.run = new RunService(this.options.run);\n    }\n\n    patchRunService(this.run, this);\n\n    var StrategyCtor = typeof this.options.strategy === 'function' ? this.options.strategy : strategiesMap[this.options.strategy];\n\n    if (!StrategyCtor) {\n        throw new Error('Specified run creation strategy was invalid:', this.options.strategy);\n    }\n\n    this.strategy = new StrategyCtor(this.run, this.options);\n}\n\nRunManager.prototype = {\n    /**\n     * Returns the run object for a 'good' run.\n     *\n     * A good run is defined by the strategy. For example, if the strategy is `always-new`, the call\n     * to `getRun()` always returns a newly created run; if the strategy is `new-if-persisted`,\n     * `getRun()` creates a new run if the previous run is in a persisted state, otherwise\n     * it returns the previous run. 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     * @return {$promise} Promise to complete the call.\n     */\n    getRun: function () {\n        return this.strategy\n                .getRun();\n    },\n\n    /**\n     * Returns the run object for a new run, regardless of strategy: force creation of a new run.\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} runServiceOptions The options object to configure the Run Service. See [Run API Service](../run-api-service/) for more.\n     * @return {Promise}\n     */\n    reset: function (runServiceOptions) {\n        return this.strategy.reset(runServiceOptions);\n    }\n};\n\nmodule.exports = RunManager;\n","/**\n * The `always-new` 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\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 (runService, options) {\n        __super.constructor.call(this, runService, 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","'use strict';\n\nvar Base = require('./none-strategy');\nvar SessionManager = require('../../store/session-manager');\nvar classFrom = require('../../util/inherit');\nvar AuthManager = require('../auth-manager');\n\nvar keyNames = require('../key-names');\n\nvar defaults = {\n    sessionKey: keyNames.STRATEGY_SESSION_KEY,\n    path: ''\n};\n\nfunction setRunInSession(sessionKey, run, sessionManager) {\n    sessionManager.getStore().set(sessionKey, JSON.stringify({ runId: run.id }));\n}\n\n/**\n* Conditional Creation Strategy\n* This strategy will try to get the run stored in the cookie and\n* evaluate if needs to create a new run by calling the 'condition' function\n*/\n\nvar Strategy = classFrom(Base, {\n    constructor: function Strategy(runService, condition, options) {\n        if (condition == null) { //eslint-disable-line\n            //TODO: not sure why this is explicitly ==\n            throw new Error('Conditional strategy needs a condition to create a run');\n        }\n\n        this._auth = new AuthManager();\n        this.run = runService;\n        this.condition = typeof condition !== 'function' ? function () { return condition; } : condition;\n        this.options = $.extend(true, {}, defaults, options);\n        this.sessionManager = new SessionManager(options);\n        this.runOptions = this.options.run;\n    },\n\n    runOptionsWithScope: function () {\n        var userSession = this._auth.getCurrentUserSessionInfo();\n        return $.extend({\n            scope: { group: userSession.groupName }\n        }, this.runOptions);\n    },\n\n    reset: function (runServiceOptions) {\n        var me = this;\n        var opt = this.runOptionsWithScope();\n\n        return this.run\n                .create(opt, runServiceOptions)\n            .then(function (run) {\n                setRunInSession(me.options.sessionKey, run, me.sessionManager);\n                run.freshlyCreated = true;\n                return run;\n            });\n    },\n\n    getRun: function () {\n        var sessionStore = this.sessionManager.getStore();\n        var runSession = JSON.parse(sessionStore.get(this.options.sessionKey));\n        var me = this;\n        if (runSession && runSession.runId) {\n            return this._loadAndCheck(runSession).fail(function () {\n                return me.reset(); //if it got the wrong cookie for e.g.\n            });\n        } else {\n            return this.reset();\n        }\n    },\n\n    _loadAndCheck: function (runSession) {\n        var shouldCreate = false;\n        var me = this;\n\n        return this.run\n            .load(runSession.runId, null, {\n                success: function (run, msg, headers) {\n                    shouldCreate = me.condition(run, headers);\n                }\n            })\n            .then(function (run) {\n                if (shouldCreate) {\n                    var opt = me.runOptionsWithScope();\n                    return me.run.create(opt)\n                    .then(function (run) {\n                        setRunInSession(me.options.sessionKey, run, me.sessionManager);\n                        run.freshlyCreated = true;\n                        return run;\n                    });\n                }\n                return run;\n            });\n    }\n});\n\nmodule.exports = Strategy;\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/).\n * \n * Using this strategy means that end users in projects with multiplayer worlds always see the most current run and world. 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');\nvar AuthManager = require('../auth-manager');\n\nvar defaults = {\n    store: {\n        synchronous: true\n    }\n};\n\nvar Strategy = classFrom(IdentityStrategy, {\n\n    constructor: function (runService, options) {\n        this.runService = runService;\n        this.options = $.extend(true, {}, defaults, options);\n        this._auth = new AuthManager();\n        this._loadRun = this._loadRun.bind(this);\n        this.worldApi = new WorldApiAdapter(this.options.run);\n    },\n\n    reset: function () {\n        var session = this._auth.getCurrentUserSessionInfo();\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);\n            }.bind(this));\n    },\n\n    getRun: function () {\n        var session = this._auth.getCurrentUserSessionInfo();\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\n            return worldApi.getCurrentRunId({ model: model, filter: world.id })\n                .then(me._loadRun)\n                .then(dtd.resolve)\n                .fail(dtd.reject);\n        };\n\n        var serverError = function (error) {\n            // is this possible?\n            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    _loadRun: function (id, options) {\n        return this.runService.load(id, null, options);\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 for Vensim models; it can be set manually for other models.\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\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 (runService, options) {\n        __super.constructor.call(this, runService, this.createIf, options);\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-missing` 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.\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 other words, you care whether you have a run, but as long as you have one, you are certain that this run is the one you are interested in. \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\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 (runService, options) {\n        __super.constructor.call(this, runService, 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 * 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, if they are idle for longer than the **Model Session Timeout**, that 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\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 (runService, options) {\n        __super.constructor.call(this, runService, this.createIf, options);\n    },\n\n    createIf: function (run, headers) {\n        return headers.getResponseHeader('pragma') === 'persistent';\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 * Also, this strategy is necessary if you are working with a multiplayer project and using the [World Manager](../world-manager/) &mdash; or other, similar situations where you do not have direct control over creating the [Run Service](../run-api-service/) instance.\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 (runService, options) {\n        this.runService = runService;\n    },\n\n    reset: function () {\n        // return a newly created run\n        return $.Deferred().resolve().promise();\n    },\n\n    getRun: function () {\n        // return a usable run\n        return $.Deferred().resolve(this.runService).promise();\n    }\n});\n","/**\n * The `persistent-single-player` 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 your project executes your 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). It is useful if end users play with your project for an extended period of time, possibly over several sessions.\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 *         * If the most recent run is currently in the database, bring it back into memory so that the end user can continue working with it. (See more background on [Run Persistence](../../../run_persistence/), or read more on the underlying [State API](../../../rest_apis/other_apis/model_apis/state/) for bringing runs from the database back into memory.) \n */\n\n'use strict';\n\nvar classFrom = require('../../util/inherit');\nvar IdentityStrategy = require('./none-strategy');\nvar StorageFactory = require('../../store/store-factory');\nvar StateApi = require('../../service/state-api-adapter');\nvar AuthManager = require('../auth-manager');\n\nvar keyNames = require('../key-names');\n\nvar defaults = {\n    store: {\n        synchronous: true\n    }\n};\n\nvar Strategy = classFrom(IdentityStrategy, {\n    constructor: function Strategy(runService, options) {\n        this.run = runService;\n        this.options = $.extend(true, {}, defaults, options);\n        this.runOptions = this.options.run;\n        this._store = new StorageFactory(this.options.store);\n        this.stateApi = new StateApi();\n        this._auth = new AuthManager();\n\n        this._loadAndCheck = this._loadAndCheck.bind(this);\n        this._restoreRun = this._restoreRun.bind(this);\n        this._getAllRuns = this._getAllRuns.bind(this);\n        this._loadRun = this._loadRun.bind(this);\n    },\n\n    reset: function (runServiceOptions) {\n        var session = this._auth.getCurrentUserSessionInfo();\n        var opt = $.extend({\n            scope: { group: session.groupName }\n        }, this.runOptions);\n\n        return this.run\n            .create(opt, runServiceOptions)\n            .then(function (run) {\n                run.freshlyCreated = true;\n                return run;\n            });\n    },\n\n    getRun: function () {\n        return this._getAllRuns()\n            .then(this._loadAndCheck);\n    },\n\n    _getAllRuns: function () {\n        var session = JSON.parse(this._store.get(keyNames.EPI_SESSION_KEY) || '{}');\n        return this.run.query({\n            'user.id': session.userId || '0000',\n            'scope.group': session.groupName\n        });\n    },\n\n    _loadAndCheck: function (runs) {\n        if (!runs || !runs.length) {\n            return this.reset();\n        }\n\n        var dateComp = function (a, b) { return new Date(b.date) - new Date(a.date); };\n        var latestRun = runs.sort(dateComp)[0];\n        var me = this;\n        var shouldReplay = false;\n\n        return this.run.load(latestRun.id, null, {\n            success: function (run, msg, headers) {\n                shouldReplay = headers.getResponseHeader('pragma') === 'persistent';\n            }\n        }).then(function (run) {\n            return shouldReplay ? me._restoreRun(run.id) : run;\n        });\n    },\n\n    _restoreRun: function (runId) {\n        var me = this;\n        return this.stateApi.replay({ runId: runId })\n            .then(function (resp) {\n                return me._loadRun(resp.run);\n            });\n    },\n\n    _loadRun: function (id, options) {\n        return this.run.load(id, null, options);\n    }\n\n});\n\nmodule.exports = Strategy;\n","module.exports = {\n    'new-if-initialized': require('./new-if-initialized-strategy'),\n    'new-if-persisted': require('./new-if-persisted-strategy'),\n    'new-if-missing': require('./new-if-missing-strategy'),\n    'always-new': require('./always-new-strategy'),\n    multiplayer: require('./multiplayer-strategy'),\n    'persistent-single-player': require('./persistent-single-player-strategy'),\n    none: require('./none-strategy')\n};\n","'use strict';\nvar RunService = require('../service/run-api-service');\n\nvar defaults = {\n    validFilter: { saved: true }\n};\n\nfunction ScenarioManager(options) {\n    this.options = $.extend(true, {}, defaults, options);\n    this.runService = this.options.run || new RunService(this.options);\n}\n\nScenarioManager.prototype = {\n    getRuns: function (filter) {\n        this.filter = $.extend(true, {}, this.options.validFilter, filter);\n        return this.runService.query(this.filter);\n    },\n\n    loadVariables: function (vars) {\n        return this.runService.query(this.filter, { include: vars });\n    },\n\n    save: function (run, meta) {\n        return this._getService(run).save($.extend(true, {}, { saved: true }, meta));\n    },\n\n    archive: function (run) {\n        return this._getService(run).save({ saved: false });\n    },\n\n    _getService: function (run) {\n        if (typeof run === 'string') {\n            return new RunService($.extend(true, {}, this.options, { filter: run }));\n        }\n\n        if (typeof run === 'object' && run instanceof RunService) {\n            return run;\n        }\n\n        throw new Error('Save method requires a run service or a runId');\n    },\n\n    getRun: function (runId) {\n        return new RunService($.extend(true, {}, this.options, { filter: runId }));\n    }\n};\n\nmodule.exports = ScenarioManager;\n\n","'use strict';\n\n\nmodule.exports = {\n    reset: function (params, options, manager) {\n        return manager.reset(options);\n    }\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(runService, options) {\n        this.runService = runService;\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 () {\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 me.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({model: '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\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 * The Asset API Service (https://forio.com/epicenter/docs/public/api_adapters/generated/asset-api-adapter/) is typically used for all project use cases, and it's unlikely this File Service will be used directly except by Admin tools (e.g. Flow Inspector).\n *\n * Partially implemented.\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        var upload = uploadBody(fileName, contents);\n\n        return $.extend(true, {}, serviceOptions, options, {\n            url: urlConfig.getAPIPath('file') + path,\n            data: upload.body,\n            contentType: 'multipart/form-data; boundary=' + upload.boundary\n        });\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} 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\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} 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 * 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 * Typically, you use the [Epicenter Channel Manager](../epicenter-channel-manager/) to create or retrieve channels, then use the Channel Service `subscribe()` and `publish()` methods to listen to or update data. (For additional background on Epicenter's push channel, see the introductory notes on the [Push Channel API](../../../rest_apis/multiplayer/channel/) page.)\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 * 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 * ## 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 gives you control over run creation depending on run states.\n *\n * However, many of the Epicenter sample projects use a Run Service, because generally the sample projects are played in one end user session and don't care about run states or [run strategies](../strategies/). The Run API Service is also useful for building an interface for a facilitator, because it makes it easy to list data across multiple runs (using the `filter()` and `query()` methods).\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 * 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.\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 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         * Criteria by which to filter runs. Defaults to empty string.\n         * @type {String}\n         */\n        filter: '',\n\n        /**\n         * Convenience alias for filter.\n         * @type {String}\n         */\n        id: '',\n\n        /**\n         * Flag determines if `X-AutoRestore: true` header is sent to Epicenter. 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    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    urlConfig.filter = ';';\n    urlConfig.getFilterURL = function () {\n        var url = urlConfig.getAPIPath('run');\n        var filter = qutil.toMatrixFormat(serviceOptions.filter);\n\n        if (filter) {\n            url += filter + '/';\n        }\n        return url;\n    };\n\n    urlConfig.addAutoRestoreHeader = function (options) {\n        var filter = serviceOptions.filter;\n        // The semicolon separated filter is used when filter is an object\n        var isFilterRunId = filter && $.type(filter) === 'string';\n        if (serviceOptions.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\n    var 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    var http = new TransportFactory(httpOptions);\n    http.splitGet = rutil.splitGetFactory(httpOptions);\n\n    var setFilterOrThrowError = 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 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 `always-new`, 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'];\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 Julia model).\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            serviceOptions.filter = qs; //shouldn't be able to over-ride\n            var httpOptions = $.extend(true, {}, serviceOptions, options);\n            httpOptions = urlConfig.addAutoRestoreHeader(httpOptions);\n\n            return http.splitGet(outputModifier, httpOptions);\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 Julia model).\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);\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, 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 in your Julia 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        /**\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         * **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 a method from the model.\n         *\n         * Depending on the language in which you have written your model, the 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         *      // method \"solve\" takes no arguments\n         *     rs.do('solve');\n         *      // method \"echo\" takes one argument, a string\n         *     rs.do('echo', ['hello']);\n         *      // method \"echo\" takes one argument, a string\n         *     rs.do('echo', 'hello');\n         *      // method \"sumArray\" takes one argument, an array\n         *     rs.do('sumArray', [[4,2,1]]);\n         *      // method \"add\" takes two arguments, both integers\n         *     rs.do({ name:'add', params:[2,4] });\n         *\n         * **Parameters**\n         * @param {String} operation Name of method.\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 methods from the model, sequentially.\n         *\n         * Depending on the language in which you have written your model, the methods 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         *      // methods \"initialize\" and \"solve\" do not take any arguments\n         *     rs.serial(['initialize', 'solve']);\n         *      // methods \"init\" and \"reset\" take two arguments each\n         *     rs.serial([  { name: 'init', params: [1,2] },\n         *                  { name: 'reset', params: [2,3] }]);\n         *      // method \"init\" takes two arguments,\n         *      // method \"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 methods take parameters, pass an array of the method names (strings). If any of the methods do take parameters, pass an array of objects, each of which contains a method 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}\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 doSingleOp = function () {\n                var op = ops.shift();\n                var arg = args.shift();\n\n                me.do(op, arg, {\n                    success: function () {\n                        if (ops.length) {\n                            doSingleOp();\n                        } else {\n                            $d.resolve.apply(this, arguments);\n                            postOptions.success.apply(this, arguments);\n                        }\n                    },\n                    error: function () {\n                        $d.reject.apply(this, arguments);\n                        postOptions.error.apply(this, arguments);\n                    }\n                });\n            };\n\n            doSingleOp();\n\n            return $d.promise();\n        },\n\n        /**\n         * Call several methods from the model, executing them in parallel.\n         *\n         * Depending on the language in which you have written your model, the methods 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         *      // methods \"solve\" and \"reset\" do not take any arguments\n         *     rs.parallel(['solve', 'reset']);\n         *      // methods \"add\" and \"subtract\" take two arguments each\n         *     rs.parallel([ { name: 'add', params: [1,2] },\n         *                   { name: 'subtract', params:[2,3] }]);\n         *      // methods \"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 methods take parameters, pass an array of the method names (as strings). If any of the methods do take parameters, you have two options. You can pass an array of objects, each of which contains a method name and its own (possibly empty) array of parameters. Alternatively, you can pass a single object with the method 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}\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            $.when.apply(this, queue)\n                .then(function () {\n                    $d.resolve.apply(this, arguments);\n                    postOptions.success.apply(this.arguments);\n                })\n                .fail(function () {\n                    $d.reject.apply(this, arguments);\n                    postOptions.error.apply(this.arguments);\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> }`.\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        /**\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 replay or clone runs. It 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        * 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'];\n\n            if (api === '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 + api + '/';\n\n            if ($.inArray(api, 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         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                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        * @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            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 session = getStore(options).get(EPI_SESSION_KEY) || '{}';\n            // return JSON.parse(session);\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            };\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};\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         * returns operations of the form `[[op1,op2], [arg1, arg2]]`\n         * @param  {Object|Array|String} operations operations to perform\n         * @param  {Array} args arguments for operation\n         * @return {String}    Matrix-format query parameters\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 01dbbeb9..9aa0c098 100644 --- a/dist/epicenter.min.js +++ b/dist/epicenter.min.js @@ -80,7 +80,7 @@ module.exports={"new-if-initialized":require("./new-if-initialized-strategy"),"n },{"../store/session-manager":40,"../transport/http-transport-factory":43,"../util/object-util":45,"./configuration-service":27}],39:[function(require,module,exports){ "use strict";var Cookie=function(){this.get=function(){return document.cookie},this.set=function(newCookie){document.cookie=newCookie}};module.exports=function(config){var host=window.location.hostname;var validHost=host.split(".").length>1;var domain=validHost?"."+host:null;var defaults={root:"/",domain:domain,cookie:new Cookie};this.serviceOptions=$.extend({},defaults,config);var publicAPI={set:function(key,value,options){var setOptions=$.extend(!0,{},this.serviceOptions,options);var domain=setOptions.domain;var path=setOptions.root;var cookie=setOptions.cookie;return cookie.set(encodeURIComponent(key)+"="+encodeURIComponent(value)+(domain?"; domain="+domain:"")+(path?"; path="+path:"")),value},get:function(key){var cookie=this.serviceOptions.cookie;var cookieReg=new RegExp("(?:^|;)\\s*"+encodeURIComponent(key).replace(/[\-\.\+\*]/g,"\\$&")+"\\s*\\=\\s*([^;]*).*$");var res=cookieReg.exec(cookie.get());var val=res?decodeURIComponent(res[1]):null;return val},remove:function(key,options){var remOptions=$.extend(!0,{},this.serviceOptions,options);var domain=remOptions.domain;var path=remOptions.root;var cookie=remOptions.cookie;return cookie.set(encodeURIComponent(key)+"=; expires=Thu, 01 Jan 1970 00:00:00 GMT"+(domain?"; domain="+domain:"")+(path?"; path="+path:"")),key},destroy:function(){var cookie=this.serviceOptions.cookie;var aKeys=cookie.get().replace(/((?:^|\s*;)[^\=]+)(?=;|$)|^\s*|\s*(?:\=[^;]*)?(?:\1|$)/g,"").split(/\s*(?:\=[^;]*)?;\s*/);for(var nIdx=0;nIdx> 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 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');\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');\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');\n\nF.manager.strategy['always-new'] = require('./managers/run-strategies/always-new-strategy');\nF.manager.strategy['conditional-creation'] = require('./managers/run-strategies/conditional-creation-strategy');\nF.manager.strategy.identity = require('./managers/run-strategies/none-strategy');\nF.manager.strategy['new-if-missing'] = require('./managers/run-strategies/new-if-missing-strategy');\nF.manager.strategy['new-if-missing'] = require('./managers/run-strategies/new-if-missing-strategy');\nF.manager.strategy['new-if-persisted'] = require('./managers/run-strategies/new-if-persisted-strategy');\nF.manager.strategy['new-if-initialized'] = require('./managers/run-strategies/new-if-initialized-strategy');\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\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) {\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 });\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 sessionInfo = {\n auth_token: token,\n account: adapterOptions.account,\n project: project,\n userId: userInfo.user_id,\n groups: oldGroups,\n isTeamMember: isTeamMember\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', 401, data);\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);\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 handleGroupList(groups);\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 * 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 * 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 * While you can work directly with the Channel Manager through Node.js (for example, `require('manager/channel-manager')`) -- or even work directly with `$.cometd` and Epicenter's underlying [Push Channel API](../../../rest_apis/multiplayer/channel/) -- most often it will be easiest to work with the [Epicenter Channel Manager](../epicenter-channel-manager/). The Epicenter Channel Manager is a wrapper that instantiates a Channel Manager with Epicenter-specific defaults.\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 the channel, 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 channel = cm.getChannel();\n *\n * channel.subscribe('topic', callback);\n * channel.publish('topic', { myData: 100 });\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. See 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). There are two main use cases for the channel: event notifications and chat messages.\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 probably 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 * To use the Epicenter Channel Manager: instantiate it, get the channel of the scope you want ([user](../../../glossary/#users), [world](../../../glossary/#world), or [group](../../../glossary/#groups)), 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 * gc.subscribe('broadcasts', callback);\n *\n * For additional background on Epicenter's push channel, see the introductory notes on the [Push Channel API](../../../rest_apis/multiplayer/channel/) page.\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 classFrom = require('../util/inherit');\nvar urlService = require('../service/url-config-service');\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};\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 urlOpts = urlService(defaultCometOptions.server);\n if (!defaultCometOptions.url) {\n //Default epicenter cometd endpoint\n defaultCometOptions.url = urlOpts.protocol + '://' + urlOpts.host + '/channel/subscribe';\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 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 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 and world. 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 shared world are also online.\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 * model: 'model.eqn'\n * });\n * worldManager.getCurrentWorld().then(function (worldObject, worldService) {\n * var presenceChannel = cm.getPresenceChannel(worldObject);\n * presenceChannel.on('presence', function (evt, notification) {\n * console.log(notification.online, notification.userId);\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} userid (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 getPresenceChannel: function (world, userid, 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\n var session = this.sessionManager.getMergedOptions(this.options);\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].join('/');\n var channel = __super.getChannel.call(this, { base: baseTopic });\n\n var lastPingTime = { };\n\n var PING_INTERVAL = 6000;\n channel.subscribe('internal-ping-channel', function (notification) {\n var incomingUserId = notification.data.user;\n if (!lastPingTime[incomingUserId] && incomingUserId !== userid) {\n channel.trigger('presence', { userId: incomingUserId, online: true });\n }\n lastPingTime[incomingUserId] = (new Date()).valueOf();\n });\n\n setInterval(function () {\n channel.publish('internal-ping-channel', { user: userid });\n\n $.each(lastPingTime, function (key, value) {\n var now = (new Date()).valueOf();\n if (value && value + (PING_INTERVAL * 2) < now) {\n lastPingTime[key] = null;\n channel.trigger('presence', { userId: key, online: false });\n }\n });\n }, PING_INTERVAL);\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 };\n var actualData = payload.data.data;\n if (actualData.data) { //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/), the RESFTful [Run API](../../../rest_apis/aggregate_run_api) and the [Model Run API](../../../rest_apis/other_apis/model_apis/run/). 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. (Note that many of the Epicenter sample projects use a Run Service directly, because generally the sample projects are played in one end user session and don't care about run states or run strategies.)\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 pass a `files` object with the names of the files, for example: `\"files\": {\"data\": \"myExtraData.xls\"}`. (Note that you'll also need to add this same files object to your Vensim [configuration file](../../../model_code/vensim/).) See the [underlying Model Run API](../../../rest_apis/other_apis/model_apis/run/#post-creating-a-new-run-for-this-project) for additional information.\n*\n* * `strategy`: (optional) Run creation strategy for when to create a new run and when to reuse an end user's existing run. See [Run Manager Strategies](../strategies/) for details. Defaults to `new-if-initialized`.\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.\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: 'always-new',\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\n'use strict';\nvar strategiesMap = require('./run-strategies/strategies-map');\nvar specialOperations = require('./special-operations');\nvar RunService = require('../service/run-api-service');\n\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\n\nvar defaults = {\n /**\n * Run creation strategy for when to create a new run and when to reuse an end user's existing run. See [Run Manager Strategies](../strategies/) for details. Defaults to `new-if-initialized`.\n * @type {String}\n */\n\n strategy: 'new-if-initialized'\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 {\n this.run = new RunService(this.options.run);\n }\n\n patchRunService(this.run, this);\n\n var StrategyCtor = typeof this.options.strategy === 'function' ? this.options.strategy : strategiesMap[this.options.strategy];\n\n if (!StrategyCtor) {\n throw new Error('Specified run creation strategy was invalid:', this.options.strategy);\n }\n\n this.strategy = new StrategyCtor(this.run, this.options);\n}\n\nRunManager.prototype = {\n /**\n * Returns the run object for a 'good' run.\n *\n * A good run is defined by the strategy. For example, if the strategy is `always-new`, the call\n * to `getRun()` always returns a newly created run; if the strategy is `new-if-persisted`,\n * `getRun()` creates a new run if the previous run is in a persisted state, otherwise\n * it returns the previous run. 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 * @return {$promise} Promise to complete the call.\n */\n getRun: function () {\n return this.strategy\n .getRun();\n },\n\n /**\n * Returns the run object for a new run, regardless of strategy: force creation of a new run.\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} runServiceOptions The options object to configure the Run Service. See [Run API Service](../run-api-service/) for more.\n * @return {Promise}\n */\n reset: function (runServiceOptions) {\n return this.strategy.reset(runServiceOptions);\n }\n};\n\nmodule.exports = RunManager;\n","/**\n * The `always-new` 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\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 (runService, options) {\n __super.constructor.call(this, runService, 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","'use strict';\n\nvar Base = require('./none-strategy');\nvar SessionManager = require('../../store/session-manager');\nvar classFrom = require('../../util/inherit');\nvar AuthManager = require('../auth-manager');\n\nvar keyNames = require('../key-names');\n\nvar defaults = {\n sessionKey: keyNames.STRATEGY_SESSION_KEY,\n path: ''\n};\n\nfunction setRunInSession(sessionKey, run, sessionManager) {\n sessionManager.getStore().set(sessionKey, JSON.stringify({ runId: run.id }));\n}\n\n/**\n* Conditional Creation Strategy\n* This strategy will try to get the run stored in the cookie and\n* evaluate if needs to create a new run by calling the 'condition' function\n*/\n\nvar Strategy = classFrom(Base, {\n constructor: function Strategy(runService, condition, options) {\n if (condition == null) { //eslint-disable-line\n //TODO: not sure why this is explicitly ==\n throw new Error('Conditional strategy needs a condition to create a run');\n }\n\n this._auth = new AuthManager();\n this.run = runService;\n this.condition = typeof condition !== 'function' ? function () { return condition; } : condition;\n this.options = $.extend(true, {}, defaults, options);\n this.sessionManager = new SessionManager(options);\n this.runOptions = this.options.run;\n },\n\n runOptionsWithScope: function () {\n var userSession = this._auth.getCurrentUserSessionInfo();\n return $.extend({\n scope: { group: userSession.groupName }\n }, this.runOptions);\n },\n\n reset: function (runServiceOptions) {\n var me = this;\n var opt = this.runOptionsWithScope();\n\n return this.run\n .create(opt, runServiceOptions)\n .then(function (run) {\n setRunInSession(me.options.sessionKey, run, me.sessionManager);\n run.freshlyCreated = true;\n return run;\n });\n },\n\n getRun: function () {\n var sessionStore = this.sessionManager.getStore();\n var runSession = JSON.parse(sessionStore.get(this.options.sessionKey));\n var me = this;\n if (runSession && runSession.runId) {\n return this._loadAndCheck(runSession).fail(function () {\n return me.reset(); //if it got the wrong cookie for e.g.\n });\n } else {\n return this.reset();\n }\n },\n\n _loadAndCheck: function (runSession) {\n var shouldCreate = false;\n var me = this;\n\n return this.run\n .load(runSession.runId, null, {\n success: function (run, msg, headers) {\n shouldCreate = me.condition(run, headers);\n }\n })\n .then(function (run) {\n if (shouldCreate) {\n var opt = me.runOptionsWithScope();\n return me.run.create(opt)\n .then(function (run) {\n setRunInSession(me.options.sessionKey, run, me.sessionManager);\n run.freshlyCreated = true;\n return run;\n });\n }\n return run;\n });\n }\n});\n\nmodule.exports = Strategy;\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/).\n * \n * Using this strategy means that end users in projects with multiplayer worlds always see the most current run and world. 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');\nvar AuthManager = require('../auth-manager');\n\nvar defaults = {\n store: {\n synchronous: true\n }\n};\n\nvar Strategy = classFrom(IdentityStrategy, {\n\n constructor: function (runService, options) {\n this.runService = runService;\n this.options = $.extend(true, {}, defaults, options);\n this._auth = new AuthManager();\n this._loadRun = this._loadRun.bind(this);\n this.worldApi = new WorldApiAdapter(this.options.run);\n },\n\n reset: function () {\n var session = this._auth.getCurrentUserSessionInfo();\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);\n }.bind(this));\n },\n\n getRun: function () {\n var session = this._auth.getCurrentUserSessionInfo();\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\n return worldApi.getCurrentRunId({ model: model, filter: world.id })\n .then(me._loadRun)\n .then(dtd.resolve)\n .fail(dtd.reject);\n };\n\n var serverError = function (error) {\n // is this possible?\n 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 _loadRun: function (id, options) {\n return this.runService.load(id, null, options);\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 for Vensim models; it can be set manually for other models.\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\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 (runService, options) {\n __super.constructor.call(this, runService, this.createIf, options);\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-missing` 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.\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 other words, you care whether you have a run, but as long as you have one, you are certain that this run is the one you are interested in. \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\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 (runService, options) {\n __super.constructor.call(this, runService, 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 * 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, if they are idle for longer than the **Model Session Timeout**, that 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\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 (runService, options) {\n __super.constructor.call(this, runService, this.createIf, options);\n },\n\n createIf: function (run, headers) {\n return headers.getResponseHeader('pragma') === 'persistent';\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 * Also, this strategy is necessary if you are working with a multiplayer project and using the [World Manager](../world-manager/) — or other, similar situations where you do not have direct control over creating the [Run Service](../run-api-service/) instance.\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 (runService, options) {\n this.runService = runService;\n },\n\n reset: function () {\n // return a newly created run\n return $.Deferred().resolve().promise();\n },\n\n getRun: function () {\n // return a usable run\n return $.Deferred().resolve(this.runService).promise();\n }\n});\n","/**\n * The `persistent-single-player` 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 your project executes your 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). It is useful if end users play with your project for an extended period of time, possibly over several sessions.\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 * * If the most recent run is currently in the database, bring it back into memory so that the end user can continue working with it. (See more background on [Run Persistence](../../../run_persistence/), or read more on the underlying [State API](../../../rest_apis/other_apis/model_apis/state/) for bringing runs from the database back into memory.) \n */\n\n'use strict';\n\nvar classFrom = require('../../util/inherit');\nvar IdentityStrategy = require('./none-strategy');\nvar StorageFactory = require('../../store/store-factory');\nvar StateApi = require('../../service/state-api-adapter');\nvar AuthManager = require('../auth-manager');\n\nvar keyNames = require('../key-names');\n\nvar defaults = {\n store: {\n synchronous: true\n }\n};\n\nvar Strategy = classFrom(IdentityStrategy, {\n constructor: function Strategy(runService, options) {\n this.run = runService;\n this.options = $.extend(true, {}, defaults, options);\n this.runOptions = this.options.run;\n this._store = new StorageFactory(this.options.store);\n this.stateApi = new StateApi();\n this._auth = new AuthManager();\n\n this._loadAndCheck = this._loadAndCheck.bind(this);\n this._restoreRun = this._restoreRun.bind(this);\n this._getAllRuns = this._getAllRuns.bind(this);\n this._loadRun = this._loadRun.bind(this);\n },\n\n reset: function (runServiceOptions) {\n var session = this._auth.getCurrentUserSessionInfo();\n var opt = $.extend({\n scope: { group: session.groupName }\n }, this.runOptions);\n\n return this.run\n .create(opt, runServiceOptions)\n .then(function (run) {\n run.freshlyCreated = true;\n return run;\n });\n },\n\n getRun: function () {\n return this._getAllRuns()\n .then(this._loadAndCheck);\n },\n\n _getAllRuns: function () {\n var session = JSON.parse(this._store.get(keyNames.EPI_SESSION_KEY) || '{}');\n return this.run.query({\n 'user.id': session.userId || '0000',\n 'scope.group': session.groupName\n });\n },\n\n _loadAndCheck: function (runs) {\n if (!runs || !runs.length) {\n return this.reset();\n }\n\n var dateComp = function (a, b) { return new Date(b.date) - new Date(a.date); };\n var latestRun = runs.sort(dateComp)[0];\n var me = this;\n var shouldReplay = false;\n\n return this.run.load(latestRun.id, null, {\n success: function (run, msg, headers) {\n shouldReplay = headers.getResponseHeader('pragma') === 'persistent';\n }\n }).then(function (run) {\n return shouldReplay ? me._restoreRun(run.id) : run;\n });\n },\n\n _restoreRun: function (runId) {\n var me = this;\n return this.stateApi.replay({ runId: runId })\n .then(function (resp) {\n return me._loadRun(resp.run);\n });\n },\n\n _loadRun: function (id, options) {\n return this.run.load(id, null, options);\n }\n\n});\n\nmodule.exports = Strategy;\n","module.exports = {\n 'new-if-initialized': require('./new-if-initialized-strategy'),\n 'new-if-persisted': require('./new-if-persisted-strategy'),\n 'new-if-missing': require('./new-if-missing-strategy'),\n 'always-new': require('./always-new-strategy'),\n multiplayer: require('./multiplayer-strategy'),\n 'persistent-single-player': require('./persistent-single-player-strategy'),\n none: require('./none-strategy')\n};\n","'use strict';\nvar RunService = require('../service/run-api-service');\n\nvar defaults = {\n validFilter: { saved: true }\n};\n\nfunction ScenarioManager(options) {\n this.options = $.extend(true, {}, defaults, options);\n this.runService = this.options.run || new RunService(this.options);\n}\n\nScenarioManager.prototype = {\n getRuns: function (filter) {\n this.filter = $.extend(true, {}, this.options.validFilter, filter);\n return this.runService.query(this.filter);\n },\n\n loadVariables: function (vars) {\n return this.runService.query(this.filter, { include: vars });\n },\n\n save: function (run, meta) {\n return this._getService(run).save($.extend(true, {}, { saved: true }, meta));\n },\n\n archive: function (run) {\n return this._getService(run).save({ saved: false });\n },\n\n _getService: function (run) {\n if (typeof run === 'string') {\n return new RunService($.extend(true, {}, this.options, { filter: run }));\n }\n\n if (typeof run === 'object' && run instanceof RunService) {\n return run;\n }\n\n throw new Error('Save method requires a run service or a runId');\n },\n\n getRun: function (runId) {\n return new RunService($.extend(true, {}, this.options, { filter: runId }));\n }\n};\n\nmodule.exports = ScenarioManager;\n\n","'use strict';\n\n\nmodule.exports = {\n reset: function (params, options, manager) {\n return manager.reset(options);\n }\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(runService, options) {\n this.runService = runService;\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 () {\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 me.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({model: '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\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 * The Asset API Service (https://forio.com/epicenter/docs/public/api_adapters/generated/asset-api-adapter/) is typically used for all project use cases, and it's unlikely this File Service will be used directly except by Admin tools (e.g. Flow Inspector).\n *\n * Partially implemented.\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 var upload = uploadBody(fileName, contents);\n\n return $.extend(true, {}, serviceOptions, options, {\n url: urlConfig.getAPIPath('file') + path,\n data: upload.body,\n contentType: 'multipart/form-data; boundary=' + upload.boundary\n });\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} 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\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} 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 * 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 * Typically, you use the [Epicenter Channel Manager](../epicenter-channel-manager/) to create or retrieve channels, then use the Channel Service `subscribe()` and `publish()` methods to listen to or update data. (For additional background on Epicenter's push channel, see the introductory notes on the [Push Channel API](../../../rest_apis/multiplayer/channel/) page.)\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 * 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 Julia model).\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 serviceOptions.filter = qs; //shouldn't be able to over-ride\n var httpOptions = $.extend(true, {}, serviceOptions, options);\n httpOptions = urlConfig.addAutoRestoreHeader(httpOptions);\n\n return http.splitGet(outputModifier, httpOptions);\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 Julia model).\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);\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, 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 in your Julia 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 /**\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 * **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 a method from the model.\n *\n * Depending on the language in which you have written your model, the 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 * // method \"solve\" takes no arguments\n * rs.do('solve');\n * // method \"echo\" takes one argument, a string\n * rs.do('echo', ['hello']);\n * // method \"echo\" takes one argument, a string\n * rs.do('echo', 'hello');\n * // method \"sumArray\" takes one argument, an array\n * rs.do('sumArray', [[4,2,1]]);\n * // method \"add\" takes two arguments, both integers\n * rs.do({ name:'add', params:[2,4] });\n *\n * **Parameters**\n * @param {String} operation Name of method.\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 methods from the model, sequentially.\n *\n * Depending on the language in which you have written your model, the methods 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 * // methods \"initialize\" and \"solve\" do not take any arguments\n * rs.serial(['initialize', 'solve']);\n * // methods \"init\" and \"reset\" take two arguments each\n * rs.serial([ { name: 'init', params: [1,2] },\n * { name: 'reset', params: [2,3] }]);\n * // method \"init\" takes two arguments,\n * // method \"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 methods take parameters, pass an array of the method names (strings). If any of the methods do take parameters, pass an array of objects, each of which contains a method 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}\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 doSingleOp = function () {\n var op = ops.shift();\n var arg = args.shift();\n\n me.do(op, arg, {\n success: function () {\n if (ops.length) {\n doSingleOp();\n } else {\n $d.resolve.apply(this, arguments);\n postOptions.success.apply(this, arguments);\n }\n },\n error: function () {\n $d.reject.apply(this, arguments);\n postOptions.error.apply(this, arguments);\n }\n });\n };\n\n doSingleOp();\n\n return $d.promise();\n },\n\n /**\n * Call several methods from the model, executing them in parallel.\n *\n * Depending on the language in which you have written your model, the methods 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 * // methods \"solve\" and \"reset\" do not take any arguments\n * rs.parallel(['solve', 'reset']);\n * // methods \"add\" and \"subtract\" take two arguments each\n * rs.parallel([ { name: 'add', params: [1,2] },\n * { name: 'subtract', params:[2,3] }]);\n * // methods \"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 methods take parameters, pass an array of the method names (as strings). If any of the methods do take parameters, you have two options. You can pass an array of objects, each of which contains a method name and its own (possibly empty) array of parameters. Alternatively, you can pass a single object with the method 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}\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 $.when.apply(this, queue)\n .then(function () {\n $d.resolve.apply(this, arguments);\n postOptions.success.apply(this.arguments);\n })\n .fail(function () {\n $d.reject.apply(this, arguments);\n postOptions.error.apply(this.arguments);\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: }`.\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 /**\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 replay or clone runs. It 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 * 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'];\n\n if (api === '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 + api + '/';\n\n if ($.inArray(api, 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 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 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 * @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 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 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 session = getStore(options).get(EPI_SESSION_KEY) || '{}';\n // return JSON.parse(session);\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 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: session.auth_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 };\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};\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 * returns operations of the form `[[op1,op2], [arg1, arg2]]`\n * @param {Object|Array|String} operations operations to perform\n * @param {Array} args arguments for operation\n * @return {String} Matrix-format query parameters\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/always-new-strategy.js","src/managers/run-strategies/conditional-creation-strategy.js","src/managers/run-strategies/multiplayer-strategy.js","src/managers/run-strategies/new-if-initialized-strategy.js","src/managers/run-strategies/new-if-missing-strategy.js","src/managers/run-strategies/new-if-persisted-strategy.js","src/managers/run-strategies/none-strategy.js","src/managers/run-strategies/persistent-single-player-strategy.js","src/managers/run-strategies/strategies-map.js","src/managers/scenario-manager.js","src/managers/special-operations.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/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","util","factory","transport","store","service","manager","strategy","load","require","global","SKIP_ENV_LOAD","query","run","classFrom","Transport","Ajax","URL","Config","Run","File","Variables","Data","Auth","World","State","User","Member","Asset","Group","Introspect","Cookie","Store","ScenarioManager","RunManager","AuthManager","WorldManager","identity","ChannelManager","Channel","version","api","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","logout","statusText","status","reject","handleSuccess","response","access_token","userInfo","oldGroups","getSession","groups","userGroupOpts","auth","user","project","isTeamMember","parent_account_id","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","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","validTypes","world","general","chat","getFromSessionOrError","value","sessionKeyName","settings","__super","EpicenterChannelManager","constructor","urlOpts","protocol","host","userName","userProp","ext","authorization","channelOpts","allowAllChannels","baseParts","channelType","getGroupChannel","baseTopic","getWorldChannel","worldid","getUserChannel","userid","getPresenceChannel","lastPingTime","PING_INTERVAL","notification","incomingUserId","online","Date","valueOf","setInterval","publish","now","getDataChannel","collection","oldsubs","topic","context","callbackWithCleanData","payload","meta","path","subType","date","actualData","EPI_SESSION_KEY","STRATEGY_SESSION_KEY","patchRunService","patched","orig","do","operation","reservedOps","specialOperations","RunService","StrategyCtor","strategiesMap","getRun","reset","runServiceOptions","ConditionalStrategy","Strategy","runService","createIf","headers","setRunInSession","sessionKey","getStore","set","stringify","runId","Base","keyNames","condition","_auth","runOptions","runOptionsWithScope","userSession","scope","opt","create","freshlyCreated","sessionStore","runSession","get","_loadAndCheck","shouldCreate","msg","IdentityStrategy","WorldApiAdapter","synchronous","_loadRun","worldApi","curUserId","curGroupName","getCurrentWorldForUser","newRunForWorld","model","dtd","loadRunFromWorld","getCurrentRunId","filter","serverError","getResponseHeader","initialized","StorageFactory","StateApi","_store","stateApi","_restoreRun","_getAllRuns","user.id","scope.group","runs","dateComp","a","b","latestRun","sort","shouldReplay","replay","resp","new-if-initialized","new-if-persisted","new-if-missing","always-new","multiplayer","persistent-single-player","none","validFilter","saved","getRuns","loadVariables","vars","include","save","_getService","archive","buildStrategy","worldId","resolveWith","WorldApi","getCurrentWorld","getCurrentRun","getAndRestoreLatestRun","currentWorldId","runOpts","rm","ConfigService","TransportFactory","config","uploadBody","fileName","contents","boundary","body","uploadFileOptions","filePath","pop","serviceOptions","folderType","upload","urlConfig","contentType","accountPath","projectPath","Authorization","http","publicAsyncAPI","getContents","put","replaceExisting","prom","post","xhr","conflictStatus","remove","delete","rename","newName","patch","apiEndpoint","fullUrl","processData","transportOptions","assetApiParams","scopeConfig","validateFilename","filename","validateUrlParams","partKeys","buildUrl","parts","method","toLowerCase","FormData","urlOptions","createOptions","publicAPI","getServiceOptions","getOptions","list","files","fullPathFiles","file","assetUrl","password","statusMessage","postParams","topicResolver","channelOptions","makeName","channelName","topics","subscriptionIds","push","returnObjs","console","warn","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","rutil","VariablesService","IntrospectionService","autoRestore","getFilterURL","toMatrixFormat","addAutoRestoreHeader","isFilterRunId","type","autorestoreOpts","X-AutoRestore","splitGet","splitGetFactory","setFilterOrThrowError","runApiParams","oldSuccess","qs","filters","attributes","opsArgs","postOptions","result","normalizeOperations","prms","args","ops","serial","operations","opParams","doSingleOp","op","shift","arg","parallel","queue","when","introspect","introspectionConfig","introspection","publicSyncAPI","getCurrentConfig","variables","vs","rest","slice","parseRunIdOrError","replayOptions","action","clone","getLocalHost","existingFn","localHostFn","isFunction","isLocal","epiVersion","location","pathname","UrlConfigService","envConf","isLocalhost","actingHost","API_PROTOCOL","HOST_API_MAPPING","forio.com","foriodev.com","publicExports","apiHost","isCustomDomain","pathHasApp","appPath","accnt","prj","versionPath","PROJECT_APIS","actualProtocol","configProtocol","apiPath","inArray","toQFilter","toIdFilters","getFilters","toQueryFormat","threshold","getById","variable","apiBase","assignmentEndpoint","projectEndpoint","setIdFilterOrThrowError","validateModelOrThrowError","worldApiParams","validParams","update","whitelist","updateOptions","deleteOptions","updateConfig","getWorldsForUser","addUsers","users","u","isObject","updateUser","patchOptions","removeUser","worlds","lastModified","currentWorld","deleteRun","currentRunOptions","autoAssign","maxUsers","getProjectSettings","document","cookie","newCookie","hostname","validHost","domain","setOptions","encodeURIComponent","cookieReg","RegExp","exec","decodeURIComponent","remOptions","destroy","aKeys","nIdx","cookieKey","optionUtils","EPI_MANAGER_KEY","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","current","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"],"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 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');\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');\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');\n\nF.manager.strategy['always-new'] = require('./managers/run-strategies/always-new-strategy');\nF.manager.strategy['conditional-creation'] = require('./managers/run-strategies/conditional-creation-strategy');\nF.manager.strategy.identity = require('./managers/run-strategies/none-strategy');\nF.manager.strategy['new-if-missing'] = require('./managers/run-strategies/new-if-missing-strategy');\nF.manager.strategy['new-if-missing'] = require('./managers/run-strategies/new-if-missing-strategy');\nF.manager.strategy['new-if-persisted'] = require('./managers/run-strategies/new-if-persisted-strategy');\nF.manager.strategy['new-if-initialized'] = require('./managers/run-strategies/new-if-initialized-strategy');\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\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) {\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 });\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 sessionInfo = {\n auth_token: token,\n account: adapterOptions.account,\n project: project,\n userId: userInfo.user_id,\n groups: oldGroups,\n isTeamMember: isTeamMember\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', 401, data);\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);\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 handleGroupList(groups);\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 * 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 * 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 * While you can work directly with the Channel Manager through Node.js (for example, `require('manager/channel-manager')`) -- or even work directly with `$.cometd` and Epicenter's underlying [Push Channel API](../../../rest_apis/multiplayer/channel/) -- most often it will be easiest to work with the [Epicenter Channel Manager](../epicenter-channel-manager/). The Epicenter Channel Manager is a wrapper that instantiates a Channel Manager with Epicenter-specific defaults.\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 the channel, 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 channel = cm.getChannel();\n *\n * channel.subscribe('topic', callback);\n * channel.publish('topic', { myData: 100 });\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. See 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). There are two main use cases for the channel: event notifications and chat messages.\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 probably 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 * To use the Epicenter Channel Manager: instantiate it, get the channel of the scope you want ([user](../../../glossary/#users), [world](../../../glossary/#world), or [group](../../../glossary/#groups)), 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 * gc.subscribe('broadcasts', callback);\n *\n * For additional background on Epicenter's push channel, see the introductory notes on the [Push Channel API](../../../rest_apis/multiplayer/channel/) page.\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 classFrom = require('../util/inherit');\nvar urlService = require('../service/url-config-service');\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};\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 urlOpts = urlService(defaultCometOptions.server);\n if (!defaultCometOptions.url) {\n //Default epicenter cometd endpoint\n defaultCometOptions.url = urlOpts.protocol + '://' + urlOpts.host + '/channel/subscribe';\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 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 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 and world. 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 shared world are also online.\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 * model: 'model.eqn'\n * });\n * worldManager.getCurrentWorld().then(function (worldObject, worldService) {\n * var presenceChannel = cm.getPresenceChannel(worldObject);\n * presenceChannel.on('presence', function (evt, notification) {\n * console.log(notification.online, notification.userId);\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} userid (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 getPresenceChannel: function (world, userid, 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\n var session = this.sessionManager.getMergedOptions(this.options);\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].join('/');\n var channel = __super.getChannel.call(this, { base: baseTopic });\n\n var lastPingTime = { };\n\n var PING_INTERVAL = 6000;\n channel.subscribe('internal-ping-channel', function (notification) {\n var incomingUserId = notification.data.user;\n if (!lastPingTime[incomingUserId] && incomingUserId !== userid) {\n channel.trigger('presence', { userId: incomingUserId, online: true });\n }\n lastPingTime[incomingUserId] = (new Date()).valueOf();\n });\n\n setInterval(function () {\n channel.publish('internal-ping-channel', { user: userid });\n\n $.each(lastPingTime, function (key, value) {\n var now = (new Date()).valueOf();\n if (value && value + (PING_INTERVAL * 2) < now) {\n lastPingTime[key] = null;\n channel.trigger('presence', { userId: key, online: false });\n }\n });\n }, PING_INTERVAL);\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 };\n var actualData = payload.data.data;\n if (actualData.data) { //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/), the RESFTful [Run API](../../../rest_apis/aggregate_run_api) and the [Model Run API](../../../rest_apis/other_apis/model_apis/run/). 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. (Note that many of the Epicenter sample projects use a Run Service directly, because generally the sample projects are played in one end user session and don't care about run states or run strategies.)\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 pass a `files` object with the names of the files, for example: `\"files\": {\"data\": \"myExtraData.xls\"}`. (Note that you'll also need to add this same files object to your Vensim [configuration file](../../../model_code/vensim/).) See the [underlying Model Run API](../../../rest_apis/other_apis/model_apis/run/#post-creating-a-new-run-for-this-project) for additional information.\n*\n* * `strategy`: (optional) Run creation strategy for when to create a new run and when to reuse an end user's existing run. See [Run Manager Strategies](../strategies/) for details. Defaults to `new-if-initialized`.\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.\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: 'always-new',\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\n'use strict';\nvar strategiesMap = require('./run-strategies/strategies-map');\nvar specialOperations = require('./special-operations');\nvar RunService = require('../service/run-api-service');\n\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\n\nvar defaults = {\n /**\n * Run creation strategy for when to create a new run and when to reuse an end user's existing run. See [Run Manager Strategies](../strategies/) for details. Defaults to `new-if-initialized`.\n * @type {String}\n */\n\n strategy: 'new-if-initialized'\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 {\n this.run = new RunService(this.options.run);\n }\n\n patchRunService(this.run, this);\n\n var StrategyCtor = typeof this.options.strategy === 'function' ? this.options.strategy : strategiesMap[this.options.strategy];\n\n if (!StrategyCtor) {\n throw new Error('Specified run creation strategy was invalid:', this.options.strategy);\n }\n\n this.strategy = new StrategyCtor(this.run, this.options);\n}\n\nRunManager.prototype = {\n /**\n * Returns the run object for a 'good' run.\n *\n * A good run is defined by the strategy. For example, if the strategy is `always-new`, the call\n * to `getRun()` always returns a newly created run; if the strategy is `new-if-persisted`,\n * `getRun()` creates a new run if the previous run is in a persisted state, otherwise\n * it returns the previous run. 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 * @return {$promise} Promise to complete the call.\n */\n getRun: function () {\n return this.strategy\n .getRun();\n },\n\n /**\n * Returns the run object for a new run, regardless of strategy: force creation of a new run.\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} runServiceOptions The options object to configure the Run Service. See [Run API Service](../run-api-service/) for more.\n * @return {Promise}\n */\n reset: function (runServiceOptions) {\n return this.strategy.reset(runServiceOptions);\n }\n};\n\nmodule.exports = RunManager;\n","/**\n * The `always-new` 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\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 (runService, options) {\n __super.constructor.call(this, runService, 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","'use strict';\n\nvar Base = require('./none-strategy');\nvar SessionManager = require('../../store/session-manager');\nvar classFrom = require('../../util/inherit');\nvar AuthManager = require('../auth-manager');\n\nvar keyNames = require('../key-names');\n\nvar defaults = {\n sessionKey: keyNames.STRATEGY_SESSION_KEY,\n path: ''\n};\n\nfunction setRunInSession(sessionKey, run, sessionManager) {\n sessionManager.getStore().set(sessionKey, JSON.stringify({ runId: run.id }));\n}\n\n/**\n* Conditional Creation Strategy\n* This strategy will try to get the run stored in the cookie and\n* evaluate if needs to create a new run by calling the 'condition' function\n*/\n\nvar Strategy = classFrom(Base, {\n constructor: function Strategy(runService, condition, options) {\n if (condition == null) { //eslint-disable-line\n //TODO: not sure why this is explicitly ==\n throw new Error('Conditional strategy needs a condition to create a run');\n }\n\n this._auth = new AuthManager();\n this.run = runService;\n this.condition = typeof condition !== 'function' ? function () { return condition; } : condition;\n this.options = $.extend(true, {}, defaults, options);\n this.sessionManager = new SessionManager(options);\n this.runOptions = this.options.run;\n },\n\n runOptionsWithScope: function () {\n var userSession = this._auth.getCurrentUserSessionInfo();\n return $.extend({\n scope: { group: userSession.groupName }\n }, this.runOptions);\n },\n\n reset: function (runServiceOptions) {\n var me = this;\n var opt = this.runOptionsWithScope();\n\n return this.run\n .create(opt, runServiceOptions)\n .then(function (run) {\n setRunInSession(me.options.sessionKey, run, me.sessionManager);\n run.freshlyCreated = true;\n return run;\n });\n },\n\n getRun: function () {\n var sessionStore = this.sessionManager.getStore();\n var runSession = JSON.parse(sessionStore.get(this.options.sessionKey));\n var me = this;\n if (runSession && runSession.runId) {\n return this._loadAndCheck(runSession).fail(function () {\n return me.reset(); //if it got the wrong cookie for e.g.\n });\n } else {\n return this.reset();\n }\n },\n\n _loadAndCheck: function (runSession) {\n var shouldCreate = false;\n var me = this;\n\n return this.run\n .load(runSession.runId, null, {\n success: function (run, msg, headers) {\n shouldCreate = me.condition(run, headers);\n }\n })\n .then(function (run) {\n if (shouldCreate) {\n var opt = me.runOptionsWithScope();\n return me.run.create(opt)\n .then(function (run) {\n setRunInSession(me.options.sessionKey, run, me.sessionManager);\n run.freshlyCreated = true;\n return run;\n });\n }\n return run;\n });\n }\n});\n\nmodule.exports = Strategy;\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/).\n * \n * Using this strategy means that end users in projects with multiplayer worlds always see the most current run and world. 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');\nvar AuthManager = require('../auth-manager');\n\nvar defaults = {\n store: {\n synchronous: true\n }\n};\n\nvar Strategy = classFrom(IdentityStrategy, {\n\n constructor: function (runService, options) {\n this.runService = runService;\n this.options = $.extend(true, {}, defaults, options);\n this._auth = new AuthManager();\n this._loadRun = this._loadRun.bind(this);\n this.worldApi = new WorldApiAdapter(this.options.run);\n },\n\n reset: function () {\n var session = this._auth.getCurrentUserSessionInfo();\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);\n }.bind(this));\n },\n\n getRun: function () {\n var session = this._auth.getCurrentUserSessionInfo();\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\n return worldApi.getCurrentRunId({ model: model, filter: world.id })\n .then(me._loadRun)\n .then(dtd.resolve)\n .fail(dtd.reject);\n };\n\n var serverError = function (error) {\n // is this possible?\n 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 _loadRun: function (id, options) {\n return this.runService.load(id, null, options);\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 for Vensim models; it can be set manually for other models.\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\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 (runService, options) {\n __super.constructor.call(this, runService, this.createIf, options);\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-missing` 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.\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 other words, you care whether you have a run, but as long as you have one, you are certain that this run is the one you are interested in. \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\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 (runService, options) {\n __super.constructor.call(this, runService, 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 * 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, if they are idle for longer than the **Model Session Timeout**, that 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\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 (runService, options) {\n __super.constructor.call(this, runService, this.createIf, options);\n },\n\n createIf: function (run, headers) {\n return headers.getResponseHeader('pragma') === 'persistent';\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 * Also, this strategy is necessary if you are working with a multiplayer project and using the [World Manager](../world-manager/) — or other, similar situations where you do not have direct control over creating the [Run Service](../run-api-service/) instance.\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 (runService, options) {\n this.runService = runService;\n },\n\n reset: function () {\n // return a newly created run\n return $.Deferred().resolve().promise();\n },\n\n getRun: function () {\n // return a usable run\n return $.Deferred().resolve(this.runService).promise();\n }\n});\n","/**\n * The `persistent-single-player` 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 your project executes your 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). It is useful if end users play with your project for an extended period of time, possibly over several sessions.\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 * * If the most recent run is currently in the database, bring it back into memory so that the end user can continue working with it. (See more background on [Run Persistence](../../../run_persistence/), or read more on the underlying [State API](../../../rest_apis/other_apis/model_apis/state/) for bringing runs from the database back into memory.) \n */\n\n'use strict';\n\nvar classFrom = require('../../util/inherit');\nvar IdentityStrategy = require('./none-strategy');\nvar StorageFactory = require('../../store/store-factory');\nvar StateApi = require('../../service/state-api-adapter');\nvar AuthManager = require('../auth-manager');\n\nvar keyNames = require('../key-names');\n\nvar defaults = {\n store: {\n synchronous: true\n }\n};\n\nvar Strategy = classFrom(IdentityStrategy, {\n constructor: function Strategy(runService, options) {\n this.run = runService;\n this.options = $.extend(true, {}, defaults, options);\n this.runOptions = this.options.run;\n this._store = new StorageFactory(this.options.store);\n this.stateApi = new StateApi();\n this._auth = new AuthManager();\n\n this._loadAndCheck = this._loadAndCheck.bind(this);\n this._restoreRun = this._restoreRun.bind(this);\n this._getAllRuns = this._getAllRuns.bind(this);\n this._loadRun = this._loadRun.bind(this);\n },\n\n reset: function (runServiceOptions) {\n var session = this._auth.getCurrentUserSessionInfo();\n var opt = $.extend({\n scope: { group: session.groupName }\n }, this.runOptions);\n\n return this.run\n .create(opt, runServiceOptions)\n .then(function (run) {\n run.freshlyCreated = true;\n return run;\n });\n },\n\n getRun: function () {\n return this._getAllRuns()\n .then(this._loadAndCheck);\n },\n\n _getAllRuns: function () {\n var session = JSON.parse(this._store.get(keyNames.EPI_SESSION_KEY) || '{}');\n return this.run.query({\n 'user.id': session.userId || '0000',\n 'scope.group': session.groupName\n });\n },\n\n _loadAndCheck: function (runs) {\n if (!runs || !runs.length) {\n return this.reset();\n }\n\n var dateComp = function (a, b) { return new Date(b.date) - new Date(a.date); };\n var latestRun = runs.sort(dateComp)[0];\n var me = this;\n var shouldReplay = false;\n\n return this.run.load(latestRun.id, null, {\n success: function (run, msg, headers) {\n shouldReplay = headers.getResponseHeader('pragma') === 'persistent';\n }\n }).then(function (run) {\n return shouldReplay ? me._restoreRun(run.id) : run;\n });\n },\n\n _restoreRun: function (runId) {\n var me = this;\n return this.stateApi.replay({ runId: runId })\n .then(function (resp) {\n return me._loadRun(resp.run);\n });\n },\n\n _loadRun: function (id, options) {\n return this.run.load(id, null, options);\n }\n\n});\n\nmodule.exports = Strategy;\n","module.exports = {\n 'new-if-initialized': require('./new-if-initialized-strategy'),\n 'new-if-persisted': require('./new-if-persisted-strategy'),\n 'new-if-missing': require('./new-if-missing-strategy'),\n 'always-new': require('./always-new-strategy'),\n multiplayer: require('./multiplayer-strategy'),\n 'persistent-single-player': require('./persistent-single-player-strategy'),\n none: require('./none-strategy')\n};\n","'use strict';\nvar RunService = require('../service/run-api-service');\n\nvar defaults = {\n validFilter: { saved: true }\n};\n\nfunction ScenarioManager(options) {\n this.options = $.extend(true, {}, defaults, options);\n this.runService = this.options.run || new RunService(this.options);\n}\n\nScenarioManager.prototype = {\n getRuns: function (filter) {\n this.filter = $.extend(true, {}, this.options.validFilter, filter);\n return this.runService.query(this.filter);\n },\n\n loadVariables: function (vars) {\n return this.runService.query(this.filter, { include: vars });\n },\n\n save: function (run, meta) {\n return this._getService(run).save($.extend(true, {}, { saved: true }, meta));\n },\n\n archive: function (run) {\n return this._getService(run).save({ saved: false });\n },\n\n _getService: function (run) {\n if (typeof run === 'string') {\n return new RunService($.extend(true, {}, this.options, { filter: run }));\n }\n\n if (typeof run === 'object' && run instanceof RunService) {\n return run;\n }\n\n throw new Error('Save method requires a run service or a runId');\n },\n\n getRun: function (runId) {\n return new RunService($.extend(true, {}, this.options, { filter: runId }));\n }\n};\n\nmodule.exports = ScenarioManager;\n\n","'use strict';\n\n\nmodule.exports = {\n reset: function (params, options, manager) {\n return manager.reset(options);\n }\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(runService, options) {\n this.runService = runService;\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 () {\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 me.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({model: '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\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 * The Asset API Service (https://forio.com/epicenter/docs/public/api_adapters/generated/asset-api-adapter/) is typically used for all project use cases, and it's unlikely this File Service will be used directly except by Admin tools (e.g. Flow Inspector).\n *\n * Partially implemented.\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 var upload = uploadBody(fileName, contents);\n\n return $.extend(true, {}, serviceOptions, options, {\n url: urlConfig.getAPIPath('file') + path,\n data: upload.body,\n contentType: 'multipart/form-data; boundary=' + upload.boundary\n });\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} 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\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} 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 * 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 * Typically, you use the [Epicenter Channel Manager](../epicenter-channel-manager/) to create or retrieve channels, then use the Channel Service `subscribe()` and `publish()` methods to listen to or update data. (For additional background on Epicenter's push channel, see the introductory notes on the [Push Channel API](../../../rest_apis/multiplayer/channel/) page.)\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 * 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 Julia model).\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 serviceOptions.filter = qs; //shouldn't be able to over-ride\n var httpOptions = $.extend(true, {}, serviceOptions, options);\n httpOptions = urlConfig.addAutoRestoreHeader(httpOptions);\n\n return http.splitGet(outputModifier, httpOptions);\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 Julia model).\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);\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, 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 in your Julia 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 /**\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 * **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 a method from the model.\n *\n * Depending on the language in which you have written your model, the 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 * // method \"solve\" takes no arguments\n * rs.do('solve');\n * // method \"echo\" takes one argument, a string\n * rs.do('echo', ['hello']);\n * // method \"echo\" takes one argument, a string\n * rs.do('echo', 'hello');\n * // method \"sumArray\" takes one argument, an array\n * rs.do('sumArray', [[4,2,1]]);\n * // method \"add\" takes two arguments, both integers\n * rs.do({ name:'add', params:[2,4] });\n *\n * **Parameters**\n * @param {String} operation Name of method.\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 methods from the model, sequentially.\n *\n * Depending on the language in which you have written your model, the methods 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 * // methods \"initialize\" and \"solve\" do not take any arguments\n * rs.serial(['initialize', 'solve']);\n * // methods \"init\" and \"reset\" take two arguments each\n * rs.serial([ { name: 'init', params: [1,2] },\n * { name: 'reset', params: [2,3] }]);\n * // method \"init\" takes two arguments,\n * // method \"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 methods take parameters, pass an array of the method names (strings). If any of the methods do take parameters, pass an array of objects, each of which contains a method 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}\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 doSingleOp = function () {\n var op = ops.shift();\n var arg = args.shift();\n\n me.do(op, arg, {\n success: function () {\n if (ops.length) {\n doSingleOp();\n } else {\n $d.resolve.apply(this, arguments);\n postOptions.success.apply(this, arguments);\n }\n },\n error: function () {\n $d.reject.apply(this, arguments);\n postOptions.error.apply(this, arguments);\n }\n });\n };\n\n doSingleOp();\n\n return $d.promise();\n },\n\n /**\n * Call several methods from the model, executing them in parallel.\n *\n * Depending on the language in which you have written your model, the methods 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 * // methods \"solve\" and \"reset\" do not take any arguments\n * rs.parallel(['solve', 'reset']);\n * // methods \"add\" and \"subtract\" take two arguments each\n * rs.parallel([ { name: 'add', params: [1,2] },\n * { name: 'subtract', params:[2,3] }]);\n * // methods \"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 methods take parameters, pass an array of the method names (as strings). If any of the methods do take parameters, you have two options. You can pass an array of objects, each of which contains a method name and its own (possibly empty) array of parameters. Alternatively, you can pass a single object with the method 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}\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 $.when.apply(this, queue)\n .then(function () {\n $d.resolve.apply(this, arguments);\n postOptions.success.apply(this.arguments);\n })\n .fail(function () {\n $d.reject.apply(this, arguments);\n postOptions.error.apply(this.arguments);\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: }`.\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 /**\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 replay or clone runs. It 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 * 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'];\n\n if (api === '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 + api + '/';\n\n if ($.inArray(api, 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 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 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 * @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 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 session = getStore(options).get(EPI_SESSION_KEY) || '{}';\n // return JSON.parse(session);\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 };\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};\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 * returns operations of the form `[[op1,op2], [arg1, arg2]]`\n * @param {Object|Array|String} operations operations to perform\n * @param {Array} args arguments for operation\n * @return {String} Matrix-format query parameters\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 diff --git a/package.json b/package.json index 3369a6f1..86f60abf 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "epicenter-js", - "version": "2.0.0", + "version": "2.1.0", "repository": { "type": "git", "url": "https://github.com/forio/epicenter-js-libs"