parse-server 6.0.0-alpha.2 → 6.0.0-alpha.21
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +45 -17
- package/lib/AccountLockout.js +11 -26
- package/lib/Adapters/AdapterLoader.js +8 -14
- package/lib/Adapters/Analytics/AnalyticsAdapter.js +2 -8
- package/lib/Adapters/Auth/AuthAdapter.js +7 -16
- package/lib/Adapters/Auth/OAuth1Client.js +32 -57
- package/lib/Adapters/Auth/apple.js +6 -22
- package/lib/Adapters/Auth/facebook.js +7 -37
- package/lib/Adapters/Auth/gcenter.js +8 -37
- package/lib/Adapters/Auth/github.js +7 -10
- package/lib/Adapters/Auth/google.js +11 -34
- package/lib/Adapters/Auth/gpgames.js +5 -8
- package/lib/Adapters/Auth/httpsRequest.js +1 -7
- package/lib/Adapters/Auth/index.js +20 -65
- package/lib/Adapters/Auth/instagram.js +5 -9
- package/lib/Adapters/Auth/janraincapture.js +8 -12
- package/lib/Adapters/Auth/janrainengage.js +7 -11
- package/lib/Adapters/Auth/keycloak.js +5 -19
- package/lib/Adapters/Auth/ldap.js +1 -15
- package/lib/Adapters/Auth/line.js +7 -10
- package/lib/Adapters/Auth/linkedin.js +7 -12
- package/lib/Adapters/Auth/meetup.js +7 -10
- package/lib/Adapters/Auth/microsoft.js +7 -10
- package/lib/Adapters/Auth/oauth2.js +6 -18
- package/lib/Adapters/Auth/phantauth.js +8 -10
- package/lib/Adapters/Auth/qq.js +7 -13
- package/lib/Adapters/Auth/spotify.js +7 -14
- package/lib/Adapters/Auth/twitter.js +5 -15
- package/lib/Adapters/Auth/vkontakte.js +9 -15
- package/lib/Adapters/Auth/wechat.js +7 -10
- package/lib/Adapters/Auth/weibo.js +7 -11
- package/lib/Adapters/Cache/CacheAdapter.js +4 -12
- package/lib/Adapters/Cache/InMemoryCache.js +5 -19
- package/lib/Adapters/Cache/InMemoryCacheAdapter.js +1 -11
- package/lib/Adapters/Cache/LRUCache.js +1 -11
- package/lib/Adapters/Cache/NullCacheAdapter.js +1 -8
- package/lib/Adapters/Cache/RedisCacheAdapter.js +46 -87
- package/lib/Adapters/Cache/SchemaCache.js +1 -6
- package/lib/Adapters/Email/MailAdapter.js +2 -7
- package/lib/Adapters/Files/FilesAdapter.js +7 -21
- package/lib/Adapters/Files/GridFSBucketAdapter.js +6 -44
- package/lib/Adapters/Files/GridStoreAdapter.js +1 -1
- package/lib/Adapters/Logger/LoggerAdapter.js +2 -11
- package/lib/Adapters/Logger/WinstonLogger.js +3 -30
- package/lib/Adapters/Logger/WinstonLoggerAdapter.js +5 -16
- package/lib/Adapters/MessageQueue/EventEmitterMQ.js +3 -20
- package/lib/Adapters/PubSub/EventEmitterPubSub.js +1 -16
- package/lib/Adapters/PubSub/PubSubAdapter.js +2 -9
- package/lib/Adapters/PubSub/RedisPubSub.js +13 -10
- package/lib/Adapters/Push/PushAdapter.js +2 -8
- package/lib/Adapters/Storage/Mongo/MongoCollection.js +12 -37
- package/lib/Adapters/Storage/Mongo/MongoSchemaCollection.js +26 -79
- package/lib/Adapters/Storage/Mongo/MongoStorageAdapter.js +78 -209
- package/lib/Adapters/Storage/Mongo/MongoTransform.js +82 -371
- package/lib/Adapters/Storage/Postgres/PostgresClient.js +1 -13
- package/lib/Adapters/Storage/Postgres/PostgresConfigParser.js +1 -20
- package/lib/Adapters/Storage/Postgres/PostgresStorageAdapter.js +119 -446
- package/lib/Adapters/Storage/Postgres/sql/index.js +4 -7
- package/lib/Adapters/Storage/StorageAdapter.js +1 -1
- package/lib/Adapters/WebSocketServer/WSAdapter.js +3 -12
- package/lib/Adapters/WebSocketServer/WSSAdapter.js +7 -12
- package/lib/Auth.js +54 -121
- package/lib/ClientSDK.js +3 -11
- package/lib/Config.js +69 -113
- package/lib/Controllers/AdaptableController.js +6 -18
- package/lib/Controllers/AnalyticsController.js +1 -9
- package/lib/Controllers/CacheController.js +3 -23
- package/lib/Controllers/DatabaseController.js +147 -345
- package/lib/Controllers/FilesController.js +5 -34
- package/lib/Controllers/HooksController.js +1 -51
- package/lib/Controllers/LiveQueryController.js +4 -23
- package/lib/Controllers/LoggerController.js +15 -54
- package/lib/Controllers/ParseGraphQLController.js +49 -104
- package/lib/Controllers/PushController.js +20 -59
- package/lib/Controllers/SchemaController.js +154 -344
- package/lib/Controllers/UserController.js +11 -72
- package/lib/Controllers/index.js +19 -68
- package/lib/Controllers/types.js +1 -1
- package/lib/Deprecator/Deprecations.js +1 -8
- package/lib/Deprecator/Deprecator.js +9 -18
- package/lib/GraphQL/ParseGraphQLSchema.js +16 -100
- package/lib/GraphQL/ParseGraphQLServer.js +2 -29
- package/lib/GraphQL/helpers/objectsMutations.js +2 -12
- package/lib/GraphQL/helpers/objectsQueries.js +18 -76
- package/lib/GraphQL/loaders/defaultGraphQLMutations.js +1 -9
- package/lib/GraphQL/loaders/defaultGraphQLQueries.js +1 -8
- package/lib/GraphQL/loaders/defaultGraphQLTypes.js +9 -115
- package/lib/GraphQL/loaders/defaultRelaySchema.js +6 -18
- package/lib/GraphQL/loaders/filesMutations.js +2 -19
- package/lib/GraphQL/loaders/functionsMutations.js +6 -17
- package/lib/GraphQL/loaders/parseClassMutations.js +6 -44
- package/lib/GraphQL/loaders/parseClassQueries.js +1 -26
- package/lib/GraphQL/loaders/parseClassTypes.js +10 -64
- package/lib/GraphQL/loaders/schemaDirectives.js +1 -17
- package/lib/GraphQL/loaders/schemaMutations.js +1 -20
- package/lib/GraphQL/loaders/schemaQueries.js +1 -14
- package/lib/GraphQL/loaders/schemaTypes.js +2 -6
- package/lib/GraphQL/loaders/usersMutations.js +6 -28
- package/lib/GraphQL/loaders/usersQueries.js +4 -26
- package/lib/GraphQL/parseGraphQLUtils.js +6 -19
- package/lib/GraphQL/transformers/className.js +1 -4
- package/lib/GraphQL/transformers/constraintType.js +1 -20
- package/lib/GraphQL/transformers/inputType.js +1 -20
- package/lib/GraphQL/transformers/mutation.js +6 -51
- package/lib/GraphQL/transformers/outputType.js +1 -20
- package/lib/GraphQL/transformers/query.js +6 -42
- package/lib/GraphQL/transformers/schemaFields.js +7 -34
- package/lib/KeyPromiseQueue.js +1 -12
- package/lib/LiveQuery/Client.js +1 -25
- package/lib/LiveQuery/Id.js +1 -7
- package/lib/LiveQuery/ParseCloudCodePublisher.js +13 -19
- package/lib/LiveQuery/ParseLiveQueryServer.js +92 -306
- package/lib/LiveQuery/ParsePubSub.js +1 -12
- package/lib/LiveQuery/ParseWebSocketServer.js +4 -26
- package/lib/LiveQuery/QueryTools.js +14 -116
- package/lib/LiveQuery/RequestSchema.js +1 -1
- package/lib/LiveQuery/SessionTokenCache.js +1 -17
- package/lib/LiveQuery/Subscription.js +4 -18
- package/lib/LiveQuery/equalObjects.js +2 -14
- package/lib/Options/Definitions.js +79 -10
- package/lib/Options/docs.js +23 -3
- package/lib/Options/index.js +4 -12
- package/lib/Options/parsers.js +1 -18
- package/lib/Page.js +1 -9
- package/lib/ParseMessageQueue.js +1 -10
- package/lib/ParseServer.js +144 -182
- package/lib/ParseServerRESTController.js +6 -33
- package/lib/PromiseRouter.js +16 -50
- package/lib/Push/PushQueue.js +3 -15
- package/lib/Push/PushWorker.js +7 -32
- package/lib/Push/utils.js +9 -38
- package/lib/RestQuery.js +105 -242
- package/lib/RestWrite.js +212 -377
- package/lib/Routers/AggregateRouter.js +14 -51
- package/lib/Routers/AnalyticsRouter.js +2 -8
- package/lib/Routers/AudiencesRouter.js +1 -15
- package/lib/Routers/ClassesRouter.js +3 -53
- package/lib/Routers/CloudCodeRouter.js +1 -19
- package/lib/Routers/FeaturesRouter.js +1 -10
- package/lib/Routers/FilesRouter.js +29 -76
- package/lib/Routers/FunctionsRouter.js +5 -28
- package/lib/Routers/GlobalConfigRouter.js +4 -18
- package/lib/Routers/GraphQLRouter.js +1 -14
- package/lib/Routers/HooksRouter.js +1 -29
- package/lib/Routers/IAPValidationRouter.js +6 -29
- package/lib/Routers/InstallationsRouter.js +2 -12
- package/lib/Routers/LogsRouter.js +4 -16
- package/lib/Routers/PagesRouter.js +69 -129
- package/lib/Routers/PublicAPIRouter.js +3 -62
- package/lib/Routers/PurgeRouter.js +1 -15
- package/lib/Routers/PushRouter.js +2 -18
- package/lib/Routers/RolesRouter.js +1 -7
- package/lib/Routers/SchemasRouter.js +4 -34
- package/lib/Routers/SecurityRouter.js +1 -12
- package/lib/Routers/SessionsRouter.js +3 -19
- package/lib/Routers/UsersRouter.js +58 -155
- package/lib/SchemaMigrations/DefinedSchemas.js +56 -115
- package/lib/SchemaMigrations/Migrations.js +2 -8
- package/lib/Security/Check.js +8 -16
- package/lib/Security/CheckGroup.js +4 -11
- package/lib/Security/CheckGroups/CheckGroupDatabase.js +8 -18
- package/lib/Security/CheckGroups/CheckGroupServerConfig.js +5 -15
- package/lib/Security/CheckGroups/CheckGroups.js +1 -4
- package/lib/Security/CheckRunner.js +22 -41
- package/lib/StatusHandler.js +12 -69
- package/lib/TestUtils.js +1 -6
- package/lib/Utils.js +27 -66
- package/lib/batch.js +17 -28
- package/lib/cache.js +1 -3
- package/lib/cli/definitions/parse-live-query-server.js +1 -3
- package/lib/cli/definitions/parse-server.js +1 -3
- package/lib/cli/parse-live-query-server.js +1 -6
- package/lib/cli/parse-server.js +11 -21
- package/lib/cli/utils/commander.js +13 -51
- package/lib/cli/utils/runner.js +1 -14
- package/lib/cloud-code/Parse.Cloud.js +71 -92
- package/lib/cryptoUtils.js +11 -19
- package/lib/defaults.js +2 -14
- package/lib/deprecated.js +1 -2
- package/lib/index.js +16 -34
- package/lib/logger.js +6 -13
- package/lib/middlewares.js +147 -151
- package/lib/password.js +6 -10
- package/lib/request.js +173 -2
- package/lib/requiredParameter.js +1 -3
- package/lib/rest.js +19 -41
- package/lib/triggers.js +54 -252
- package/lib/vendor/mongodbUrl.js +125 -305
- package/package.json +22 -19
- package/lib/cloud-code/HTTPResponse.js +0 -73
- package/lib/cloud-code/httpRequest.js +0 -192
package/lib/ParseServer.js
CHANGED
|
@@ -4,101 +4,62 @@ Object.defineProperty(exports, "__esModule", {
|
|
|
4
4
|
value: true
|
|
5
5
|
});
|
|
6
6
|
exports.default = void 0;
|
|
7
|
-
|
|
8
7
|
var _Options = require("./Options");
|
|
9
|
-
|
|
10
8
|
var _defaults = _interopRequireDefault(require("./defaults"));
|
|
11
|
-
|
|
12
9
|
var logging = _interopRequireWildcard(require("./logger"));
|
|
13
|
-
|
|
14
10
|
var _Config = _interopRequireDefault(require("./Config"));
|
|
15
|
-
|
|
16
11
|
var _PromiseRouter = _interopRequireDefault(require("./PromiseRouter"));
|
|
17
|
-
|
|
18
12
|
var _requiredParameter = _interopRequireDefault(require("./requiredParameter"));
|
|
19
|
-
|
|
20
13
|
var _AnalyticsRouter = require("./Routers/AnalyticsRouter");
|
|
21
|
-
|
|
22
14
|
var _ClassesRouter = require("./Routers/ClassesRouter");
|
|
23
|
-
|
|
24
15
|
var _FeaturesRouter = require("./Routers/FeaturesRouter");
|
|
25
|
-
|
|
26
16
|
var _FilesRouter = require("./Routers/FilesRouter");
|
|
27
|
-
|
|
28
17
|
var _FunctionsRouter = require("./Routers/FunctionsRouter");
|
|
29
|
-
|
|
30
18
|
var _GlobalConfigRouter = require("./Routers/GlobalConfigRouter");
|
|
31
|
-
|
|
32
19
|
var _GraphQLRouter = require("./Routers/GraphQLRouter");
|
|
33
|
-
|
|
34
20
|
var _HooksRouter = require("./Routers/HooksRouter");
|
|
35
|
-
|
|
36
21
|
var _IAPValidationRouter = require("./Routers/IAPValidationRouter");
|
|
37
|
-
|
|
38
22
|
var _InstallationsRouter = require("./Routers/InstallationsRouter");
|
|
39
|
-
|
|
40
23
|
var _LogsRouter = require("./Routers/LogsRouter");
|
|
41
|
-
|
|
42
24
|
var _ParseLiveQueryServer = require("./LiveQuery/ParseLiveQueryServer");
|
|
43
|
-
|
|
44
25
|
var _PagesRouter = require("./Routers/PagesRouter");
|
|
45
|
-
|
|
46
26
|
var _PublicAPIRouter = require("./Routers/PublicAPIRouter");
|
|
47
|
-
|
|
48
27
|
var _PushRouter = require("./Routers/PushRouter");
|
|
49
|
-
|
|
50
28
|
var _CloudCodeRouter = require("./Routers/CloudCodeRouter");
|
|
51
|
-
|
|
52
29
|
var _RolesRouter = require("./Routers/RolesRouter");
|
|
53
|
-
|
|
54
30
|
var _SchemasRouter = require("./Routers/SchemasRouter");
|
|
55
|
-
|
|
56
31
|
var _SessionsRouter = require("./Routers/SessionsRouter");
|
|
57
|
-
|
|
58
32
|
var _UsersRouter = require("./Routers/UsersRouter");
|
|
59
|
-
|
|
60
33
|
var _PurgeRouter = require("./Routers/PurgeRouter");
|
|
61
|
-
|
|
62
34
|
var _AudiencesRouter = require("./Routers/AudiencesRouter");
|
|
63
|
-
|
|
64
35
|
var _AggregateRouter = require("./Routers/AggregateRouter");
|
|
65
|
-
|
|
66
36
|
var _ParseServerRESTController = require("./ParseServerRESTController");
|
|
67
|
-
|
|
68
37
|
var controllers = _interopRequireWildcard(require("./Controllers"));
|
|
69
|
-
|
|
70
38
|
var _ParseGraphQLServer = require("./GraphQL/ParseGraphQLServer");
|
|
71
|
-
|
|
72
39
|
var _SecurityRouter = require("./Routers/SecurityRouter");
|
|
73
|
-
|
|
74
40
|
var _CheckRunner = _interopRequireDefault(require("./Security/CheckRunner"));
|
|
75
|
-
|
|
76
41
|
var _Deprecator = _interopRequireDefault(require("./Deprecator/Deprecator"));
|
|
77
|
-
|
|
78
42
|
var _DefinedSchemas = require("./SchemaMigrations/DefinedSchemas");
|
|
79
|
-
|
|
80
43
|
function _getRequireWildcardCache(nodeInterop) { if (typeof WeakMap !== "function") return null; var cacheBabelInterop = new WeakMap(); var cacheNodeInterop = new WeakMap(); return (_getRequireWildcardCache = function (nodeInterop) { return nodeInterop ? cacheNodeInterop : cacheBabelInterop; })(nodeInterop); }
|
|
81
|
-
|
|
82
44
|
function _interopRequireWildcard(obj, nodeInterop) { if (!nodeInterop && obj && obj.__esModule) { return obj; } if (obj === null || typeof obj !== "object" && typeof obj !== "function") { return { default: obj }; } var cache = _getRequireWildcardCache(nodeInterop); if (cache && cache.has(obj)) { return cache.get(obj); } var newObj = {}; var hasPropertyDescriptor = Object.defineProperty && Object.getOwnPropertyDescriptor; for (var key in obj) { if (key !== "default" && Object.prototype.hasOwnProperty.call(obj, key)) { var desc = hasPropertyDescriptor ? Object.getOwnPropertyDescriptor(obj, key) : null; if (desc && (desc.get || desc.set)) { Object.defineProperty(newObj, key, desc); } else { newObj[key] = obj[key]; } } } newObj.default = obj; if (cache) { cache.set(obj, newObj); } return newObj; }
|
|
83
|
-
|
|
84
45
|
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
|
|
85
|
-
|
|
86
46
|
// ParseServer - open-source compatible API Server for Parse apps
|
|
87
|
-
var batch = require('./batch'),
|
|
88
|
-
bodyParser = require('body-parser'),
|
|
89
|
-
express = require('express'),
|
|
90
|
-
middlewares = require('./middlewares'),
|
|
91
|
-
Parse = require('parse/node').Parse,
|
|
92
|
-
{
|
|
93
|
-
parse
|
|
94
|
-
} = require('graphql'),
|
|
95
|
-
path = require('path'),
|
|
96
|
-
fs = require('fs');
|
|
97
47
|
|
|
48
|
+
var batch = require('./batch'),
|
|
49
|
+
bodyParser = require('body-parser'),
|
|
50
|
+
express = require('express'),
|
|
51
|
+
middlewares = require('./middlewares'),
|
|
52
|
+
Parse = require('parse/node').Parse,
|
|
53
|
+
{
|
|
54
|
+
parse
|
|
55
|
+
} = require('graphql'),
|
|
56
|
+
path = require('path'),
|
|
57
|
+
fs = require('fs');
|
|
98
58
|
// Mutate the Parse object to add the Cloud Code handlers
|
|
99
|
-
addParseCloud();
|
|
100
|
-
// https://parseplatform.org/parse-server/api/master/ParseServerOptions.html
|
|
59
|
+
addParseCloud();
|
|
101
60
|
|
|
61
|
+
// ParseServer works like a constructor of an express app.
|
|
62
|
+
// https://parseplatform.org/parse-server/api/master/ParseServerOptions.html
|
|
102
63
|
class ParseServer {
|
|
103
64
|
/**
|
|
104
65
|
* @constructor
|
|
@@ -106,131 +67,154 @@ class ParseServer {
|
|
|
106
67
|
*/
|
|
107
68
|
constructor(options) {
|
|
108
69
|
// Scan for deprecated Parse Server options
|
|
109
|
-
_Deprecator.default.scanParseServerOptions(options);
|
|
110
|
-
|
|
111
|
-
|
|
70
|
+
_Deprecator.default.scanParseServerOptions(options);
|
|
71
|
+
// Set option defaults
|
|
112
72
|
injectDefaults(options);
|
|
113
73
|
const {
|
|
114
74
|
appId = (0, _requiredParameter.default)('You must provide an appId!'),
|
|
115
75
|
masterKey = (0, _requiredParameter.default)('You must provide a masterKey!'),
|
|
116
|
-
cloud,
|
|
117
|
-
security,
|
|
118
76
|
javascriptKey,
|
|
119
|
-
serverURL = (0, _requiredParameter.default)('You must provide a serverURL!')
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
} = options; // Initialize the node client SDK automatically
|
|
123
|
-
|
|
77
|
+
serverURL = (0, _requiredParameter.default)('You must provide a serverURL!')
|
|
78
|
+
} = options;
|
|
79
|
+
// Initialize the node client SDK automatically
|
|
124
80
|
Parse.initialize(appId, javascriptKey || 'unused', masterKey);
|
|
125
81
|
Parse.serverURL = serverURL;
|
|
126
82
|
const allControllers = controllers.getControllers(options);
|
|
127
|
-
|
|
128
|
-
loggerController,
|
|
129
|
-
databaseController,
|
|
130
|
-
hooksController
|
|
131
|
-
} = allControllers;
|
|
83
|
+
options.state = 'initialized';
|
|
132
84
|
this.config = _Config.default.put(Object.assign({}, options, allControllers));
|
|
133
|
-
logging.setLogger(loggerController);
|
|
85
|
+
logging.setLogger(allControllers.loggerController);
|
|
86
|
+
}
|
|
134
87
|
|
|
135
|
-
|
|
88
|
+
/**
|
|
89
|
+
* Starts Parse Server as an express app; this promise resolves when Parse Server is ready to accept requests.
|
|
90
|
+
*/
|
|
91
|
+
|
|
92
|
+
async start() {
|
|
93
|
+
try {
|
|
94
|
+
if (this.config.state === 'ok') {
|
|
95
|
+
return this;
|
|
96
|
+
}
|
|
97
|
+
this.config.state = 'starting';
|
|
98
|
+
_Config.default.put(this.config);
|
|
99
|
+
const {
|
|
100
|
+
databaseController,
|
|
101
|
+
hooksController,
|
|
102
|
+
cloud,
|
|
103
|
+
security,
|
|
104
|
+
schema,
|
|
105
|
+
cacheAdapter,
|
|
106
|
+
liveQueryController
|
|
107
|
+
} = this.config;
|
|
108
|
+
try {
|
|
109
|
+
await databaseController.performInitialization();
|
|
110
|
+
} catch (e) {
|
|
111
|
+
if (e.code !== Parse.Error.DUPLICATE_VALUE) {
|
|
112
|
+
throw e;
|
|
113
|
+
}
|
|
114
|
+
}
|
|
115
|
+
await hooksController.load();
|
|
116
|
+
const startupPromises = [];
|
|
136
117
|
if (schema) {
|
|
137
|
-
|
|
118
|
+
startupPromises.push(new _DefinedSchemas.DefinedSchemas(schema, this.config).execute());
|
|
138
119
|
}
|
|
139
|
-
|
|
140
|
-
|
|
141
|
-
serverStartComplete();
|
|
120
|
+
if (cacheAdapter !== null && cacheAdapter !== void 0 && cacheAdapter.connect && typeof cacheAdapter.connect === 'function') {
|
|
121
|
+
startupPromises.push(cacheAdapter.connect());
|
|
142
122
|
}
|
|
143
|
-
|
|
144
|
-
|
|
145
|
-
|
|
146
|
-
|
|
147
|
-
|
|
148
|
-
|
|
123
|
+
startupPromises.push(liveQueryController.connect());
|
|
124
|
+
await Promise.all(startupPromises);
|
|
125
|
+
if (cloud) {
|
|
126
|
+
addParseCloud();
|
|
127
|
+
if (typeof cloud === 'function') {
|
|
128
|
+
await Promise.resolve(cloud(Parse));
|
|
129
|
+
} else if (typeof cloud === 'string') {
|
|
130
|
+
var _json;
|
|
131
|
+
let json;
|
|
132
|
+
if (process.env.npm_package_json) {
|
|
133
|
+
json = require(process.env.npm_package_json);
|
|
134
|
+
}
|
|
135
|
+
if (process.env.npm_package_type === 'module' || ((_json = json) === null || _json === void 0 ? void 0 : _json.type) === 'module') {
|
|
136
|
+
await import(path.resolve(process.cwd(), cloud)).default;
|
|
137
|
+
} else {
|
|
138
|
+
require(path.resolve(process.cwd(), cloud));
|
|
139
|
+
}
|
|
140
|
+
} else {
|
|
141
|
+
throw "argument 'cloud' must either be a string or a function";
|
|
142
|
+
}
|
|
143
|
+
await new Promise(resolve => setTimeout(resolve, 10));
|
|
149
144
|
}
|
|
150
|
-
|
|
151
|
-
|
|
152
|
-
if (cloud) {
|
|
153
|
-
addParseCloud();
|
|
154
|
-
|
|
155
|
-
if (typeof cloud === 'function') {
|
|
156
|
-
cloud(Parse);
|
|
157
|
-
} else if (typeof cloud === 'string') {
|
|
158
|
-
require(path.resolve(process.cwd(), cloud));
|
|
159
|
-
} else {
|
|
160
|
-
throw "argument 'cloud' must either be a string or a function";
|
|
145
|
+
if (security && security.enableCheck && security.enableCheckLog) {
|
|
146
|
+
new _CheckRunner.default(security).run();
|
|
161
147
|
}
|
|
162
|
-
|
|
163
|
-
|
|
164
|
-
|
|
165
|
-
|
|
148
|
+
this.config.state = 'ok';
|
|
149
|
+
_Config.default.put(this.config);
|
|
150
|
+
return this;
|
|
151
|
+
} catch (error) {
|
|
152
|
+
console.error(error);
|
|
153
|
+
this.config.state = 'error';
|
|
154
|
+
throw error;
|
|
166
155
|
}
|
|
167
156
|
}
|
|
168
|
-
|
|
169
157
|
get app() {
|
|
170
158
|
if (!this._app) {
|
|
171
159
|
this._app = ParseServer.app(this.config);
|
|
172
160
|
}
|
|
173
|
-
|
|
174
161
|
return this._app;
|
|
175
162
|
}
|
|
176
|
-
|
|
177
163
|
handleShutdown() {
|
|
178
164
|
const promises = [];
|
|
179
165
|
const {
|
|
180
166
|
adapter: databaseAdapter
|
|
181
167
|
} = this.config.databaseController;
|
|
182
|
-
|
|
183
168
|
if (databaseAdapter && typeof databaseAdapter.handleShutdown === 'function') {
|
|
184
169
|
promises.push(databaseAdapter.handleShutdown());
|
|
185
170
|
}
|
|
186
|
-
|
|
187
171
|
const {
|
|
188
172
|
adapter: fileAdapter
|
|
189
173
|
} = this.config.filesController;
|
|
190
|
-
|
|
191
174
|
if (fileAdapter && typeof fileAdapter.handleShutdown === 'function') {
|
|
192
175
|
promises.push(fileAdapter.handleShutdown());
|
|
193
176
|
}
|
|
194
|
-
|
|
195
177
|
const {
|
|
196
178
|
adapter: cacheAdapter
|
|
197
179
|
} = this.config.cacheController;
|
|
198
|
-
|
|
199
180
|
if (cacheAdapter && typeof cacheAdapter.handleShutdown === 'function') {
|
|
200
181
|
promises.push(cacheAdapter.handleShutdown());
|
|
201
182
|
}
|
|
202
|
-
|
|
203
183
|
return (promises.length > 0 ? Promise.all(promises) : Promise.resolve()).then(() => {
|
|
204
184
|
if (this.config.serverCloseComplete) {
|
|
205
185
|
this.config.serverCloseComplete();
|
|
206
186
|
}
|
|
207
187
|
});
|
|
208
188
|
}
|
|
189
|
+
|
|
209
190
|
/**
|
|
210
191
|
* @static
|
|
211
192
|
* Create an express app for the parse server
|
|
212
193
|
* @param {Object} options let you specify the maxUploadSize when creating the express app */
|
|
213
|
-
|
|
214
|
-
|
|
215
194
|
static app(options) {
|
|
216
195
|
const {
|
|
217
196
|
maxUploadSize = '20mb',
|
|
218
197
|
appId,
|
|
219
198
|
directAccess,
|
|
220
|
-
pages
|
|
221
|
-
|
|
199
|
+
pages,
|
|
200
|
+
rateLimit = []
|
|
201
|
+
} = options;
|
|
202
|
+
// This app serves the Parse API directly.
|
|
222
203
|
// It's the equivalent of https://api.parse.com/1 in the hosted Parse API.
|
|
223
|
-
|
|
224
|
-
|
|
225
|
-
|
|
226
|
-
|
|
227
|
-
|
|
204
|
+
var api = express();
|
|
205
|
+
//api.use("/apps", express.static(__dirname + "/public"));
|
|
206
|
+
api.use(middlewares.allowCrossDomain(appId));
|
|
207
|
+
// File handling needs to be before default middlewares are applied
|
|
228
208
|
api.use('/', new _FilesRouter.FilesRouter().expressRouter({
|
|
229
209
|
maxUploadSize: maxUploadSize
|
|
230
210
|
}));
|
|
231
211
|
api.use('/health', function (req, res) {
|
|
212
|
+
res.status(options.state === 'ok' ? 200 : 503);
|
|
213
|
+
if (options.state === 'starting') {
|
|
214
|
+
res.set('Retry-After', 1);
|
|
215
|
+
}
|
|
232
216
|
res.json({
|
|
233
|
-
status:
|
|
217
|
+
status: options.state
|
|
234
218
|
});
|
|
235
219
|
});
|
|
236
220
|
api.use('/', bodyParser.urlencoded({
|
|
@@ -242,15 +226,20 @@ class ParseServer {
|
|
|
242
226
|
}));
|
|
243
227
|
api.use(middlewares.allowMethodOverride);
|
|
244
228
|
api.use(middlewares.handleParseHeaders);
|
|
229
|
+
const routes = Array.isArray(rateLimit) ? rateLimit : [rateLimit];
|
|
230
|
+
for (const route of routes) {
|
|
231
|
+
middlewares.addRateLimit(route, options);
|
|
232
|
+
}
|
|
233
|
+
api.use(middlewares.handleParseSession);
|
|
245
234
|
const appRouter = ParseServer.promiseRouter({
|
|
246
235
|
appId
|
|
247
236
|
});
|
|
248
237
|
api.use(appRouter.expressRouter());
|
|
249
|
-
api.use(middlewares.handleParseErrors);
|
|
238
|
+
api.use(middlewares.handleParseErrors);
|
|
250
239
|
|
|
240
|
+
// run the following when not testing
|
|
251
241
|
if (!process.env.TESTING) {
|
|
252
242
|
//This causes tests to spew some useless warnings, so disable in test
|
|
253
|
-
|
|
254
243
|
/* istanbul ignore next */
|
|
255
244
|
process.on('uncaughtException', err => {
|
|
256
245
|
if (err.code === 'EADDRINUSE') {
|
|
@@ -260,22 +249,18 @@ class ParseServer {
|
|
|
260
249
|
} else {
|
|
261
250
|
throw err;
|
|
262
251
|
}
|
|
263
|
-
});
|
|
264
|
-
|
|
252
|
+
});
|
|
253
|
+
// verify the server url after a 'mount' event is received
|
|
265
254
|
/* istanbul ignore next */
|
|
266
|
-
|
|
267
255
|
api.on('mount', function () {
|
|
268
256
|
ParseServer.verifyServerUrl();
|
|
269
257
|
});
|
|
270
258
|
}
|
|
271
|
-
|
|
272
259
|
if (process.env.PARSE_SERVER_ENABLE_EXPERIMENTAL_DIRECT_ACCESS === '1' || directAccess) {
|
|
273
260
|
Parse.CoreManager.setRESTController((0, _ParseServerRESTController.ParseServerRESTController)(appId, appRouter));
|
|
274
261
|
}
|
|
275
|
-
|
|
276
262
|
return api;
|
|
277
263
|
}
|
|
278
|
-
|
|
279
264
|
static promiseRouter({
|
|
280
265
|
appId
|
|
281
266
|
}) {
|
|
@@ -287,20 +272,23 @@ class ParseServer {
|
|
|
287
272
|
batch.mountOnto(appRouter);
|
|
288
273
|
return appRouter;
|
|
289
274
|
}
|
|
275
|
+
|
|
290
276
|
/**
|
|
291
277
|
* starts the parse server's express app
|
|
292
278
|
* @param {ParseServerOptions} options to use to start the server
|
|
293
|
-
* @param {Function} callback called when the server has started
|
|
294
279
|
* @returns {ParseServer} the parse server instance
|
|
295
280
|
*/
|
|
296
281
|
|
|
297
|
-
|
|
298
|
-
|
|
282
|
+
async startApp(options) {
|
|
283
|
+
try {
|
|
284
|
+
await this.start();
|
|
285
|
+
} catch (e) {
|
|
286
|
+
console.error('Error on ParseServer.startApp: ', e);
|
|
287
|
+
throw e;
|
|
288
|
+
}
|
|
299
289
|
const app = express();
|
|
300
|
-
|
|
301
290
|
if (options.middleware) {
|
|
302
291
|
let middleware;
|
|
303
|
-
|
|
304
292
|
if (typeof options.middleware == 'string') {
|
|
305
293
|
middleware = require(path.resolve(process.cwd(), options.middleware));
|
|
306
294
|
} else {
|
|
@@ -309,96 +297,86 @@ class ParseServer {
|
|
|
309
297
|
|
|
310
298
|
app.use(middleware);
|
|
311
299
|
}
|
|
312
|
-
|
|
313
300
|
app.use(options.mountPath, this.app);
|
|
314
|
-
|
|
315
301
|
if (options.mountGraphQL === true || options.mountPlayground === true) {
|
|
316
302
|
let graphQLCustomTypeDefs = undefined;
|
|
317
|
-
|
|
318
303
|
if (typeof options.graphQLSchema === 'string') {
|
|
319
304
|
graphQLCustomTypeDefs = parse(fs.readFileSync(options.graphQLSchema, 'utf8'));
|
|
320
305
|
} else if (typeof options.graphQLSchema === 'object' || typeof options.graphQLSchema === 'function') {
|
|
321
306
|
graphQLCustomTypeDefs = options.graphQLSchema;
|
|
322
307
|
}
|
|
323
|
-
|
|
324
308
|
const parseGraphQLServer = new _ParseGraphQLServer.ParseGraphQLServer(this, {
|
|
325
309
|
graphQLPath: options.graphQLPath,
|
|
326
310
|
playgroundPath: options.playgroundPath,
|
|
327
311
|
graphQLCustomTypeDefs
|
|
328
312
|
});
|
|
329
|
-
|
|
330
313
|
if (options.mountGraphQL) {
|
|
331
314
|
parseGraphQLServer.applyGraphQL(app);
|
|
332
315
|
}
|
|
333
|
-
|
|
334
316
|
if (options.mountPlayground) {
|
|
335
317
|
parseGraphQLServer.applyPlayground(app);
|
|
336
318
|
}
|
|
337
319
|
}
|
|
338
|
-
|
|
339
|
-
|
|
320
|
+
const server = await new Promise(resolve => {
|
|
321
|
+
app.listen(options.port, options.host, function () {
|
|
322
|
+
resolve(this);
|
|
323
|
+
});
|
|
324
|
+
});
|
|
340
325
|
this.server = server;
|
|
341
|
-
|
|
342
326
|
if (options.startLiveQueryServer || options.liveQueryServerOptions) {
|
|
343
|
-
this.liveQueryServer = ParseServer.createLiveQueryServer(server, options.liveQueryServerOptions, options);
|
|
327
|
+
this.liveQueryServer = await ParseServer.createLiveQueryServer(server, options.liveQueryServerOptions, options);
|
|
328
|
+
}
|
|
329
|
+
if (options.trustProxy) {
|
|
330
|
+
app.set('trust proxy', options.trustProxy);
|
|
344
331
|
}
|
|
345
332
|
/* istanbul ignore next */
|
|
346
|
-
|
|
347
|
-
|
|
348
333
|
if (!process.env.TESTING) {
|
|
349
334
|
configureListeners(this);
|
|
350
335
|
}
|
|
351
|
-
|
|
352
336
|
this.expressApp = app;
|
|
353
337
|
return this;
|
|
354
338
|
}
|
|
339
|
+
|
|
355
340
|
/**
|
|
356
341
|
* Creates a new ParseServer and starts it.
|
|
357
342
|
* @param {ParseServerOptions} options used to start the server
|
|
358
|
-
* @param {Function} callback called when the server has started
|
|
359
343
|
* @returns {ParseServer} the parse server instance
|
|
360
344
|
*/
|
|
361
|
-
|
|
362
|
-
|
|
363
|
-
static start(options, callback) {
|
|
345
|
+
static async startApp(options) {
|
|
364
346
|
const parseServer = new ParseServer(options);
|
|
365
|
-
return parseServer.
|
|
347
|
+
return parseServer.startApp(options);
|
|
366
348
|
}
|
|
349
|
+
|
|
367
350
|
/**
|
|
368
351
|
* Helper method to create a liveQuery server
|
|
369
352
|
* @static
|
|
370
353
|
* @param {Server} httpServer an optional http server to pass
|
|
371
354
|
* @param {LiveQueryServerOptions} config options for the liveQueryServer
|
|
372
355
|
* @param {ParseServerOptions} options options for the ParseServer
|
|
373
|
-
* @returns {ParseLiveQueryServer} the live query server instance
|
|
356
|
+
* @returns {Promise<ParseLiveQueryServer>} the live query server instance
|
|
374
357
|
*/
|
|
375
|
-
|
|
376
|
-
|
|
377
|
-
static createLiveQueryServer(httpServer, config, options) {
|
|
358
|
+
static async createLiveQueryServer(httpServer, config, options) {
|
|
378
359
|
if (!httpServer || config && config.port) {
|
|
379
360
|
var app = express();
|
|
380
361
|
httpServer = require('http').createServer(app);
|
|
381
362
|
httpServer.listen(config.port);
|
|
382
363
|
}
|
|
383
|
-
|
|
384
|
-
|
|
364
|
+
const server = new _ParseLiveQueryServer.ParseLiveQueryServer(httpServer, config, options);
|
|
365
|
+
await server.connect();
|
|
366
|
+
return server;
|
|
385
367
|
}
|
|
386
|
-
|
|
387
368
|
static verifyServerUrl(callback) {
|
|
388
369
|
// perform a health check on the serverURL value
|
|
389
370
|
if (Parse.serverURL) {
|
|
390
371
|
const request = require('./request');
|
|
391
|
-
|
|
392
372
|
request({
|
|
393
373
|
url: Parse.serverURL.replace(/\/$/, '') + '/health'
|
|
394
374
|
}).catch(response => response).then(response => {
|
|
395
375
|
const json = response.data || null;
|
|
396
|
-
|
|
397
376
|
if (response.status !== 200 || !json || json && json.status !== 'ok') {
|
|
398
377
|
/* eslint-disable no-console */
|
|
399
378
|
console.warn(`\nWARNING, Unable to connect to '${Parse.serverURL}'.` + ` Cloud code and push notifications may be unavailable!\n`);
|
|
400
379
|
/* eslint-enable no-console */
|
|
401
|
-
|
|
402
380
|
if (callback) {
|
|
403
381
|
callback(false);
|
|
404
382
|
}
|
|
@@ -410,73 +388,63 @@ class ParseServer {
|
|
|
410
388
|
});
|
|
411
389
|
}
|
|
412
390
|
}
|
|
413
|
-
|
|
414
391
|
}
|
|
415
|
-
|
|
416
392
|
function addParseCloud() {
|
|
417
393
|
const ParseCloud = require('./cloud-code/Parse.Cloud');
|
|
418
|
-
|
|
419
394
|
Object.defineProperty(Parse, 'Server', {
|
|
420
395
|
get() {
|
|
421
396
|
return _Config.default.get(Parse.applicationId);
|
|
422
397
|
},
|
|
423
|
-
|
|
424
398
|
set(newVal) {
|
|
425
399
|
newVal.appId = Parse.applicationId;
|
|
426
|
-
|
|
427
400
|
_Config.default.put(newVal);
|
|
428
401
|
},
|
|
429
|
-
|
|
430
402
|
configurable: true
|
|
431
403
|
});
|
|
432
404
|
Object.assign(Parse.Cloud, ParseCloud);
|
|
433
405
|
global.Parse = Parse;
|
|
434
406
|
}
|
|
435
|
-
|
|
436
407
|
function injectDefaults(options) {
|
|
437
408
|
Object.keys(_defaults.default).forEach(key => {
|
|
438
409
|
if (!Object.prototype.hasOwnProperty.call(options, key)) {
|
|
439
410
|
options[key] = _defaults.default[key];
|
|
440
411
|
}
|
|
441
412
|
});
|
|
442
|
-
|
|
443
413
|
if (!Object.prototype.hasOwnProperty.call(options, 'serverURL')) {
|
|
444
414
|
options.serverURL = `http://localhost:${options.port}${options.mountPath}`;
|
|
445
|
-
}
|
|
446
|
-
|
|
415
|
+
}
|
|
447
416
|
|
|
417
|
+
// Reserved Characters
|
|
448
418
|
if (options.appId) {
|
|
449
419
|
const regex = /[!#$%'()*+&/:;=?@[\]{}^,|<>]/g;
|
|
450
|
-
|
|
451
420
|
if (options.appId.match(regex)) {
|
|
452
421
|
console.warn(`\nWARNING, appId that contains special characters can cause issues while using with urls.\n`);
|
|
453
422
|
}
|
|
454
|
-
}
|
|
455
|
-
|
|
423
|
+
}
|
|
456
424
|
|
|
425
|
+
// Backwards compatibility
|
|
457
426
|
if (options.userSensitiveFields) {
|
|
458
427
|
/* eslint-disable no-console */
|
|
459
428
|
!process.env.TESTING && console.warn(`\nDEPRECATED: userSensitiveFields has been replaced by protectedFields allowing the ability to protect fields in all classes with CLP. \n`);
|
|
460
429
|
/* eslint-enable no-console */
|
|
461
430
|
|
|
462
|
-
const userSensitiveFields = Array.from(new Set([...(_defaults.default.userSensitiveFields || []), ...(options.userSensitiveFields || [])]));
|
|
431
|
+
const userSensitiveFields = Array.from(new Set([...(_defaults.default.userSensitiveFields || []), ...(options.userSensitiveFields || [])]));
|
|
432
|
+
|
|
433
|
+
// If the options.protectedFields is unset,
|
|
463
434
|
// it'll be assigned the default above.
|
|
464
435
|
// Here, protect against the case where protectedFields
|
|
465
436
|
// is set, but doesn't have _User.
|
|
466
|
-
|
|
467
437
|
if (!('_User' in options.protectedFields)) {
|
|
468
438
|
options.protectedFields = Object.assign({
|
|
469
439
|
_User: []
|
|
470
440
|
}, options.protectedFields);
|
|
471
441
|
}
|
|
472
|
-
|
|
473
442
|
options.protectedFields['_User']['*'] = Array.from(new Set([...(options.protectedFields['_User']['*'] || []), ...userSensitiveFields]));
|
|
474
|
-
}
|
|
475
|
-
|
|
443
|
+
}
|
|
476
444
|
|
|
445
|
+
// Merge protectedFields options with defaults.
|
|
477
446
|
Object.keys(_defaults.default.protectedFields).forEach(c => {
|
|
478
447
|
const cur = options.protectedFields[c];
|
|
479
|
-
|
|
480
448
|
if (!cur) {
|
|
481
449
|
options.protectedFields[c] = _defaults.default.protectedFields[c];
|
|
482
450
|
} else {
|
|
@@ -487,17 +455,15 @@ function injectDefaults(options) {
|
|
|
487
455
|
}
|
|
488
456
|
});
|
|
489
457
|
options.masterKeyIps = Array.from(new Set(options.masterKeyIps.concat(_defaults.default.masterKeyIps, options.masterKeyIps)));
|
|
490
|
-
}
|
|
458
|
+
}
|
|
491
459
|
|
|
460
|
+
// Those can't be tested as it requires a subprocess
|
|
492
461
|
/* istanbul ignore next */
|
|
493
|
-
|
|
494
|
-
|
|
495
462
|
function configureListeners(parseServer) {
|
|
496
463
|
const server = parseServer.server;
|
|
497
464
|
const sockets = {};
|
|
498
465
|
/* Currently, express doesn't shut down immediately after receiving SIGINT/SIGTERM if it has client connections that haven't timed out. (This is a known issue with node - https://github.com/nodejs/node/issues/2642)
|
|
499
466
|
This function, along with `destroyAliveConnections()`, intend to fix this behavior such that parse server will close all open connections and initiate the shutdown process as soon as it receives a SIGINT/SIGTERM signal. */
|
|
500
|
-
|
|
501
467
|
server.on('connection', socket => {
|
|
502
468
|
const socketId = socket.remoteAddress + ':' + socket.remotePort;
|
|
503
469
|
sockets[socketId] = socket;
|
|
@@ -505,7 +471,6 @@ function configureListeners(parseServer) {
|
|
|
505
471
|
delete sockets[socketId];
|
|
506
472
|
});
|
|
507
473
|
});
|
|
508
|
-
|
|
509
474
|
const destroyAliveConnections = function () {
|
|
510
475
|
for (const socketId in sockets) {
|
|
511
476
|
try {
|
|
@@ -515,18 +480,15 @@ function configureListeners(parseServer) {
|
|
|
515
480
|
}
|
|
516
481
|
}
|
|
517
482
|
};
|
|
518
|
-
|
|
519
483
|
const handleShutdown = function () {
|
|
520
484
|
process.stdout.write('Termination signal received. Shutting down.');
|
|
521
485
|
destroyAliveConnections();
|
|
522
486
|
server.close();
|
|
523
487
|
parseServer.handleShutdown();
|
|
524
488
|
};
|
|
525
|
-
|
|
526
489
|
process.on('SIGTERM', handleShutdown);
|
|
527
490
|
process.on('SIGINT', handleShutdown);
|
|
528
491
|
}
|
|
529
|
-
|
|
530
492
|
var _default = ParseServer;
|
|
531
493
|
exports.default = _default;
|
|
532
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../src/ParseServer.js"],"names":["batch","require","bodyParser","express","middlewares","Parse","parse","path","fs","addParseCloud","ParseServer","constructor","options","Deprecator","scanParseServerOptions","injectDefaults","appId","masterKey","cloud","security","javascriptKey","serverURL","serverStartComplete","schema","initialize","allControllers","controllers","getControllers","loggerController","databaseController","hooksController","config","Config","put","Object","assign","logging","setLogger","performInitialization","then","load","DefinedSchemas","execute","catch","error","console","process","exit","resolve","cwd","enableCheck","enableCheckLog","CheckRunner","run","app","_app","handleShutdown","promises","adapter","databaseAdapter","push","fileAdapter","filesController","cacheAdapter","cacheController","length","Promise","all","serverCloseComplete","maxUploadSize","directAccess","pages","api","use","allowCrossDomain","FilesRouter","expressRouter","req","res","json","status","urlencoded","extended","enableRouter","PagesRouter","PublicAPIRouter","type","limit","allowMethodOverride","handleParseHeaders","appRouter","promiseRouter","handleParseErrors","env","TESTING","on","err","code","stderr","write","port","verifyServerUrl","PARSE_SERVER_ENABLE_EXPERIMENTAL_DIRECT_ACCESS","CoreManager","setRESTController","routers","ClassesRouter","UsersRouter","SessionsRouter","RolesRouter","AnalyticsRouter","InstallationsRouter","FunctionsRouter","SchemasRouter","PushRouter","LogsRouter","IAPValidationRouter","FeaturesRouter","GlobalConfigRouter","GraphQLRouter","PurgeRouter","HooksRouter","CloudCodeRouter","AudiencesRouter","AggregateRouter","SecurityRouter","routes","reduce","memo","router","concat","PromiseRouter","mountOnto","start","callback","middleware","mountPath","mountGraphQL","mountPlayground","graphQLCustomTypeDefs","undefined","graphQLSchema","readFileSync","parseGraphQLServer","ParseGraphQLServer","graphQLPath","playgroundPath","applyGraphQL","applyPlayground","server","listen","host","startLiveQueryServer","liveQueryServerOptions","liveQueryServer","createLiveQueryServer","configureListeners","expressApp","parseServer","httpServer","createServer","ParseLiveQueryServer","request","url","replace","response","data","warn","ParseCloud","defineProperty","get","applicationId","set","newVal","configurable","Cloud","global","keys","defaults","forEach","key","prototype","hasOwnProperty","call","regex","match","userSensitiveFields","Array","from","Set","protectedFields","_User","c","cur","r","unq","masterKeyIps","sockets","socket","socketId","remoteAddress","remotePort","destroyAliveConnections","destroy","e","stdout","close"],"mappings":"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","sourcesContent":["// ParseServer - open-source compatible API Server for Parse apps\n\nvar batch = require('./batch'),\n  bodyParser = require('body-parser'),\n  express = require('express'),\n  middlewares = require('./middlewares'),\n  Parse = require('parse/node').Parse,\n  { parse } = require('graphql'),\n  path = require('path'),\n  fs = require('fs');\n\nimport { ParseServerOptions, LiveQueryServerOptions } from './Options';\nimport defaults from './defaults';\nimport * as logging from './logger';\nimport Config from './Config';\nimport PromiseRouter from './PromiseRouter';\nimport requiredParameter from './requiredParameter';\nimport { AnalyticsRouter } from './Routers/AnalyticsRouter';\nimport { ClassesRouter } from './Routers/ClassesRouter';\nimport { FeaturesRouter } from './Routers/FeaturesRouter';\nimport { FilesRouter } from './Routers/FilesRouter';\nimport { FunctionsRouter } from './Routers/FunctionsRouter';\nimport { GlobalConfigRouter } from './Routers/GlobalConfigRouter';\nimport { GraphQLRouter } from './Routers/GraphQLRouter';\nimport { HooksRouter } from './Routers/HooksRouter';\nimport { IAPValidationRouter } from './Routers/IAPValidationRouter';\nimport { InstallationsRouter } from './Routers/InstallationsRouter';\nimport { LogsRouter } from './Routers/LogsRouter';\nimport { ParseLiveQueryServer } from './LiveQuery/ParseLiveQueryServer';\nimport { PagesRouter } from './Routers/PagesRouter';\nimport { PublicAPIRouter } from './Routers/PublicAPIRouter';\nimport { PushRouter } from './Routers/PushRouter';\nimport { CloudCodeRouter } from './Routers/CloudCodeRouter';\nimport { RolesRouter } from './Routers/RolesRouter';\nimport { SchemasRouter } from './Routers/SchemasRouter';\nimport { SessionsRouter } from './Routers/SessionsRouter';\nimport { UsersRouter } from './Routers/UsersRouter';\nimport { PurgeRouter } from './Routers/PurgeRouter';\nimport { AudiencesRouter } from './Routers/AudiencesRouter';\nimport { AggregateRouter } from './Routers/AggregateRouter';\nimport { ParseServerRESTController } from './ParseServerRESTController';\nimport * as controllers from './Controllers';\nimport { ParseGraphQLServer } from './GraphQL/ParseGraphQLServer';\nimport { SecurityRouter } from './Routers/SecurityRouter';\nimport CheckRunner from './Security/CheckRunner';\nimport Deprecator from './Deprecator/Deprecator';\nimport { DefinedSchemas } from './SchemaMigrations/DefinedSchemas';\n\n// Mutate the Parse object to add the Cloud Code handlers\naddParseCloud();\n\n// ParseServer works like a constructor of an express app.\n// https://parseplatform.org/parse-server/api/master/ParseServerOptions.html\nclass ParseServer {\n  /**\n   * @constructor\n   * @param {ParseServerOptions} options the parse server initialization options\n   */\n  constructor(options: ParseServerOptions) {\n    // Scan for deprecated Parse Server options\n    Deprecator.scanParseServerOptions(options);\n    // Set option defaults\n    injectDefaults(options);\n    const {\n      appId = requiredParameter('You must provide an appId!'),\n      masterKey = requiredParameter('You must provide a masterKey!'),\n      cloud,\n      security,\n      javascriptKey,\n      serverURL = requiredParameter('You must provide a serverURL!'),\n      serverStartComplete,\n      schema,\n    } = options;\n    // Initialize the node client SDK automatically\n    Parse.initialize(appId, javascriptKey || 'unused', masterKey);\n    Parse.serverURL = serverURL;\n\n    const allControllers = controllers.getControllers(options);\n\n    const { loggerController, databaseController, hooksController } = allControllers;\n    this.config = Config.put(Object.assign({}, options, allControllers));\n\n    logging.setLogger(loggerController);\n\n    // Note: Tests will start to fail if any validation happens after this is called.\n    databaseController\n      .performInitialization()\n      .then(() => hooksController.load())\n      .then(async () => {\n        if (schema) {\n          await new DefinedSchemas(schema, this.config).execute();\n        }\n        if (serverStartComplete) {\n          serverStartComplete();\n        }\n      })\n      .catch(error => {\n        if (serverStartComplete) {\n          serverStartComplete(error);\n        } else {\n          console.error(error);\n          process.exit(1);\n        }\n      });\n\n    if (cloud) {\n      addParseCloud();\n      if (typeof cloud === 'function') {\n        cloud(Parse);\n      } else if (typeof cloud === 'string') {\n        require(path.resolve(process.cwd(), cloud));\n      } else {\n        throw \"argument 'cloud' must either be a string or a function\";\n      }\n    }\n\n    if (security && security.enableCheck && security.enableCheckLog) {\n      new CheckRunner(options.security).run();\n    }\n  }\n\n  get app() {\n    if (!this._app) {\n      this._app = ParseServer.app(this.config);\n    }\n    return this._app;\n  }\n\n  handleShutdown() {\n    const promises = [];\n    const { adapter: databaseAdapter } = this.config.databaseController;\n    if (databaseAdapter && typeof databaseAdapter.handleShutdown === 'function') {\n      promises.push(databaseAdapter.handleShutdown());\n    }\n    const { adapter: fileAdapter } = this.config.filesController;\n    if (fileAdapter && typeof fileAdapter.handleShutdown === 'function') {\n      promises.push(fileAdapter.handleShutdown());\n    }\n    const { adapter: cacheAdapter } = this.config.cacheController;\n    if (cacheAdapter && typeof cacheAdapter.handleShutdown === 'function') {\n      promises.push(cacheAdapter.handleShutdown());\n    }\n    return (promises.length > 0 ? Promise.all(promises) : Promise.resolve()).then(() => {\n      if (this.config.serverCloseComplete) {\n        this.config.serverCloseComplete();\n      }\n    });\n  }\n\n  /**\n   * @static\n   * Create an express app for the parse server\n   * @param {Object} options let you specify the maxUploadSize when creating the express app  */\n  static app(options) {\n    const { maxUploadSize = '20mb', appId, directAccess, pages } = options;\n    // This app serves the Parse API directly.\n    // It's the equivalent of https://api.parse.com/1 in the hosted Parse API.\n    var api = express();\n    //api.use(\"/apps\", express.static(__dirname + \"/public\"));\n    api.use(middlewares.allowCrossDomain(appId));\n    // File handling needs to be before default middlewares are applied\n    api.use(\n      '/',\n      new FilesRouter().expressRouter({\n        maxUploadSize: maxUploadSize,\n      })\n    );\n\n    api.use('/health', function (req, res) {\n      res.json({\n        status: 'ok',\n      });\n    });\n\n    api.use(\n      '/',\n      bodyParser.urlencoded({ extended: false }),\n      pages.enableRouter\n        ? new PagesRouter(pages).expressRouter()\n        : new PublicAPIRouter().expressRouter()\n    );\n\n    api.use(bodyParser.json({ type: '*/*', limit: maxUploadSize }));\n    api.use(middlewares.allowMethodOverride);\n    api.use(middlewares.handleParseHeaders);\n\n    const appRouter = ParseServer.promiseRouter({ appId });\n    api.use(appRouter.expressRouter());\n\n    api.use(middlewares.handleParseErrors);\n\n    // run the following when not testing\n    if (!process.env.TESTING) {\n      //This causes tests to spew some useless warnings, so disable in test\n      /* istanbul ignore next */\n      process.on('uncaughtException', err => {\n        if (err.code === 'EADDRINUSE') {\n          // user-friendly message for this common error\n          process.stderr.write(`Unable to listen on port ${err.port}. The port is already in use.`);\n          process.exit(0);\n        } else {\n          throw err;\n        }\n      });\n      // verify the server url after a 'mount' event is received\n      /* istanbul ignore next */\n      api.on('mount', function () {\n        ParseServer.verifyServerUrl();\n      });\n    }\n    if (process.env.PARSE_SERVER_ENABLE_EXPERIMENTAL_DIRECT_ACCESS === '1' || directAccess) {\n      Parse.CoreManager.setRESTController(ParseServerRESTController(appId, appRouter));\n    }\n    return api;\n  }\n\n  static promiseRouter({ appId }) {\n    const routers = [\n      new ClassesRouter(),\n      new UsersRouter(),\n      new SessionsRouter(),\n      new RolesRouter(),\n      new AnalyticsRouter(),\n      new InstallationsRouter(),\n      new FunctionsRouter(),\n      new SchemasRouter(),\n      new PushRouter(),\n      new LogsRouter(),\n      new IAPValidationRouter(),\n      new FeaturesRouter(),\n      new GlobalConfigRouter(),\n      new GraphQLRouter(),\n      new PurgeRouter(),\n      new HooksRouter(),\n      new CloudCodeRouter(),\n      new AudiencesRouter(),\n      new AggregateRouter(),\n      new SecurityRouter(),\n    ];\n\n    const routes = routers.reduce((memo, router) => {\n      return memo.concat(router.routes);\n    }, []);\n\n    const appRouter = new PromiseRouter(routes, appId);\n\n    batch.mountOnto(appRouter);\n    return appRouter;\n  }\n\n  /**\n   * starts the parse server's express app\n   * @param {ParseServerOptions} options to use to start the server\n   * @param {Function} callback called when the server has started\n   * @returns {ParseServer} the parse server instance\n   */\n  start(options: ParseServerOptions, callback: ?() => void) {\n    const app = express();\n    if (options.middleware) {\n      let middleware;\n      if (typeof options.middleware == 'string') {\n        middleware = require(path.resolve(process.cwd(), options.middleware));\n      } else {\n        middleware = options.middleware; // use as-is let express fail\n      }\n      app.use(middleware);\n    }\n\n    app.use(options.mountPath, this.app);\n\n    if (options.mountGraphQL === true || options.mountPlayground === true) {\n      let graphQLCustomTypeDefs = undefined;\n      if (typeof options.graphQLSchema === 'string') {\n        graphQLCustomTypeDefs = parse(fs.readFileSync(options.graphQLSchema, 'utf8'));\n      } else if (\n        typeof options.graphQLSchema === 'object' ||\n        typeof options.graphQLSchema === 'function'\n      ) {\n        graphQLCustomTypeDefs = options.graphQLSchema;\n      }\n\n      const parseGraphQLServer = new ParseGraphQLServer(this, {\n        graphQLPath: options.graphQLPath,\n        playgroundPath: options.playgroundPath,\n        graphQLCustomTypeDefs,\n      });\n\n      if (options.mountGraphQL) {\n        parseGraphQLServer.applyGraphQL(app);\n      }\n\n      if (options.mountPlayground) {\n        parseGraphQLServer.applyPlayground(app);\n      }\n    }\n\n    const server = app.listen(options.port, options.host, callback);\n    this.server = server;\n\n    if (options.startLiveQueryServer || options.liveQueryServerOptions) {\n      this.liveQueryServer = ParseServer.createLiveQueryServer(\n        server,\n        options.liveQueryServerOptions,\n        options\n      );\n    }\n    /* istanbul ignore next */\n    if (!process.env.TESTING) {\n      configureListeners(this);\n    }\n    this.expressApp = app;\n    return this;\n  }\n\n  /**\n   * Creates a new ParseServer and starts it.\n   * @param {ParseServerOptions} options used to start the server\n   * @param {Function} callback called when the server has started\n   * @returns {ParseServer} the parse server instance\n   */\n  static start(options: ParseServerOptions, callback: ?() => void) {\n    const parseServer = new ParseServer(options);\n    return parseServer.start(options, callback);\n  }\n\n  /**\n   * Helper method to create a liveQuery server\n   * @static\n   * @param {Server} httpServer an optional http server to pass\n   * @param {LiveQueryServerOptions} config options for the liveQueryServer\n   * @param {ParseServerOptions} options options for the ParseServer\n   * @returns {ParseLiveQueryServer} the live query server instance\n   */\n  static createLiveQueryServer(\n    httpServer,\n    config: LiveQueryServerOptions,\n    options: ParseServerOptions\n  ) {\n    if (!httpServer || (config && config.port)) {\n      var app = express();\n      httpServer = require('http').createServer(app);\n      httpServer.listen(config.port);\n    }\n    return new ParseLiveQueryServer(httpServer, config, options);\n  }\n\n  static verifyServerUrl(callback) {\n    // perform a health check on the serverURL value\n    if (Parse.serverURL) {\n      const request = require('./request');\n      request({ url: Parse.serverURL.replace(/\\/$/, '') + '/health' })\n        .catch(response => response)\n        .then(response => {\n          const json = response.data || null;\n          if (response.status !== 200 || !json || (json && json.status !== 'ok')) {\n            /* eslint-disable no-console */\n            console.warn(\n              `\\nWARNING, Unable to connect to '${Parse.serverURL}'.` +\n                ` Cloud code and push notifications may be unavailable!\\n`\n            );\n            /* eslint-enable no-console */\n            if (callback) {\n              callback(false);\n            }\n          } else {\n            if (callback) {\n              callback(true);\n            }\n          }\n        });\n    }\n  }\n}\n\nfunction addParseCloud() {\n  const ParseCloud = require('./cloud-code/Parse.Cloud');\n  Object.defineProperty(Parse, 'Server', {\n    get() {\n      return Config.get(Parse.applicationId);\n    },\n    set(newVal) {\n      newVal.appId = Parse.applicationId;\n      Config.put(newVal);\n    },\n    configurable: true,\n  });\n  Object.assign(Parse.Cloud, ParseCloud);\n  global.Parse = Parse;\n}\n\nfunction injectDefaults(options: ParseServerOptions) {\n  Object.keys(defaults).forEach(key => {\n    if (!Object.prototype.hasOwnProperty.call(options, key)) {\n      options[key] = defaults[key];\n    }\n  });\n\n  if (!Object.prototype.hasOwnProperty.call(options, 'serverURL')) {\n    options.serverURL = `http://localhost:${options.port}${options.mountPath}`;\n  }\n\n  // Reserved Characters\n  if (options.appId) {\n    const regex = /[!#$%'()*+&/:;=?@[\\]{}^,|<>]/g;\n    if (options.appId.match(regex)) {\n      console.warn(\n        `\\nWARNING, appId that contains special characters can cause issues while using with urls.\\n`\n      );\n    }\n  }\n\n  // Backwards compatibility\n  if (options.userSensitiveFields) {\n    /* eslint-disable no-console */\n    !process.env.TESTING &&\n      console.warn(\n        `\\nDEPRECATED: userSensitiveFields has been replaced by protectedFields allowing the ability to protect fields in all classes with CLP. \\n`\n      );\n    /* eslint-enable no-console */\n\n    const userSensitiveFields = Array.from(\n      new Set([...(defaults.userSensitiveFields || []), ...(options.userSensitiveFields || [])])\n    );\n\n    // If the options.protectedFields is unset,\n    // it'll be assigned the default above.\n    // Here, protect against the case where protectedFields\n    // is set, but doesn't have _User.\n    if (!('_User' in options.protectedFields)) {\n      options.protectedFields = Object.assign({ _User: [] }, options.protectedFields);\n    }\n\n    options.protectedFields['_User']['*'] = Array.from(\n      new Set([...(options.protectedFields['_User']['*'] || []), ...userSensitiveFields])\n    );\n  }\n\n  // Merge protectedFields options with defaults.\n  Object.keys(defaults.protectedFields).forEach(c => {\n    const cur = options.protectedFields[c];\n    if (!cur) {\n      options.protectedFields[c] = defaults.protectedFields[c];\n    } else {\n      Object.keys(defaults.protectedFields[c]).forEach(r => {\n        const unq = new Set([\n          ...(options.protectedFields[c][r] || []),\n          ...defaults.protectedFields[c][r],\n        ]);\n        options.protectedFields[c][r] = Array.from(unq);\n      });\n    }\n  });\n\n  options.masterKeyIps = Array.from(\n    new Set(options.masterKeyIps.concat(defaults.masterKeyIps, options.masterKeyIps))\n  );\n}\n\n// Those can't be tested as it requires a subprocess\n/* istanbul ignore next */\nfunction configureListeners(parseServer) {\n  const server = parseServer.server;\n  const sockets = {};\n  /* Currently, express doesn't shut down immediately after receiving SIGINT/SIGTERM if it has client connections that haven't timed out. (This is a known issue with node - https://github.com/nodejs/node/issues/2642)\n    This function, along with `destroyAliveConnections()`, intend to fix this behavior such that parse server will close all open connections and initiate the shutdown process as soon as it receives a SIGINT/SIGTERM signal. */\n  server.on('connection', socket => {\n    const socketId = socket.remoteAddress + ':' + socket.remotePort;\n    sockets[socketId] = socket;\n    socket.on('close', () => {\n      delete sockets[socketId];\n    });\n  });\n\n  const destroyAliveConnections = function () {\n    for (const socketId in sockets) {\n      try {\n        sockets[socketId].destroy();\n      } catch (e) {\n        /* */\n      }\n    }\n  };\n\n  const handleShutdown = function () {\n    process.stdout.write('Termination signal received. Shutting down.');\n    destroyAliveConnections();\n    server.close();\n    parseServer.handleShutdown();\n  };\n  process.on('SIGTERM', handleShutdown);\n  process.on('SIGINT', handleShutdown);\n}\n\nexport default ParseServer;\n"]}
|
|
494
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["batch","require","bodyParser","express","middlewares","Parse","parse","path","fs","addParseCloud","ParseServer","constructor","options","Deprecator","scanParseServerOptions","injectDefaults","appId","requiredParameter","masterKey","javascriptKey","serverURL","initialize","allControllers","controllers","getControllers","state","config","Config","put","Object","assign","logging","setLogger","loggerController","start","databaseController","hooksController","cloud","security","schema","cacheAdapter","liveQueryController","performInitialization","e","code","Error","DUPLICATE_VALUE","load","startupPromises","push","DefinedSchemas","execute","connect","Promise","all","resolve","json","process","env","npm_package_json","npm_package_type","type","cwd","default","setTimeout","enableCheck","enableCheckLog","CheckRunner","run","error","console","app","_app","handleShutdown","promises","adapter","databaseAdapter","fileAdapter","filesController","cacheController","length","then","serverCloseComplete","maxUploadSize","directAccess","pages","rateLimit","api","use","allowCrossDomain","FilesRouter","expressRouter","req","res","status","set","urlencoded","extended","enableRouter","PagesRouter","PublicAPIRouter","limit","allowMethodOverride","handleParseHeaders","routes","Array","isArray","route","addRateLimit","handleParseSession","appRouter","promiseRouter","handleParseErrors","TESTING","on","err","stderr","write","port","exit","verifyServerUrl","PARSE_SERVER_ENABLE_EXPERIMENTAL_DIRECT_ACCESS","CoreManager","setRESTController","ParseServerRESTController","routers","ClassesRouter","UsersRouter","SessionsRouter","RolesRouter","AnalyticsRouter","InstallationsRouter","FunctionsRouter","SchemasRouter","PushRouter","LogsRouter","IAPValidationRouter","FeaturesRouter","GlobalConfigRouter","GraphQLRouter","PurgeRouter","HooksRouter","CloudCodeRouter","AudiencesRouter","AggregateRouter","SecurityRouter","reduce","memo","router","concat","PromiseRouter","mountOnto","startApp","middleware","mountPath","mountGraphQL","mountPlayground","graphQLCustomTypeDefs","undefined","graphQLSchema","readFileSync","parseGraphQLServer","ParseGraphQLServer","graphQLPath","playgroundPath","applyGraphQL","applyPlayground","server","listen","host","startLiveQueryServer","liveQueryServerOptions","liveQueryServer","createLiveQueryServer","trustProxy","configureListeners","expressApp","parseServer","httpServer","createServer","ParseLiveQueryServer","callback","request","url","replace","catch","response","data","warn","ParseCloud","defineProperty","get","applicationId","newVal","configurable","Cloud","global","keys","defaults","forEach","key","prototype","hasOwnProperty","call","regex","match","userSensitiveFields","from","Set","protectedFields","_User","c","cur","r","unq","masterKeyIps","sockets","socket","socketId","remoteAddress","remotePort","destroyAliveConnections","destroy","stdout","close"],"sources":["../src/ParseServer.js"],"sourcesContent":["// ParseServer - open-source compatible API Server for Parse apps\n\nvar batch = require('./batch'),\n  bodyParser = require('body-parser'),\n  express = require('express'),\n  middlewares = require('./middlewares'),\n  Parse = require('parse/node').Parse,\n  { parse } = require('graphql'),\n  path = require('path'),\n  fs = require('fs');\n\nimport { ParseServerOptions, LiveQueryServerOptions } from './Options';\nimport defaults from './defaults';\nimport * as logging from './logger';\nimport Config from './Config';\nimport PromiseRouter from './PromiseRouter';\nimport requiredParameter from './requiredParameter';\nimport { AnalyticsRouter } from './Routers/AnalyticsRouter';\nimport { ClassesRouter } from './Routers/ClassesRouter';\nimport { FeaturesRouter } from './Routers/FeaturesRouter';\nimport { FilesRouter } from './Routers/FilesRouter';\nimport { FunctionsRouter } from './Routers/FunctionsRouter';\nimport { GlobalConfigRouter } from './Routers/GlobalConfigRouter';\nimport { GraphQLRouter } from './Routers/GraphQLRouter';\nimport { HooksRouter } from './Routers/HooksRouter';\nimport { IAPValidationRouter } from './Routers/IAPValidationRouter';\nimport { InstallationsRouter } from './Routers/InstallationsRouter';\nimport { LogsRouter } from './Routers/LogsRouter';\nimport { ParseLiveQueryServer } from './LiveQuery/ParseLiveQueryServer';\nimport { PagesRouter } from './Routers/PagesRouter';\nimport { PublicAPIRouter } from './Routers/PublicAPIRouter';\nimport { PushRouter } from './Routers/PushRouter';\nimport { CloudCodeRouter } from './Routers/CloudCodeRouter';\nimport { RolesRouter } from './Routers/RolesRouter';\nimport { SchemasRouter } from './Routers/SchemasRouter';\nimport { SessionsRouter } from './Routers/SessionsRouter';\nimport { UsersRouter } from './Routers/UsersRouter';\nimport { PurgeRouter } from './Routers/PurgeRouter';\nimport { AudiencesRouter } from './Routers/AudiencesRouter';\nimport { AggregateRouter } from './Routers/AggregateRouter';\nimport { ParseServerRESTController } from './ParseServerRESTController';\nimport * as controllers from './Controllers';\nimport { ParseGraphQLServer } from './GraphQL/ParseGraphQLServer';\nimport { SecurityRouter } from './Routers/SecurityRouter';\nimport CheckRunner from './Security/CheckRunner';\nimport Deprecator from './Deprecator/Deprecator';\nimport { DefinedSchemas } from './SchemaMigrations/DefinedSchemas';\n\n// Mutate the Parse object to add the Cloud Code handlers\naddParseCloud();\n\n// ParseServer works like a constructor of an express app.\n// https://parseplatform.org/parse-server/api/master/ParseServerOptions.html\nclass ParseServer {\n  /**\n   * @constructor\n   * @param {ParseServerOptions} options the parse server initialization options\n   */\n  constructor(options: ParseServerOptions) {\n    // Scan for deprecated Parse Server options\n    Deprecator.scanParseServerOptions(options);\n    // Set option defaults\n    injectDefaults(options);\n    const {\n      appId = requiredParameter('You must provide an appId!'),\n      masterKey = requiredParameter('You must provide a masterKey!'),\n      javascriptKey,\n      serverURL = requiredParameter('You must provide a serverURL!'),\n    } = options;\n    // Initialize the node client SDK automatically\n    Parse.initialize(appId, javascriptKey || 'unused', masterKey);\n    Parse.serverURL = serverURL;\n\n    const allControllers = controllers.getControllers(options);\n    options.state = 'initialized';\n    this.config = Config.put(Object.assign({}, options, allControllers));\n    logging.setLogger(allControllers.loggerController);\n  }\n\n  /**\n   * Starts Parse Server as an express app; this promise resolves when Parse Server is ready to accept requests.\n   */\n\n  async start() {\n    try {\n      if (this.config.state === 'ok') {\n        return this;\n      }\n      this.config.state = 'starting';\n      Config.put(this.config);\n      const {\n        databaseController,\n        hooksController,\n        cloud,\n        security,\n        schema,\n        cacheAdapter,\n        liveQueryController,\n      } = this.config;\n      try {\n        await databaseController.performInitialization();\n      } catch (e) {\n        if (e.code !== Parse.Error.DUPLICATE_VALUE) {\n          throw e;\n        }\n      }\n      await hooksController.load();\n      const startupPromises = [];\n      if (schema) {\n        startupPromises.push(new DefinedSchemas(schema, this.config).execute());\n      }\n      if (cacheAdapter?.connect && typeof cacheAdapter.connect === 'function') {\n        startupPromises.push(cacheAdapter.connect());\n      }\n      startupPromises.push(liveQueryController.connect());\n      await Promise.all(startupPromises);\n      if (cloud) {\n        addParseCloud();\n        if (typeof cloud === 'function') {\n          await Promise.resolve(cloud(Parse));\n        } else if (typeof cloud === 'string') {\n          let json;\n          if (process.env.npm_package_json) {\n            json = require(process.env.npm_package_json);\n          }\n          if (process.env.npm_package_type === 'module' || json?.type === 'module') {\n            await import(path.resolve(process.cwd(), cloud)).default;\n          } else {\n            require(path.resolve(process.cwd(), cloud));\n          }\n        } else {\n          throw \"argument 'cloud' must either be a string or a function\";\n        }\n        await new Promise(resolve => setTimeout(resolve, 10));\n      }\n      if (security && security.enableCheck && security.enableCheckLog) {\n        new CheckRunner(security).run();\n      }\n      this.config.state = 'ok';\n      Config.put(this.config);\n      return this;\n    } catch (error) {\n      console.error(error);\n      this.config.state = 'error';\n      throw error;\n    }\n  }\n\n  get app() {\n    if (!this._app) {\n      this._app = ParseServer.app(this.config);\n    }\n    return this._app;\n  }\n\n  handleShutdown() {\n    const promises = [];\n    const { adapter: databaseAdapter } = this.config.databaseController;\n    if (databaseAdapter && typeof databaseAdapter.handleShutdown === 'function') {\n      promises.push(databaseAdapter.handleShutdown());\n    }\n    const { adapter: fileAdapter } = this.config.filesController;\n    if (fileAdapter && typeof fileAdapter.handleShutdown === 'function') {\n      promises.push(fileAdapter.handleShutdown());\n    }\n    const { adapter: cacheAdapter } = this.config.cacheController;\n    if (cacheAdapter && typeof cacheAdapter.handleShutdown === 'function') {\n      promises.push(cacheAdapter.handleShutdown());\n    }\n    return (promises.length > 0 ? Promise.all(promises) : Promise.resolve()).then(() => {\n      if (this.config.serverCloseComplete) {\n        this.config.serverCloseComplete();\n      }\n    });\n  }\n\n  /**\n   * @static\n   * Create an express app for the parse server\n   * @param {Object} options let you specify the maxUploadSize when creating the express app  */\n  static app(options) {\n    const { maxUploadSize = '20mb', appId, directAccess, pages, rateLimit = [] } = options;\n    // This app serves the Parse API directly.\n    // It's the equivalent of https://api.parse.com/1 in the hosted Parse API.\n    var api = express();\n    //api.use(\"/apps\", express.static(__dirname + \"/public\"));\n    api.use(middlewares.allowCrossDomain(appId));\n    // File handling needs to be before default middlewares are applied\n    api.use(\n      '/',\n      new FilesRouter().expressRouter({\n        maxUploadSize: maxUploadSize,\n      })\n    );\n\n    api.use('/health', function (req, res) {\n      res.status(options.state === 'ok' ? 200 : 503);\n      if (options.state === 'starting') {\n        res.set('Retry-After', 1);\n      }\n      res.json({\n        status: options.state,\n      });\n    });\n\n    api.use(\n      '/',\n      bodyParser.urlencoded({ extended: false }),\n      pages.enableRouter\n        ? new PagesRouter(pages).expressRouter()\n        : new PublicAPIRouter().expressRouter()\n    );\n\n    api.use(bodyParser.json({ type: '*/*', limit: maxUploadSize }));\n    api.use(middlewares.allowMethodOverride);\n    api.use(middlewares.handleParseHeaders);\n    const routes = Array.isArray(rateLimit) ? rateLimit : [rateLimit];\n    for (const route of routes) {\n      middlewares.addRateLimit(route, options);\n    }\n    api.use(middlewares.handleParseSession);\n\n    const appRouter = ParseServer.promiseRouter({ appId });\n    api.use(appRouter.expressRouter());\n\n    api.use(middlewares.handleParseErrors);\n\n    // run the following when not testing\n    if (!process.env.TESTING) {\n      //This causes tests to spew some useless warnings, so disable in test\n      /* istanbul ignore next */\n      process.on('uncaughtException', err => {\n        if (err.code === 'EADDRINUSE') {\n          // user-friendly message for this common error\n          process.stderr.write(`Unable to listen on port ${err.port}. The port is already in use.`);\n          process.exit(0);\n        } else {\n          throw err;\n        }\n      });\n      // verify the server url after a 'mount' event is received\n      /* istanbul ignore next */\n      api.on('mount', function () {\n        ParseServer.verifyServerUrl();\n      });\n    }\n    if (process.env.PARSE_SERVER_ENABLE_EXPERIMENTAL_DIRECT_ACCESS === '1' || directAccess) {\n      Parse.CoreManager.setRESTController(ParseServerRESTController(appId, appRouter));\n    }\n    return api;\n  }\n\n  static promiseRouter({ appId }) {\n    const routers = [\n      new ClassesRouter(),\n      new UsersRouter(),\n      new SessionsRouter(),\n      new RolesRouter(),\n      new AnalyticsRouter(),\n      new InstallationsRouter(),\n      new FunctionsRouter(),\n      new SchemasRouter(),\n      new PushRouter(),\n      new LogsRouter(),\n      new IAPValidationRouter(),\n      new FeaturesRouter(),\n      new GlobalConfigRouter(),\n      new GraphQLRouter(),\n      new PurgeRouter(),\n      new HooksRouter(),\n      new CloudCodeRouter(),\n      new AudiencesRouter(),\n      new AggregateRouter(),\n      new SecurityRouter(),\n    ];\n\n    const routes = routers.reduce((memo, router) => {\n      return memo.concat(router.routes);\n    }, []);\n\n    const appRouter = new PromiseRouter(routes, appId);\n\n    batch.mountOnto(appRouter);\n    return appRouter;\n  }\n\n  /**\n   * starts the parse server's express app\n   * @param {ParseServerOptions} options to use to start the server\n   * @returns {ParseServer} the parse server instance\n   */\n\n  async startApp(options: ParseServerOptions) {\n    try {\n      await this.start();\n    } catch (e) {\n      console.error('Error on ParseServer.startApp: ', e);\n      throw e;\n    }\n    const app = express();\n    if (options.middleware) {\n      let middleware;\n      if (typeof options.middleware == 'string') {\n        middleware = require(path.resolve(process.cwd(), options.middleware));\n      } else {\n        middleware = options.middleware; // use as-is let express fail\n      }\n      app.use(middleware);\n    }\n    app.use(options.mountPath, this.app);\n\n    if (options.mountGraphQL === true || options.mountPlayground === true) {\n      let graphQLCustomTypeDefs = undefined;\n      if (typeof options.graphQLSchema === 'string') {\n        graphQLCustomTypeDefs = parse(fs.readFileSync(options.graphQLSchema, 'utf8'));\n      } else if (\n        typeof options.graphQLSchema === 'object' ||\n        typeof options.graphQLSchema === 'function'\n      ) {\n        graphQLCustomTypeDefs = options.graphQLSchema;\n      }\n\n      const parseGraphQLServer = new ParseGraphQLServer(this, {\n        graphQLPath: options.graphQLPath,\n        playgroundPath: options.playgroundPath,\n        graphQLCustomTypeDefs,\n      });\n\n      if (options.mountGraphQL) {\n        parseGraphQLServer.applyGraphQL(app);\n      }\n\n      if (options.mountPlayground) {\n        parseGraphQLServer.applyPlayground(app);\n      }\n    }\n    const server = await new Promise(resolve => {\n      app.listen(options.port, options.host, function () {\n        resolve(this);\n      });\n    });\n    this.server = server;\n\n    if (options.startLiveQueryServer || options.liveQueryServerOptions) {\n      this.liveQueryServer = await ParseServer.createLiveQueryServer(\n        server,\n        options.liveQueryServerOptions,\n        options\n      );\n    }\n    if (options.trustProxy) {\n      app.set('trust proxy', options.trustProxy);\n    }\n    /* istanbul ignore next */\n    if (!process.env.TESTING) {\n      configureListeners(this);\n    }\n    this.expressApp = app;\n    return this;\n  }\n\n  /**\n   * Creates a new ParseServer and starts it.\n   * @param {ParseServerOptions} options used to start the server\n   * @returns {ParseServer} the parse server instance\n   */\n  static async startApp(options: ParseServerOptions) {\n    const parseServer = new ParseServer(options);\n    return parseServer.startApp(options);\n  }\n\n  /**\n   * Helper method to create a liveQuery server\n   * @static\n   * @param {Server} httpServer an optional http server to pass\n   * @param {LiveQueryServerOptions} config options for the liveQueryServer\n   * @param {ParseServerOptions} options options for the ParseServer\n   * @returns {Promise<ParseLiveQueryServer>} the live query server instance\n   */\n  static async createLiveQueryServer(\n    httpServer,\n    config: LiveQueryServerOptions,\n    options: ParseServerOptions\n  ) {\n    if (!httpServer || (config && config.port)) {\n      var app = express();\n      httpServer = require('http').createServer(app);\n      httpServer.listen(config.port);\n    }\n    const server = new ParseLiveQueryServer(httpServer, config, options);\n    await server.connect();\n    return server;\n  }\n\n  static verifyServerUrl(callback) {\n    // perform a health check on the serverURL value\n    if (Parse.serverURL) {\n      const request = require('./request');\n      request({ url: Parse.serverURL.replace(/\\/$/, '') + '/health' })\n        .catch(response => response)\n        .then(response => {\n          const json = response.data || null;\n          if (response.status !== 200 || !json || (json && json.status !== 'ok')) {\n            /* eslint-disable no-console */\n            console.warn(\n              `\\nWARNING, Unable to connect to '${Parse.serverURL}'.` +\n                ` Cloud code and push notifications may be unavailable!\\n`\n            );\n            /* eslint-enable no-console */\n            if (callback) {\n              callback(false);\n            }\n          } else {\n            if (callback) {\n              callback(true);\n            }\n          }\n        });\n    }\n  }\n}\n\nfunction addParseCloud() {\n  const ParseCloud = require('./cloud-code/Parse.Cloud');\n  Object.defineProperty(Parse, 'Server', {\n    get() {\n      return Config.get(Parse.applicationId);\n    },\n    set(newVal) {\n      newVal.appId = Parse.applicationId;\n      Config.put(newVal);\n    },\n    configurable: true,\n  });\n  Object.assign(Parse.Cloud, ParseCloud);\n  global.Parse = Parse;\n}\n\nfunction injectDefaults(options: ParseServerOptions) {\n  Object.keys(defaults).forEach(key => {\n    if (!Object.prototype.hasOwnProperty.call(options, key)) {\n      options[key] = defaults[key];\n    }\n  });\n\n  if (!Object.prototype.hasOwnProperty.call(options, 'serverURL')) {\n    options.serverURL = `http://localhost:${options.port}${options.mountPath}`;\n  }\n\n  // Reserved Characters\n  if (options.appId) {\n    const regex = /[!#$%'()*+&/:;=?@[\\]{}^,|<>]/g;\n    if (options.appId.match(regex)) {\n      console.warn(\n        `\\nWARNING, appId that contains special characters can cause issues while using with urls.\\n`\n      );\n    }\n  }\n\n  // Backwards compatibility\n  if (options.userSensitiveFields) {\n    /* eslint-disable no-console */\n    !process.env.TESTING &&\n      console.warn(\n        `\\nDEPRECATED: userSensitiveFields has been replaced by protectedFields allowing the ability to protect fields in all classes with CLP. \\n`\n      );\n    /* eslint-enable no-console */\n\n    const userSensitiveFields = Array.from(\n      new Set([...(defaults.userSensitiveFields || []), ...(options.userSensitiveFields || [])])\n    );\n\n    // If the options.protectedFields is unset,\n    // it'll be assigned the default above.\n    // Here, protect against the case where protectedFields\n    // is set, but doesn't have _User.\n    if (!('_User' in options.protectedFields)) {\n      options.protectedFields = Object.assign({ _User: [] }, options.protectedFields);\n    }\n\n    options.protectedFields['_User']['*'] = Array.from(\n      new Set([...(options.protectedFields['_User']['*'] || []), ...userSensitiveFields])\n    );\n  }\n\n  // Merge protectedFields options with defaults.\n  Object.keys(defaults.protectedFields).forEach(c => {\n    const cur = options.protectedFields[c];\n    if (!cur) {\n      options.protectedFields[c] = defaults.protectedFields[c];\n    } else {\n      Object.keys(defaults.protectedFields[c]).forEach(r => {\n        const unq = new Set([\n          ...(options.protectedFields[c][r] || []),\n          ...defaults.protectedFields[c][r],\n        ]);\n        options.protectedFields[c][r] = Array.from(unq);\n      });\n    }\n  });\n\n  options.masterKeyIps = Array.from(\n    new Set(options.masterKeyIps.concat(defaults.masterKeyIps, options.masterKeyIps))\n  );\n}\n\n// Those can't be tested as it requires a subprocess\n/* istanbul ignore next */\nfunction configureListeners(parseServer) {\n  const server = parseServer.server;\n  const sockets = {};\n  /* Currently, express doesn't shut down immediately after receiving SIGINT/SIGTERM if it has client connections that haven't timed out. (This is a known issue with node - https://github.com/nodejs/node/issues/2642)\n    This function, along with `destroyAliveConnections()`, intend to fix this behavior such that parse server will close all open connections and initiate the shutdown process as soon as it receives a SIGINT/SIGTERM signal. */\n  server.on('connection', socket => {\n    const socketId = socket.remoteAddress + ':' + socket.remotePort;\n    sockets[socketId] = socket;\n    socket.on('close', () => {\n      delete sockets[socketId];\n    });\n  });\n\n  const destroyAliveConnections = function () {\n    for (const socketId in sockets) {\n      try {\n        sockets[socketId].destroy();\n      } catch (e) {\n        /* */\n      }\n    }\n  };\n\n  const handleShutdown = function () {\n    process.stdout.write('Termination signal received. Shutting down.');\n    destroyAliveConnections();\n    server.close();\n    parseServer.handleShutdown();\n  };\n  process.on('SIGTERM', handleShutdown);\n  process.on('SIGINT', handleShutdown);\n}\n\nexport default ParseServer;\n"],"mappings":"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"}
|