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
|
@@ -4,27 +4,16 @@ Object.defineProperty(exports, "__esModule", {
|
|
|
4
4
|
value: true
|
|
5
5
|
});
|
|
6
6
|
exports.default = exports.PagesRouter = void 0;
|
|
7
|
-
|
|
8
7
|
var _PromiseRouter = _interopRequireDefault(require("../PromiseRouter"));
|
|
9
|
-
|
|
10
8
|
var _Config = _interopRequireDefault(require("../Config"));
|
|
11
|
-
|
|
12
9
|
var _express = _interopRequireDefault(require("express"));
|
|
13
|
-
|
|
14
10
|
var _path = _interopRequireDefault(require("path"));
|
|
15
|
-
|
|
16
11
|
var _fs = require("fs");
|
|
17
|
-
|
|
18
12
|
var _node = require("parse/node");
|
|
19
|
-
|
|
20
13
|
var _Utils = _interopRequireDefault(require("../Utils"));
|
|
21
|
-
|
|
22
14
|
var _mustache = _interopRequireDefault(require("mustache"));
|
|
23
|
-
|
|
24
15
|
var _Page = _interopRequireDefault(require("../Page"));
|
|
25
|
-
|
|
26
16
|
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
|
|
27
|
-
|
|
28
17
|
// All pages with custom page key for reference and file name
|
|
29
18
|
const pages = Object.freeze({
|
|
30
19
|
passwordReset: new _Page.default({
|
|
@@ -59,8 +48,9 @@ const pages = Object.freeze({
|
|
|
59
48
|
id: 'emailVerificationLinkExpired',
|
|
60
49
|
defaultFile: 'email_verification_link_expired.html'
|
|
61
50
|
})
|
|
62
|
-
});
|
|
51
|
+
});
|
|
63
52
|
|
|
53
|
+
// All page parameters for reference to be used as template placeholders or query params
|
|
64
54
|
const pageParams = Object.freeze({
|
|
65
55
|
appName: 'appName',
|
|
66
56
|
appId: 'appId',
|
|
@@ -69,23 +59,25 @@ const pageParams = Object.freeze({
|
|
|
69
59
|
error: 'error',
|
|
70
60
|
locale: 'locale',
|
|
71
61
|
publicServerUrl: 'publicServerUrl'
|
|
72
|
-
});
|
|
62
|
+
});
|
|
73
63
|
|
|
74
|
-
|
|
64
|
+
// The header prefix to add page params as response headers
|
|
65
|
+
const pageParamHeaderPrefix = 'x-parse-page-param-';
|
|
75
66
|
|
|
67
|
+
// The errors being thrown
|
|
76
68
|
const errors = Object.freeze({
|
|
77
69
|
jsonFailedFileLoading: 'failed to load JSON file',
|
|
78
70
|
fileOutsideAllowedScope: 'not allowed to read file outside of pages directory'
|
|
79
71
|
});
|
|
80
|
-
|
|
81
72
|
class PagesRouter extends _PromiseRouter.default {
|
|
82
73
|
/**
|
|
83
74
|
* Constructs a PagesRouter.
|
|
84
75
|
* @param {Object} pages The pages options from the Parse Server configuration.
|
|
85
76
|
*/
|
|
86
77
|
constructor(pages = {}) {
|
|
87
|
-
super();
|
|
78
|
+
super();
|
|
88
79
|
|
|
80
|
+
// Set instance properties
|
|
89
81
|
this.pagesConfig = pages;
|
|
90
82
|
this.pagesEndpoint = pages.pagesEndpoint ? pages.pagesEndpoint : 'apps';
|
|
91
83
|
this.pagesPath = pages.pagesPath ? _path.default.resolve('./', pages.pagesPath) : _path.default.resolve(__dirname, '../../public');
|
|
@@ -94,7 +86,6 @@ class PagesRouter extends _PromiseRouter.default {
|
|
|
94
86
|
this.mountCustomRoutes();
|
|
95
87
|
this.mountStaticRoute();
|
|
96
88
|
}
|
|
97
|
-
|
|
98
89
|
verifyEmail(req) {
|
|
99
90
|
const config = req.config;
|
|
100
91
|
const {
|
|
@@ -102,15 +93,12 @@ class PagesRouter extends _PromiseRouter.default {
|
|
|
102
93
|
token: rawToken
|
|
103
94
|
} = req.query;
|
|
104
95
|
const token = rawToken && typeof rawToken !== 'string' ? rawToken.toString() : rawToken;
|
|
105
|
-
|
|
106
96
|
if (!config) {
|
|
107
97
|
this.invalidRequest();
|
|
108
98
|
}
|
|
109
|
-
|
|
110
99
|
if (!token || !username) {
|
|
111
100
|
return this.goToPage(req, pages.emailVerificationLinkInvalid);
|
|
112
101
|
}
|
|
113
|
-
|
|
114
102
|
const userController = config.userController;
|
|
115
103
|
return userController.verifyEmail(username, token).then(() => {
|
|
116
104
|
const params = {
|
|
@@ -124,19 +112,15 @@ class PagesRouter extends _PromiseRouter.default {
|
|
|
124
112
|
return this.goToPage(req, pages.emailVerificationLinkExpired, params);
|
|
125
113
|
});
|
|
126
114
|
}
|
|
127
|
-
|
|
128
115
|
resendVerificationEmail(req) {
|
|
129
116
|
const config = req.config;
|
|
130
117
|
const username = req.body.username;
|
|
131
|
-
|
|
132
118
|
if (!config) {
|
|
133
119
|
this.invalidRequest();
|
|
134
120
|
}
|
|
135
|
-
|
|
136
121
|
if (!username) {
|
|
137
122
|
return this.goToPage(req, pages.emailVerificationLinkInvalid);
|
|
138
123
|
}
|
|
139
|
-
|
|
140
124
|
const userController = config.userController;
|
|
141
125
|
return userController.resendVerificationEmail(username).then(() => {
|
|
142
126
|
return this.goToPage(req, pages.emailVerificationSendSuccess);
|
|
@@ -144,7 +128,6 @@ class PagesRouter extends _PromiseRouter.default {
|
|
|
144
128
|
return this.goToPage(req, pages.emailVerificationSendFail);
|
|
145
129
|
});
|
|
146
130
|
}
|
|
147
|
-
|
|
148
131
|
passwordReset(req) {
|
|
149
132
|
const config = req.config;
|
|
150
133
|
const params = {
|
|
@@ -156,24 +139,19 @@ class PagesRouter extends _PromiseRouter.default {
|
|
|
156
139
|
};
|
|
157
140
|
return this.goToPage(req, pages.passwordReset, params);
|
|
158
141
|
}
|
|
159
|
-
|
|
160
142
|
requestResetPassword(req) {
|
|
161
143
|
const config = req.config;
|
|
162
|
-
|
|
163
144
|
if (!config) {
|
|
164
145
|
this.invalidRequest();
|
|
165
146
|
}
|
|
166
|
-
|
|
167
147
|
const {
|
|
168
148
|
username,
|
|
169
149
|
token: rawToken
|
|
170
150
|
} = req.query;
|
|
171
151
|
const token = rawToken && typeof rawToken !== 'string' ? rawToken.toString() : rawToken;
|
|
172
|
-
|
|
173
152
|
if (!username || !token) {
|
|
174
153
|
return this.goToPage(req, pages.passwordResetLinkInvalid);
|
|
175
154
|
}
|
|
176
|
-
|
|
177
155
|
return config.userController.checkResetTokenValidity(username, token).then(() => {
|
|
178
156
|
const params = {
|
|
179
157
|
[pageParams.token]: token,
|
|
@@ -189,37 +167,29 @@ class PagesRouter extends _PromiseRouter.default {
|
|
|
189
167
|
return this.goToPage(req, pages.passwordResetLinkInvalid, params);
|
|
190
168
|
});
|
|
191
169
|
}
|
|
192
|
-
|
|
193
170
|
resetPassword(req) {
|
|
194
171
|
const config = req.config;
|
|
195
|
-
|
|
196
172
|
if (!config) {
|
|
197
173
|
this.invalidRequest();
|
|
198
174
|
}
|
|
199
|
-
|
|
200
175
|
const {
|
|
201
176
|
username,
|
|
202
177
|
new_password,
|
|
203
178
|
token: rawToken
|
|
204
179
|
} = req.body;
|
|
205
180
|
const token = rawToken && typeof rawToken !== 'string' ? rawToken.toString() : rawToken;
|
|
206
|
-
|
|
207
181
|
if ((!username || !token || !new_password) && req.xhr === false) {
|
|
208
182
|
return this.goToPage(req, pages.passwordResetLinkInvalid);
|
|
209
183
|
}
|
|
210
|
-
|
|
211
184
|
if (!username) {
|
|
212
185
|
throw new _node.Parse.Error(_node.Parse.Error.USERNAME_MISSING, 'Missing username');
|
|
213
186
|
}
|
|
214
|
-
|
|
215
187
|
if (!token) {
|
|
216
188
|
throw new _node.Parse.Error(_node.Parse.Error.OTHER_CAUSE, 'Missing token');
|
|
217
189
|
}
|
|
218
|
-
|
|
219
190
|
if (!new_password) {
|
|
220
191
|
throw new _node.Parse.Error(_node.Parse.Error.PASSWORD_MISSING, 'Missing password');
|
|
221
192
|
}
|
|
222
|
-
|
|
223
193
|
return config.userController.updatePassword(username, token, new_password).then(() => {
|
|
224
194
|
return Promise.resolve({
|
|
225
195
|
success: true
|
|
@@ -237,12 +207,10 @@ class PagesRouter extends _PromiseRouter.default {
|
|
|
237
207
|
response: 'Password successfully reset'
|
|
238
208
|
});
|
|
239
209
|
}
|
|
240
|
-
|
|
241
210
|
if (result.err) {
|
|
242
211
|
throw new _node.Parse.Error(_node.Parse.Error.OTHER_CAUSE, `${result.err}`);
|
|
243
212
|
}
|
|
244
213
|
}
|
|
245
|
-
|
|
246
214
|
const query = result.success ? {
|
|
247
215
|
[pageParams.username]: username
|
|
248
216
|
} : {
|
|
@@ -256,6 +224,7 @@ class PagesRouter extends _PromiseRouter.default {
|
|
|
256
224
|
return this.goToPage(req, page, query, false);
|
|
257
225
|
});
|
|
258
226
|
}
|
|
227
|
+
|
|
259
228
|
/**
|
|
260
229
|
* Returns page content if the page is a local file or returns a
|
|
261
230
|
* redirect to a custom page.
|
|
@@ -271,44 +240,43 @@ class PagesRouter extends _PromiseRouter.default {
|
|
|
271
240
|
* - POST request -> redirect response (PRG pattern)
|
|
272
241
|
* @returns {Promise<Object>} The PromiseRouter response.
|
|
273
242
|
*/
|
|
274
|
-
|
|
275
|
-
|
|
276
243
|
goToPage(req, page, params = {}, responseType) {
|
|
277
|
-
const config = req.config;
|
|
244
|
+
const config = req.config;
|
|
278
245
|
|
|
279
|
-
|
|
246
|
+
// Determine redirect either by force, response setting or request method
|
|
247
|
+
const redirect = config.pages.forceRedirect ? true : responseType !== undefined ? responseType : req.method == 'POST';
|
|
280
248
|
|
|
249
|
+
// Include default parameters
|
|
281
250
|
const defaultParams = this.getDefaultParams(config);
|
|
282
|
-
|
|
283
251
|
if (Object.values(defaultParams).includes(undefined)) {
|
|
284
252
|
return this.notFound();
|
|
285
253
|
}
|
|
254
|
+
params = Object.assign(params, defaultParams);
|
|
286
255
|
|
|
287
|
-
|
|
256
|
+
// Add locale to params to ensure it is passed on with every request;
|
|
288
257
|
// that means, once a locale is set, it is passed on to any follow-up page,
|
|
289
258
|
// e.g. request_password_reset -> password_reset -> password_reset_success
|
|
290
|
-
|
|
291
259
|
const locale = this.getLocale(req);
|
|
292
|
-
params[pageParams.locale] = locale;
|
|
260
|
+
params[pageParams.locale] = locale;
|
|
293
261
|
|
|
262
|
+
// Compose paths and URLs
|
|
294
263
|
const defaultFile = page.defaultFile;
|
|
295
264
|
const defaultPath = this.defaultPagePath(defaultFile);
|
|
296
|
-
const defaultUrl = this.composePageUrl(defaultFile, config.publicServerURL);
|
|
265
|
+
const defaultUrl = this.composePageUrl(defaultFile, config.publicServerURL);
|
|
297
266
|
|
|
267
|
+
// If custom URL is set redirect to it without localization
|
|
298
268
|
const customUrl = config.pages.customUrls[page.id];
|
|
299
|
-
|
|
300
269
|
if (customUrl && !_Utils.default.isPath(customUrl)) {
|
|
301
270
|
return this.redirectResponse(customUrl, params);
|
|
302
|
-
}
|
|
303
|
-
|
|
271
|
+
}
|
|
304
272
|
|
|
273
|
+
// Get JSON placeholders
|
|
305
274
|
let placeholders = {};
|
|
306
|
-
|
|
307
275
|
if (config.pages.enableLocalization && config.pages.localizationJsonPath) {
|
|
308
276
|
placeholders = this.getJsonPlaceholders(locale, params);
|
|
309
|
-
}
|
|
310
|
-
|
|
277
|
+
}
|
|
311
278
|
|
|
279
|
+
// Send response
|
|
312
280
|
if (config.pages.enableLocalization && locale) {
|
|
313
281
|
return _Utils.default.getLocalizedPath(defaultPath, locale).then(({
|
|
314
282
|
path,
|
|
@@ -318,37 +286,37 @@ class PagesRouter extends _PromiseRouter.default {
|
|
|
318
286
|
return redirect ? this.redirectResponse(defaultUrl, params) : this.pageResponse(defaultPath, params, placeholders);
|
|
319
287
|
}
|
|
320
288
|
}
|
|
289
|
+
|
|
321
290
|
/**
|
|
322
291
|
* Serves a request to a static resource and localizes the resource if it
|
|
323
292
|
* is a HTML file.
|
|
324
293
|
* @param {Object} req The request object.
|
|
325
294
|
* @returns {Promise<Object>} The response.
|
|
326
295
|
*/
|
|
327
|
-
|
|
328
|
-
|
|
329
296
|
staticRoute(req) {
|
|
330
297
|
// Get requested path
|
|
331
|
-
const relativePath = req.params[0];
|
|
332
|
-
|
|
333
|
-
const absolutePath = _path.default.resolve(this.pagesPath, relativePath); // If the requested file is not a HTML file send its raw content
|
|
298
|
+
const relativePath = req.params[0];
|
|
334
299
|
|
|
300
|
+
// Resolve requested path to absolute path
|
|
301
|
+
const absolutePath = _path.default.resolve(this.pagesPath, relativePath);
|
|
335
302
|
|
|
303
|
+
// If the requested file is not a HTML file send its raw content
|
|
336
304
|
if (!absolutePath || !absolutePath.endsWith('.html')) {
|
|
337
305
|
return this.fileResponse(absolutePath);
|
|
338
|
-
}
|
|
339
|
-
|
|
306
|
+
}
|
|
340
307
|
|
|
308
|
+
// Get parameters
|
|
341
309
|
const params = this.getDefaultParams(req.config);
|
|
342
310
|
const locale = this.getLocale(req);
|
|
343
|
-
|
|
344
311
|
if (locale) {
|
|
345
312
|
params.locale = locale;
|
|
346
|
-
}
|
|
347
|
-
|
|
313
|
+
}
|
|
348
314
|
|
|
315
|
+
// Get JSON placeholders
|
|
349
316
|
const placeholders = this.getJsonPlaceholders(locale, params);
|
|
350
317
|
return this.pageResponse(absolutePath, params, placeholders);
|
|
351
318
|
}
|
|
319
|
+
|
|
352
320
|
/**
|
|
353
321
|
* Returns a translation from the JSON resource for a given locale. The JSON
|
|
354
322
|
* resource is parsed according to i18next syntax.
|
|
@@ -377,22 +345,22 @@ class PagesRouter extends _PromiseRouter.default {
|
|
|
377
345
|
* @returns {Object} The translation or an empty object if no matching
|
|
378
346
|
* translation was found.
|
|
379
347
|
*/
|
|
380
|
-
|
|
381
|
-
|
|
382
348
|
getJsonTranslation(locale) {
|
|
383
349
|
// If there is no JSON resource
|
|
384
350
|
if (this.jsonParameters === undefined) {
|
|
385
351
|
return {};
|
|
386
|
-
}
|
|
387
|
-
|
|
352
|
+
}
|
|
388
353
|
|
|
389
|
-
|
|
354
|
+
// If locale is not set use the fallback locale
|
|
355
|
+
locale = locale || this.pagesConfig.localizationFallbackLocale;
|
|
390
356
|
|
|
357
|
+
// Get matching translation by locale, language or fallback locale
|
|
391
358
|
const language = locale.split('-')[0];
|
|
392
359
|
const resource = this.jsonParameters[locale] || this.jsonParameters[language] || this.jsonParameters[this.pagesConfig.localizationFallbackLocale] || {};
|
|
393
360
|
const translation = resource.translation || {};
|
|
394
361
|
return translation;
|
|
395
362
|
}
|
|
363
|
+
|
|
396
364
|
/**
|
|
397
365
|
* Returns a translation from the JSON resource for a given locale with
|
|
398
366
|
* placeholders filled in by given parameters.
|
|
@@ -402,23 +370,23 @@ class PagesRouter extends _PromiseRouter.default {
|
|
|
402
370
|
* @returns {Object} The translation or an empty object if no matching
|
|
403
371
|
* translation was found.
|
|
404
372
|
*/
|
|
405
|
-
|
|
406
|
-
|
|
407
373
|
getJsonPlaceholders(locale, params = {}) {
|
|
408
374
|
// If localization is disabled or there is no JSON resource
|
|
409
375
|
if (!this.pagesConfig.enableLocalization || !this.pagesConfig.localizationJsonPath) {
|
|
410
376
|
return {};
|
|
411
|
-
}
|
|
377
|
+
}
|
|
412
378
|
|
|
379
|
+
// Get JSON placeholders
|
|
380
|
+
let placeholders = this.getJsonTranslation(locale);
|
|
413
381
|
|
|
414
|
-
|
|
382
|
+
// Fill in any placeholders in the translation; this allows a translation
|
|
415
383
|
// to contain default placeholders like {{appName}} which are filled here
|
|
416
|
-
|
|
417
384
|
placeholders = JSON.stringify(placeholders);
|
|
418
385
|
placeholders = _mustache.default.render(placeholders, params);
|
|
419
386
|
placeholders = JSON.parse(placeholders);
|
|
420
387
|
return placeholders;
|
|
421
388
|
}
|
|
389
|
+
|
|
422
390
|
/**
|
|
423
391
|
* Creates a response with file content.
|
|
424
392
|
* @param {String} path The path of the file to return.
|
|
@@ -428,36 +396,32 @@ class PagesRouter extends _PromiseRouter.default {
|
|
|
428
396
|
* These will not be included in the response header.
|
|
429
397
|
* @returns {Object} The Promise Router response.
|
|
430
398
|
*/
|
|
431
|
-
|
|
432
|
-
|
|
433
399
|
async pageResponse(path, params = {}, placeholders = {}) {
|
|
434
400
|
// Get file content
|
|
435
401
|
let data;
|
|
436
|
-
|
|
437
402
|
try {
|
|
438
403
|
data = await this.readFile(path);
|
|
439
404
|
} catch (e) {
|
|
440
405
|
return this.notFound();
|
|
441
|
-
}
|
|
442
|
-
|
|
406
|
+
}
|
|
443
407
|
|
|
408
|
+
// Get config placeholders; can be an object, a function or an async function
|
|
444
409
|
let configPlaceholders = typeof this.pagesConfig.placeholders === 'function' ? this.pagesConfig.placeholders(params) : Object.prototype.toString.call(this.pagesConfig.placeholders) === '[object Object]' ? this.pagesConfig.placeholders : {};
|
|
445
|
-
|
|
446
410
|
if (configPlaceholders instanceof Promise) {
|
|
447
411
|
configPlaceholders = await configPlaceholders;
|
|
448
|
-
}
|
|
449
|
-
|
|
412
|
+
}
|
|
450
413
|
|
|
414
|
+
// Fill placeholders
|
|
451
415
|
const allPlaceholders = Object.assign({}, configPlaceholders, placeholders);
|
|
452
416
|
const paramsAndPlaceholders = Object.assign({}, params, allPlaceholders);
|
|
453
|
-
data = _mustache.default.render(data, paramsAndPlaceholders);
|
|
454
|
-
// of response, instead of having to parse the HTML content.
|
|
417
|
+
data = _mustache.default.render(data, paramsAndPlaceholders);
|
|
455
418
|
|
|
419
|
+
// Add placeholders in header to allow parsing for programmatic use
|
|
420
|
+
// of response, instead of having to parse the HTML content.
|
|
456
421
|
const headers = Object.entries(params).reduce((m, p) => {
|
|
457
422
|
if (p[1] !== undefined) {
|
|
458
423
|
m[`${pageParamHeaderPrefix}${p[0].toLowerCase()}`] = p[1];
|
|
459
424
|
}
|
|
460
|
-
|
|
461
425
|
return m;
|
|
462
426
|
}, {});
|
|
463
427
|
return {
|
|
@@ -465,27 +429,25 @@ class PagesRouter extends _PromiseRouter.default {
|
|
|
465
429
|
headers: headers
|
|
466
430
|
};
|
|
467
431
|
}
|
|
432
|
+
|
|
468
433
|
/**
|
|
469
434
|
* Creates a response with file content.
|
|
470
435
|
* @param {String} path The path of the file to return.
|
|
471
436
|
* @returns {Object} The PromiseRouter response.
|
|
472
437
|
*/
|
|
473
|
-
|
|
474
|
-
|
|
475
438
|
async fileResponse(path) {
|
|
476
439
|
// Get file content
|
|
477
440
|
let data;
|
|
478
|
-
|
|
479
441
|
try {
|
|
480
442
|
data = await this.readFile(path);
|
|
481
443
|
} catch (e) {
|
|
482
444
|
return this.notFound();
|
|
483
445
|
}
|
|
484
|
-
|
|
485
446
|
return {
|
|
486
447
|
text: data
|
|
487
448
|
};
|
|
488
449
|
}
|
|
450
|
+
|
|
489
451
|
/**
|
|
490
452
|
* Reads and returns the content of a file at a given path. File reading to
|
|
491
453
|
* serve content on the static route is only allowed from the pages
|
|
@@ -497,40 +459,35 @@ class PagesRouter extends _PromiseRouter.default {
|
|
|
497
459
|
* @param {String} filePath The path to the file to read.
|
|
498
460
|
* @returns {Promise<String>} The file content.
|
|
499
461
|
*/
|
|
500
|
-
|
|
501
|
-
|
|
502
462
|
async readFile(filePath) {
|
|
503
463
|
// Normalize path to prevent it from containing any directory changing
|
|
504
464
|
// UNIX patterns which could expose the whole file system, e.g.
|
|
505
465
|
// `http://example.com/parse/apps/../file.txt` requests a file outside
|
|
506
466
|
// of the pages directory scope.
|
|
507
|
-
const normalizedPath = _path.default.normalize(filePath);
|
|
508
|
-
|
|
467
|
+
const normalizedPath = _path.default.normalize(filePath);
|
|
509
468
|
|
|
469
|
+
// Abort if the path is outside of the path directory scope
|
|
510
470
|
if (!normalizedPath.startsWith(this.pagesPath)) {
|
|
511
471
|
throw errors.fileOutsideAllowedScope;
|
|
512
472
|
}
|
|
513
|
-
|
|
514
473
|
return await _fs.promises.readFile(normalizedPath, 'utf-8');
|
|
515
474
|
}
|
|
475
|
+
|
|
516
476
|
/**
|
|
517
477
|
* Loads a language resource JSON file that is used for translations.
|
|
518
478
|
*/
|
|
519
|
-
|
|
520
|
-
|
|
521
479
|
loadJsonResource() {
|
|
522
480
|
if (this.pagesConfig.localizationJsonPath === undefined) {
|
|
523
481
|
return;
|
|
524
482
|
}
|
|
525
|
-
|
|
526
483
|
try {
|
|
527
484
|
const json = require(_path.default.resolve('./', this.pagesConfig.localizationJsonPath));
|
|
528
|
-
|
|
529
485
|
this.jsonParameters = json;
|
|
530
486
|
} catch (e) {
|
|
531
487
|
throw errors.jsonFailedFileLoading;
|
|
532
488
|
}
|
|
533
489
|
}
|
|
490
|
+
|
|
534
491
|
/**
|
|
535
492
|
* Extracts and returns the page default parameters from the Parse Server
|
|
536
493
|
* configuration. These parameters are made accessible in every page served
|
|
@@ -538,8 +495,6 @@ class PagesRouter extends _PromiseRouter.default {
|
|
|
538
495
|
* @param {Object} config The Parse Server configuration.
|
|
539
496
|
* @returns {Object} The default parameters.
|
|
540
497
|
*/
|
|
541
|
-
|
|
542
|
-
|
|
543
498
|
getDefaultParams(config) {
|
|
544
499
|
return config ? {
|
|
545
500
|
[pageParams.appId]: config.appId,
|
|
@@ -547,17 +502,17 @@ class PagesRouter extends _PromiseRouter.default {
|
|
|
547
502
|
[pageParams.publicServerUrl]: config.publicServerURL
|
|
548
503
|
} : {};
|
|
549
504
|
}
|
|
505
|
+
|
|
550
506
|
/**
|
|
551
507
|
* Extracts and returns the locale from an express request.
|
|
552
508
|
* @param {Object} req The express request.
|
|
553
509
|
* @returns {String|undefined} The locale, or undefined if no locale was set.
|
|
554
510
|
*/
|
|
555
|
-
|
|
556
|
-
|
|
557
511
|
getLocale(req) {
|
|
558
512
|
const locale = (req.query || {})[pageParams.locale] || (req.body || {})[pageParams.locale] || (req.params || {})[pageParams.locale] || (req.headers || {})[pageParamHeaderPrefix + pageParams.locale];
|
|
559
513
|
return locale;
|
|
560
514
|
}
|
|
515
|
+
|
|
561
516
|
/**
|
|
562
517
|
* Creates a response with http redirect.
|
|
563
518
|
* @param {Object} req The express request.
|
|
@@ -565,28 +520,26 @@ class PagesRouter extends _PromiseRouter.default {
|
|
|
565
520
|
* @param {Object} params The query parameters to include.
|
|
566
521
|
* @returns {Object} The Promise Router response.
|
|
567
522
|
*/
|
|
568
|
-
|
|
569
|
-
|
|
570
523
|
async redirectResponse(url, params) {
|
|
571
524
|
// Remove any parameters with undefined value
|
|
572
525
|
params = Object.entries(params).reduce((m, p) => {
|
|
573
526
|
if (p[1] !== undefined) {
|
|
574
527
|
m[p[0]] = p[1];
|
|
575
528
|
}
|
|
576
|
-
|
|
577
529
|
return m;
|
|
578
|
-
}, {});
|
|
530
|
+
}, {});
|
|
579
531
|
|
|
532
|
+
// Compose URL with parameters in query
|
|
580
533
|
const location = new URL(url);
|
|
581
534
|
Object.entries(params).forEach(p => location.searchParams.set(p[0], p[1]));
|
|
582
|
-
const locationString = location.toString();
|
|
583
|
-
// of response, instead of having to parse the HTML content.
|
|
535
|
+
const locationString = location.toString();
|
|
584
536
|
|
|
537
|
+
// Add parameters to header to allow parsing for programmatic use
|
|
538
|
+
// of response, instead of having to parse the HTML content.
|
|
585
539
|
const headers = Object.entries(params).reduce((m, p) => {
|
|
586
540
|
if (p[1] !== undefined) {
|
|
587
541
|
m[`${pageParamHeaderPrefix}${p[0].toLowerCase()}`] = p[1];
|
|
588
542
|
}
|
|
589
|
-
|
|
590
543
|
return m;
|
|
591
544
|
}, {});
|
|
592
545
|
return {
|
|
@@ -595,11 +548,9 @@ class PagesRouter extends _PromiseRouter.default {
|
|
|
595
548
|
headers: headers
|
|
596
549
|
};
|
|
597
550
|
}
|
|
598
|
-
|
|
599
551
|
defaultPagePath(file) {
|
|
600
552
|
return _path.default.join(this.pagesPath, file);
|
|
601
553
|
}
|
|
602
|
-
|
|
603
554
|
composePageUrl(file, publicServerUrl, locale) {
|
|
604
555
|
let url = publicServerUrl;
|
|
605
556
|
url += url.endsWith('/') ? '' : '/';
|
|
@@ -608,20 +559,19 @@ class PagesRouter extends _PromiseRouter.default {
|
|
|
608
559
|
url += file;
|
|
609
560
|
return url;
|
|
610
561
|
}
|
|
611
|
-
|
|
612
562
|
notFound() {
|
|
613
563
|
return {
|
|
614
564
|
text: 'Not found.',
|
|
615
565
|
status: 404
|
|
616
566
|
};
|
|
617
567
|
}
|
|
618
|
-
|
|
619
568
|
invalidRequest() {
|
|
620
569
|
const error = new Error();
|
|
621
570
|
error.status = 403;
|
|
622
571
|
error.message = 'unauthorized';
|
|
623
572
|
throw error;
|
|
624
573
|
}
|
|
574
|
+
|
|
625
575
|
/**
|
|
626
576
|
* Sets the Parse Server configuration in the request object to make it
|
|
627
577
|
* easily accessible throughtout request processing.
|
|
@@ -629,18 +579,13 @@ class PagesRouter extends _PromiseRouter.default {
|
|
|
629
579
|
* @param {Boolean} failGracefully Is true if failing to set the config should
|
|
630
580
|
* not result in an invalid request response. Default is `false`.
|
|
631
581
|
*/
|
|
632
|
-
|
|
633
|
-
|
|
634
582
|
setConfig(req, failGracefully = false) {
|
|
635
583
|
req.config = _Config.default.get(req.params.appId || req.query.appId);
|
|
636
|
-
|
|
637
584
|
if (!req.config && !failGracefully) {
|
|
638
585
|
this.invalidRequest();
|
|
639
586
|
}
|
|
640
|
-
|
|
641
587
|
return Promise.resolve();
|
|
642
588
|
}
|
|
643
|
-
|
|
644
589
|
mountPagesRoutes() {
|
|
645
590
|
this.route('GET', `/${this.pagesEndpoint}/:appId/verify_email`, req => {
|
|
646
591
|
this.setConfig(req);
|
|
@@ -668,7 +613,6 @@ class PagesRouter extends _PromiseRouter.default {
|
|
|
668
613
|
return this.requestResetPassword(req);
|
|
669
614
|
});
|
|
670
615
|
}
|
|
671
|
-
|
|
672
616
|
mountCustomRoutes() {
|
|
673
617
|
for (const route of this.pagesConfig.customRoutes || []) {
|
|
674
618
|
this.route(route.method, `/${this.pagesEndpoint}/:appId/${route.path}`, req => {
|
|
@@ -677,13 +621,14 @@ class PagesRouter extends _PromiseRouter.default {
|
|
|
677
621
|
const {
|
|
678
622
|
file,
|
|
679
623
|
query = {}
|
|
680
|
-
} = (await route.handler(req)) || {};
|
|
624
|
+
} = (await route.handler(req)) || {};
|
|
681
625
|
|
|
626
|
+
// If route handler did not return a page send 404 response
|
|
682
627
|
if (!file) {
|
|
683
628
|
return this.notFound();
|
|
684
|
-
}
|
|
685
|
-
|
|
629
|
+
}
|
|
686
630
|
|
|
631
|
+
// Send page response
|
|
687
632
|
const page = new _Page.default({
|
|
688
633
|
id: file,
|
|
689
634
|
defaultFile: file
|
|
@@ -692,7 +637,6 @@ class PagesRouter extends _PromiseRouter.default {
|
|
|
692
637
|
});
|
|
693
638
|
}
|
|
694
639
|
}
|
|
695
|
-
|
|
696
640
|
mountStaticRoute() {
|
|
697
641
|
this.route('GET', `/${this.pagesEndpoint}/(*)?`, req => {
|
|
698
642
|
this.setConfig(req, true);
|
|
@@ -700,16 +644,12 @@ class PagesRouter extends _PromiseRouter.default {
|
|
|
700
644
|
return this.staticRoute(req);
|
|
701
645
|
});
|
|
702
646
|
}
|
|
703
|
-
|
|
704
647
|
expressRouter() {
|
|
705
648
|
const router = _express.default.Router();
|
|
706
|
-
|
|
707
649
|
router.use('/', super.expressRouter());
|
|
708
650
|
return router;
|
|
709
651
|
}
|
|
710
|
-
|
|
711
652
|
}
|
|
712
|
-
|
|
713
653
|
exports.PagesRouter = PagesRouter;
|
|
714
654
|
var _default = PagesRouter;
|
|
715
655
|
exports.default = _default;
|
|
@@ -719,4 +659,4 @@ module.exports = {
|
|
|
719
659
|
pageParams,
|
|
720
660
|
pages
|
|
721
661
|
};
|
|
722
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/Routers/PagesRouter.js"],"names":["pages","Object","freeze","passwordReset","Page","id","defaultFile","passwordResetSuccess","passwordResetLinkInvalid","emailVerificationSuccess","emailVerificationSendFail","emailVerificationSendSuccess","emailVerificationLinkInvalid","emailVerificationLinkExpired","pageParams","appName","appId","token","username","error","locale","publicServerUrl","pageParamHeaderPrefix","errors","jsonFailedFileLoading","fileOutsideAllowedScope","PagesRouter","PromiseRouter","constructor","pagesConfig","pagesEndpoint","pagesPath","path","resolve","__dirname","loadJsonResource","mountPagesRoutes","mountCustomRoutes","mountStaticRoute","verifyEmail","req","config","rawToken","query","toString","invalidRequest","goToPage","userController","then","params","resendVerificationEmail","body","publicServerURL","requestResetPassword","checkResetTokenValidity","applicationId","resetPassword","new_password","xhr","Parse","Error","USERNAME_MISSING","OTHER_CAUSE","PASSWORD_MISSING","updatePassword","Promise","success","err","result","status","response","page","responseType","redirect","forceRedirect","undefined","method","defaultParams","getDefaultParams","values","includes","notFound","assign","getLocale","defaultPath","defaultPagePath","defaultUrl","composePageUrl","customUrl","customUrls","Utils","isPath","redirectResponse","placeholders","enableLocalization","localizationJsonPath","getJsonPlaceholders","getLocalizedPath","subdir","pageResponse","staticRoute","relativePath","absolutePath","endsWith","fileResponse","getJsonTranslation","jsonParameters","localizationFallbackLocale","language","split","resource","translation","JSON","stringify","mustache","render","parse","data","readFile","e","configPlaceholders","prototype","call","allPlaceholders","paramsAndPlaceholders","headers","entries","reduce","m","p","toLowerCase","text","filePath","normalizedPath","normalize","startsWith","fs","json","require","url","location","URL","forEach","searchParams","set","locationString","file","join","message","setConfig","failGracefully","Config","get","route","customRoutes","handler","expressRouter","router","express","Router","use","module","exports"],"mappings":"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","sourcesContent":["import PromiseRouter from '../PromiseRouter';\nimport Config from '../Config';\nimport express from 'express';\nimport path from 'path';\nimport { promises as fs } from 'fs';\nimport { Parse } from 'parse/node';\nimport Utils from '../Utils';\nimport mustache from 'mustache';\nimport Page from '../Page';\n\n// All pages with custom page key for reference and file name\nconst pages = Object.freeze({\n  passwordReset: new Page({ id: 'passwordReset', defaultFile: 'password_reset.html' }),\n  passwordResetSuccess: new Page({\n    id: 'passwordResetSuccess',\n    defaultFile: 'password_reset_success.html',\n  }),\n  passwordResetLinkInvalid: new Page({\n    id: 'passwordResetLinkInvalid',\n    defaultFile: 'password_reset_link_invalid.html',\n  }),\n  emailVerificationSuccess: new Page({\n    id: 'emailVerificationSuccess',\n    defaultFile: 'email_verification_success.html',\n  }),\n  emailVerificationSendFail: new Page({\n    id: 'emailVerificationSendFail',\n    defaultFile: 'email_verification_send_fail.html',\n  }),\n  emailVerificationSendSuccess: new Page({\n    id: 'emailVerificationSendSuccess',\n    defaultFile: 'email_verification_send_success.html',\n  }),\n  emailVerificationLinkInvalid: new Page({\n    id: 'emailVerificationLinkInvalid',\n    defaultFile: 'email_verification_link_invalid.html',\n  }),\n  emailVerificationLinkExpired: new Page({\n    id: 'emailVerificationLinkExpired',\n    defaultFile: 'email_verification_link_expired.html',\n  }),\n});\n\n// All page parameters for reference to be used as template placeholders or query params\nconst pageParams = Object.freeze({\n  appName: 'appName',\n  appId: 'appId',\n  token: 'token',\n  username: 'username',\n  error: 'error',\n  locale: 'locale',\n  publicServerUrl: 'publicServerUrl',\n});\n\n// The header prefix to add page params as response headers\nconst pageParamHeaderPrefix = 'x-parse-page-param-';\n\n// The errors being thrown\nconst errors = Object.freeze({\n  jsonFailedFileLoading: 'failed to load JSON file',\n  fileOutsideAllowedScope: 'not allowed to read file outside of pages directory',\n});\n\nexport class PagesRouter extends PromiseRouter {\n  /**\n   * Constructs a PagesRouter.\n   * @param {Object} pages The pages options from the Parse Server configuration.\n   */\n  constructor(pages = {}) {\n    super();\n\n    // Set instance properties\n    this.pagesConfig = pages;\n    this.pagesEndpoint = pages.pagesEndpoint ? pages.pagesEndpoint : 'apps';\n    this.pagesPath = pages.pagesPath\n      ? path.resolve('./', pages.pagesPath)\n      : path.resolve(__dirname, '../../public');\n    this.loadJsonResource();\n    this.mountPagesRoutes();\n    this.mountCustomRoutes();\n    this.mountStaticRoute();\n  }\n\n  verifyEmail(req) {\n    const config = req.config;\n    const { username, token: rawToken } = req.query;\n    const token = rawToken && typeof rawToken !== 'string' ? rawToken.toString() : rawToken;\n\n    if (!config) {\n      this.invalidRequest();\n    }\n\n    if (!token || !username) {\n      return this.goToPage(req, pages.emailVerificationLinkInvalid);\n    }\n\n    const userController = config.userController;\n    return userController.verifyEmail(username, token).then(\n      () => {\n        const params = {\n          [pageParams.username]: username,\n        };\n        return this.goToPage(req, pages.emailVerificationSuccess, params);\n      },\n      () => {\n        const params = {\n          [pageParams.username]: username,\n        };\n        return this.goToPage(req, pages.emailVerificationLinkExpired, params);\n      }\n    );\n  }\n\n  resendVerificationEmail(req) {\n    const config = req.config;\n    const username = req.body.username;\n\n    if (!config) {\n      this.invalidRequest();\n    }\n\n    if (!username) {\n      return this.goToPage(req, pages.emailVerificationLinkInvalid);\n    }\n\n    const userController = config.userController;\n\n    return userController.resendVerificationEmail(username).then(\n      () => {\n        return this.goToPage(req, pages.emailVerificationSendSuccess);\n      },\n      () => {\n        return this.goToPage(req, pages.emailVerificationSendFail);\n      }\n    );\n  }\n\n  passwordReset(req) {\n    const config = req.config;\n    const params = {\n      [pageParams.appId]: req.params.appId,\n      [pageParams.appName]: config.appName,\n      [pageParams.token]: req.query.token,\n      [pageParams.username]: req.query.username,\n      [pageParams.publicServerUrl]: config.publicServerURL,\n    };\n    return this.goToPage(req, pages.passwordReset, params);\n  }\n\n  requestResetPassword(req) {\n    const config = req.config;\n\n    if (!config) {\n      this.invalidRequest();\n    }\n\n    const { username, token: rawToken } = req.query;\n    const token = rawToken && typeof rawToken !== 'string' ? rawToken.toString() : rawToken;\n\n    if (!username || !token) {\n      return this.goToPage(req, pages.passwordResetLinkInvalid);\n    }\n\n    return config.userController.checkResetTokenValidity(username, token).then(\n      () => {\n        const params = {\n          [pageParams.token]: token,\n          [pageParams.username]: username,\n          [pageParams.appId]: config.applicationId,\n          [pageParams.appName]: config.appName,\n        };\n        return this.goToPage(req, pages.passwordReset, params);\n      },\n      () => {\n        const params = {\n          [pageParams.username]: username,\n        };\n        return this.goToPage(req, pages.passwordResetLinkInvalid, params);\n      }\n    );\n  }\n\n  resetPassword(req) {\n    const config = req.config;\n\n    if (!config) {\n      this.invalidRequest();\n    }\n\n    const { username, new_password, token: rawToken } = req.body;\n    const token = rawToken && typeof rawToken !== 'string' ? rawToken.toString() : rawToken;\n\n    if ((!username || !token || !new_password) && req.xhr === false) {\n      return this.goToPage(req, pages.passwordResetLinkInvalid);\n    }\n\n    if (!username) {\n      throw new Parse.Error(Parse.Error.USERNAME_MISSING, 'Missing username');\n    }\n\n    if (!token) {\n      throw new Parse.Error(Parse.Error.OTHER_CAUSE, 'Missing token');\n    }\n\n    if (!new_password) {\n      throw new Parse.Error(Parse.Error.PASSWORD_MISSING, 'Missing password');\n    }\n\n    return config.userController\n      .updatePassword(username, token, new_password)\n      .then(\n        () => {\n          return Promise.resolve({\n            success: true,\n          });\n        },\n        err => {\n          return Promise.resolve({\n            success: false,\n            err,\n          });\n        }\n      )\n      .then(result => {\n        if (req.xhr) {\n          if (result.success) {\n            return Promise.resolve({\n              status: 200,\n              response: 'Password successfully reset',\n            });\n          }\n          if (result.err) {\n            throw new Parse.Error(Parse.Error.OTHER_CAUSE, `${result.err}`);\n          }\n        }\n\n        const query = result.success\n          ? {\n            [pageParams.username]: username,\n          }\n          : {\n            [pageParams.username]: username,\n            [pageParams.token]: token,\n            [pageParams.appId]: config.applicationId,\n            [pageParams.error]: result.err,\n            [pageParams.appName]: config.appName,\n          };\n        const page = result.success ? pages.passwordResetSuccess : pages.passwordReset;\n\n        return this.goToPage(req, page, query, false);\n      });\n  }\n\n  /**\n   * Returns page content if the page is a local file or returns a\n   * redirect to a custom page.\n   * @param {Object} req The express request.\n   * @param {Page} page The page to go to.\n   * @param {Object} [params={}] The query parameters to attach to the URL in case of\n   * HTTP redirect responses for POST requests, or the placeholders to fill into\n   * the response content in case of HTTP content responses for GET requests.\n   * @param {Boolean} [responseType] Is true if a redirect response should be forced,\n   * false if a content response should be forced, undefined if the response type\n   * should depend on the request type by default:\n   * - GET request -> content response\n   * - POST request -> redirect response (PRG pattern)\n   * @returns {Promise<Object>} The PromiseRouter response.\n   */\n  goToPage(req, page, params = {}, responseType) {\n    const config = req.config;\n\n    // Determine redirect either by force, response setting or request method\n    const redirect = config.pages.forceRedirect\n      ? true\n      : responseType !== undefined\n        ? responseType\n        : req.method == 'POST';\n\n    // Include default parameters\n    const defaultParams = this.getDefaultParams(config);\n    if (Object.values(defaultParams).includes(undefined)) {\n      return this.notFound();\n    }\n    params = Object.assign(params, defaultParams);\n\n    // Add locale to params to ensure it is passed on with every request;\n    // that means, once a locale is set, it is passed on to any follow-up page,\n    // e.g. request_password_reset -> password_reset -> password_reset_success\n    const locale = this.getLocale(req);\n    params[pageParams.locale] = locale;\n\n    // Compose paths and URLs\n    const defaultFile = page.defaultFile;\n    const defaultPath = this.defaultPagePath(defaultFile);\n    const defaultUrl = this.composePageUrl(defaultFile, config.publicServerURL);\n\n    // If custom URL is set redirect to it without localization\n    const customUrl = config.pages.customUrls[page.id];\n    if (customUrl && !Utils.isPath(customUrl)) {\n      return this.redirectResponse(customUrl, params);\n    }\n\n    // Get JSON placeholders\n    let placeholders = {};\n    if (config.pages.enableLocalization && config.pages.localizationJsonPath) {\n      placeholders = this.getJsonPlaceholders(locale, params);\n    }\n\n    // Send response\n    if (config.pages.enableLocalization && locale) {\n      return Utils.getLocalizedPath(defaultPath, locale).then(({ path, subdir }) =>\n        redirect\n          ? this.redirectResponse(\n            this.composePageUrl(defaultFile, config.publicServerURL, subdir),\n            params\n          )\n          : this.pageResponse(path, params, placeholders)\n      );\n    } else {\n      return redirect\n        ? this.redirectResponse(defaultUrl, params)\n        : this.pageResponse(defaultPath, params, placeholders);\n    }\n  }\n\n  /**\n   * Serves a request to a static resource and localizes the resource if it\n   * is a HTML file.\n   * @param {Object} req The request object.\n   * @returns {Promise<Object>} The response.\n   */\n  staticRoute(req) {\n    // Get requested path\n    const relativePath = req.params[0];\n\n    // Resolve requested path to absolute path\n    const absolutePath = path.resolve(this.pagesPath, relativePath);\n\n    // If the requested file is not a HTML file send its raw content\n    if (!absolutePath || !absolutePath.endsWith('.html')) {\n      return this.fileResponse(absolutePath);\n    }\n\n    // Get parameters\n    const params = this.getDefaultParams(req.config);\n    const locale = this.getLocale(req);\n    if (locale) {\n      params.locale = locale;\n    }\n\n    // Get JSON placeholders\n    const placeholders = this.getJsonPlaceholders(locale, params);\n\n    return this.pageResponse(absolutePath, params, placeholders);\n  }\n\n  /**\n   * Returns a translation from the JSON resource for a given locale. The JSON\n   * resource is parsed according to i18next syntax.\n   *\n   * Example JSON content:\n   * ```js\n   *  {\n   *    \"en\": {               // resource for language `en` (English)\n   *      \"translation\": {\n   *        \"greeting\": \"Hello!\"\n   *      }\n   *    },\n   *    \"de\": {               // resource for language `de` (German)\n   *      \"translation\": {\n   *        \"greeting\": \"Hallo!\"\n   *      }\n   *    }\n   *    \"de-CH\": {            // resource for locale `de-CH` (Swiss German)\n   *      \"translation\": {\n   *        \"greeting\": \"Grüezi!\"\n   *      }\n   *    }\n   *  }\n   * ```\n   * @param {String} locale The locale to translate to.\n   * @returns {Object} The translation or an empty object if no matching\n   * translation was found.\n   */\n  getJsonTranslation(locale) {\n    // If there is no JSON resource\n    if (this.jsonParameters === undefined) {\n      return {};\n    }\n\n    // If locale is not set use the fallback locale\n    locale = locale || this.pagesConfig.localizationFallbackLocale;\n\n    // Get matching translation by locale, language or fallback locale\n    const language = locale.split('-')[0];\n    const resource =\n      this.jsonParameters[locale] ||\n      this.jsonParameters[language] ||\n      this.jsonParameters[this.pagesConfig.localizationFallbackLocale] ||\n      {};\n    const translation = resource.translation || {};\n    return translation;\n  }\n\n  /**\n   * Returns a translation from the JSON resource for a given locale with\n   * placeholders filled in by given parameters.\n   * @param {String} locale The locale to translate to.\n   * @param {Object} params The parameters to fill into any placeholders\n   * within the translations.\n   * @returns {Object} The translation or an empty object if no matching\n   * translation was found.\n   */\n  getJsonPlaceholders(locale, params = {}) {\n    // If localization is disabled or there is no JSON resource\n    if (!this.pagesConfig.enableLocalization || !this.pagesConfig.localizationJsonPath) {\n      return {};\n    }\n\n    // Get JSON placeholders\n    let placeholders = this.getJsonTranslation(locale);\n\n    // Fill in any placeholders in the translation; this allows a translation\n    // to contain default placeholders like {{appName}} which are filled here\n    placeholders = JSON.stringify(placeholders);\n    placeholders = mustache.render(placeholders, params);\n    placeholders = JSON.parse(placeholders);\n\n    return placeholders;\n  }\n\n  /**\n   * Creates a response with file content.\n   * @param {String} path The path of the file to return.\n   * @param {Object} [params={}] The parameters to be included in the response\n   * header. These will also be used to fill placeholders.\n   * @param {Object} [placeholders={}] The placeholders to fill in the content.\n   * These will not be included in the response header.\n   * @returns {Object} The Promise Router response.\n   */\n  async pageResponse(path, params = {}, placeholders = {}) {\n    // Get file content\n    let data;\n    try {\n      data = await this.readFile(path);\n    } catch (e) {\n      return this.notFound();\n    }\n\n    // Get config placeholders; can be an object, a function or an async function\n    let configPlaceholders =\n      typeof this.pagesConfig.placeholders === 'function'\n        ? this.pagesConfig.placeholders(params)\n        : Object.prototype.toString.call(this.pagesConfig.placeholders) === '[object Object]'\n          ? this.pagesConfig.placeholders\n          : {};\n    if (configPlaceholders instanceof Promise) {\n      configPlaceholders = await configPlaceholders;\n    }\n\n    // Fill placeholders\n    const allPlaceholders = Object.assign({}, configPlaceholders, placeholders);\n    const paramsAndPlaceholders = Object.assign({}, params, allPlaceholders);\n    data = mustache.render(data, paramsAndPlaceholders);\n\n    // Add placeholders in header to allow parsing for programmatic use\n    // of response, instead of having to parse the HTML content.\n    const headers = Object.entries(params).reduce((m, p) => {\n      if (p[1] !== undefined) {\n        m[`${pageParamHeaderPrefix}${p[0].toLowerCase()}`] = p[1];\n      }\n      return m;\n    }, {});\n\n    return { text: data, headers: headers };\n  }\n\n  /**\n   * Creates a response with file content.\n   * @param {String} path The path of the file to return.\n   * @returns {Object} The PromiseRouter response.\n   */\n  async fileResponse(path) {\n    // Get file content\n    let data;\n    try {\n      data = await this.readFile(path);\n    } catch (e) {\n      return this.notFound();\n    }\n\n    return { text: data };\n  }\n\n  /**\n   * Reads and returns the content of a file at a given path. File reading to\n   * serve content on the static route is only allowed from the pages\n   * directory on downwards.\n   * -----------------------------------------------------------------------\n   * **WARNING:** All file reads in the PagesRouter must be executed by this\n   * wrapper because it also detects and prevents common exploits.\n   * -----------------------------------------------------------------------\n   * @param {String} filePath The path to the file to read.\n   * @returns {Promise<String>} The file content.\n   */\n  async readFile(filePath) {\n    // Normalize path to prevent it from containing any directory changing\n    // UNIX patterns which could expose the whole file system, e.g.\n    // `http://example.com/parse/apps/../file.txt` requests a file outside\n    // of the pages directory scope.\n    const normalizedPath = path.normalize(filePath);\n\n    // Abort if the path is outside of the path directory scope\n    if (!normalizedPath.startsWith(this.pagesPath)) {\n      throw errors.fileOutsideAllowedScope;\n    }\n\n    return await fs.readFile(normalizedPath, 'utf-8');\n  }\n\n  /**\n   * Loads a language resource JSON file that is used for translations.\n   */\n  loadJsonResource() {\n    if (this.pagesConfig.localizationJsonPath === undefined) {\n      return;\n    }\n    try {\n      const json = require(path.resolve('./', this.pagesConfig.localizationJsonPath));\n      this.jsonParameters = json;\n    } catch (e) {\n      throw errors.jsonFailedFileLoading;\n    }\n  }\n\n  /**\n   * Extracts and returns the page default parameters from the Parse Server\n   * configuration. These parameters are made accessible in every page served\n   * by this router.\n   * @param {Object} config The Parse Server configuration.\n   * @returns {Object} The default parameters.\n   */\n  getDefaultParams(config) {\n    return config\n      ? {\n        [pageParams.appId]: config.appId,\n        [pageParams.appName]: config.appName,\n        [pageParams.publicServerUrl]: config.publicServerURL,\n      }\n      : {};\n  }\n\n  /**\n   * Extracts and returns the locale from an express request.\n   * @param {Object} req The express request.\n   * @returns {String|undefined} The locale, or undefined if no locale was set.\n   */\n  getLocale(req) {\n    const locale =\n      (req.query || {})[pageParams.locale] ||\n      (req.body || {})[pageParams.locale] ||\n      (req.params || {})[pageParams.locale] ||\n      (req.headers || {})[pageParamHeaderPrefix + pageParams.locale];\n    return locale;\n  }\n\n  /**\n   * Creates a response with http redirect.\n   * @param {Object} req The express request.\n   * @param {String} path The path of the file to return.\n   * @param {Object} params The query parameters to include.\n   * @returns {Object} The Promise Router response.\n   */\n  async redirectResponse(url, params) {\n    // Remove any parameters with undefined value\n    params = Object.entries(params).reduce((m, p) => {\n      if (p[1] !== undefined) {\n        m[p[0]] = p[1];\n      }\n      return m;\n    }, {});\n\n    // Compose URL with parameters in query\n    const location = new URL(url);\n    Object.entries(params).forEach(p => location.searchParams.set(p[0], p[1]));\n    const locationString = location.toString();\n\n    // Add parameters to header to allow parsing for programmatic use\n    // of response, instead of having to parse the HTML content.\n    const headers = Object.entries(params).reduce((m, p) => {\n      if (p[1] !== undefined) {\n        m[`${pageParamHeaderPrefix}${p[0].toLowerCase()}`] = p[1];\n      }\n      return m;\n    }, {});\n\n    return {\n      status: 303,\n      location: locationString,\n      headers: headers,\n    };\n  }\n\n  defaultPagePath(file) {\n    return path.join(this.pagesPath, file);\n  }\n\n  composePageUrl(file, publicServerUrl, locale) {\n    let url = publicServerUrl;\n    url += url.endsWith('/') ? '' : '/';\n    url += this.pagesEndpoint + '/';\n    url += locale === undefined ? '' : locale + '/';\n    url += file;\n    return url;\n  }\n\n  notFound() {\n    return {\n      text: 'Not found.',\n      status: 404,\n    };\n  }\n\n  invalidRequest() {\n    const error = new Error();\n    error.status = 403;\n    error.message = 'unauthorized';\n    throw error;\n  }\n\n  /**\n   * Sets the Parse Server configuration in the request object to make it\n   * easily accessible throughtout request processing.\n   * @param {Object} req The request.\n   * @param {Boolean} failGracefully Is true if failing to set the config should\n   * not result in an invalid request response. Default is `false`.\n   */\n  setConfig(req, failGracefully = false) {\n    req.config = Config.get(req.params.appId || req.query.appId);\n    if (!req.config && !failGracefully) {\n      this.invalidRequest();\n    }\n    return Promise.resolve();\n  }\n\n  mountPagesRoutes() {\n    this.route(\n      'GET',\n      `/${this.pagesEndpoint}/:appId/verify_email`,\n      req => {\n        this.setConfig(req);\n      },\n      req => {\n        return this.verifyEmail(req);\n      }\n    );\n\n    this.route(\n      'POST',\n      `/${this.pagesEndpoint}/:appId/resend_verification_email`,\n      req => {\n        this.setConfig(req);\n      },\n      req => {\n        return this.resendVerificationEmail(req);\n      }\n    );\n\n    this.route(\n      'GET',\n      `/${this.pagesEndpoint}/choose_password`,\n      req => {\n        this.setConfig(req);\n      },\n      req => {\n        return this.passwordReset(req);\n      }\n    );\n\n    this.route(\n      'POST',\n      `/${this.pagesEndpoint}/:appId/request_password_reset`,\n      req => {\n        this.setConfig(req);\n      },\n      req => {\n        return this.resetPassword(req);\n      }\n    );\n\n    this.route(\n      'GET',\n      `/${this.pagesEndpoint}/:appId/request_password_reset`,\n      req => {\n        this.setConfig(req);\n      },\n      req => {\n        return this.requestResetPassword(req);\n      }\n    );\n  }\n\n  mountCustomRoutes() {\n    for (const route of this.pagesConfig.customRoutes || []) {\n      this.route(\n        route.method,\n        `/${this.pagesEndpoint}/:appId/${route.path}`,\n        req => {\n          this.setConfig(req);\n        },\n        async req => {\n          const { file, query = {} } = (await route.handler(req)) || {};\n\n          // If route handler did not return a page send 404 response\n          if (!file) {\n            return this.notFound();\n          }\n\n          // Send page response\n          const page = new Page({ id: file, defaultFile: file });\n          return this.goToPage(req, page, query, false);\n        }\n      );\n    }\n  }\n\n  mountStaticRoute() {\n    this.route(\n      'GET',\n      `/${this.pagesEndpoint}/(*)?`,\n      req => {\n        this.setConfig(req, true);\n      },\n      req => {\n        return this.staticRoute(req);\n      }\n    );\n  }\n\n  expressRouter() {\n    const router = express.Router();\n    router.use('/', super.expressRouter());\n    return router;\n  }\n}\n\nexport default PagesRouter;\nmodule.exports = {\n  PagesRouter,\n  pageParamHeaderPrefix,\n  pageParams,\n  pages,\n};\n"]}
|
|
662
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["pages","Object","freeze","passwordReset","Page","id","defaultFile","passwordResetSuccess","passwordResetLinkInvalid","emailVerificationSuccess","emailVerificationSendFail","emailVerificationSendSuccess","emailVerificationLinkInvalid","emailVerificationLinkExpired","pageParams","appName","appId","token","username","error","locale","publicServerUrl","pageParamHeaderPrefix","errors","jsonFailedFileLoading","fileOutsideAllowedScope","PagesRouter","PromiseRouter","constructor","pagesConfig","pagesEndpoint","pagesPath","path","resolve","__dirname","loadJsonResource","mountPagesRoutes","mountCustomRoutes","mountStaticRoute","verifyEmail","req","config","rawToken","query","toString","invalidRequest","goToPage","userController","then","params","resendVerificationEmail","body","publicServerURL","requestResetPassword","checkResetTokenValidity","applicationId","resetPassword","new_password","xhr","Parse","Error","USERNAME_MISSING","OTHER_CAUSE","PASSWORD_MISSING","updatePassword","Promise","success","err","result","status","response","page","responseType","redirect","forceRedirect","undefined","method","defaultParams","getDefaultParams","values","includes","notFound","assign","getLocale","defaultPath","defaultPagePath","defaultUrl","composePageUrl","customUrl","customUrls","Utils","isPath","redirectResponse","placeholders","enableLocalization","localizationJsonPath","getJsonPlaceholders","getLocalizedPath","subdir","pageResponse","staticRoute","relativePath","absolutePath","endsWith","fileResponse","getJsonTranslation","jsonParameters","localizationFallbackLocale","language","split","resource","translation","JSON","stringify","mustache","render","parse","data","readFile","e","configPlaceholders","prototype","call","allPlaceholders","paramsAndPlaceholders","headers","entries","reduce","m","p","toLowerCase","text","filePath","normalizedPath","normalize","startsWith","fs","json","require","url","location","URL","forEach","searchParams","set","locationString","file","join","message","setConfig","failGracefully","Config","get","route","customRoutes","handler","expressRouter","router","express","Router","use","module","exports"],"sources":["../../src/Routers/PagesRouter.js"],"sourcesContent":["import PromiseRouter from '../PromiseRouter';\nimport Config from '../Config';\nimport express from 'express';\nimport path from 'path';\nimport { promises as fs } from 'fs';\nimport { Parse } from 'parse/node';\nimport Utils from '../Utils';\nimport mustache from 'mustache';\nimport Page from '../Page';\n\n// All pages with custom page key for reference and file name\nconst pages = Object.freeze({\n  passwordReset: new Page({ id: 'passwordReset', defaultFile: 'password_reset.html' }),\n  passwordResetSuccess: new Page({\n    id: 'passwordResetSuccess',\n    defaultFile: 'password_reset_success.html',\n  }),\n  passwordResetLinkInvalid: new Page({\n    id: 'passwordResetLinkInvalid',\n    defaultFile: 'password_reset_link_invalid.html',\n  }),\n  emailVerificationSuccess: new Page({\n    id: 'emailVerificationSuccess',\n    defaultFile: 'email_verification_success.html',\n  }),\n  emailVerificationSendFail: new Page({\n    id: 'emailVerificationSendFail',\n    defaultFile: 'email_verification_send_fail.html',\n  }),\n  emailVerificationSendSuccess: new Page({\n    id: 'emailVerificationSendSuccess',\n    defaultFile: 'email_verification_send_success.html',\n  }),\n  emailVerificationLinkInvalid: new Page({\n    id: 'emailVerificationLinkInvalid',\n    defaultFile: 'email_verification_link_invalid.html',\n  }),\n  emailVerificationLinkExpired: new Page({\n    id: 'emailVerificationLinkExpired',\n    defaultFile: 'email_verification_link_expired.html',\n  }),\n});\n\n// All page parameters for reference to be used as template placeholders or query params\nconst pageParams = Object.freeze({\n  appName: 'appName',\n  appId: 'appId',\n  token: 'token',\n  username: 'username',\n  error: 'error',\n  locale: 'locale',\n  publicServerUrl: 'publicServerUrl',\n});\n\n// The header prefix to add page params as response headers\nconst pageParamHeaderPrefix = 'x-parse-page-param-';\n\n// The errors being thrown\nconst errors = Object.freeze({\n  jsonFailedFileLoading: 'failed to load JSON file',\n  fileOutsideAllowedScope: 'not allowed to read file outside of pages directory',\n});\n\nexport class PagesRouter extends PromiseRouter {\n  /**\n   * Constructs a PagesRouter.\n   * @param {Object} pages The pages options from the Parse Server configuration.\n   */\n  constructor(pages = {}) {\n    super();\n\n    // Set instance properties\n    this.pagesConfig = pages;\n    this.pagesEndpoint = pages.pagesEndpoint ? pages.pagesEndpoint : 'apps';\n    this.pagesPath = pages.pagesPath\n      ? path.resolve('./', pages.pagesPath)\n      : path.resolve(__dirname, '../../public');\n    this.loadJsonResource();\n    this.mountPagesRoutes();\n    this.mountCustomRoutes();\n    this.mountStaticRoute();\n  }\n\n  verifyEmail(req) {\n    const config = req.config;\n    const { username, token: rawToken } = req.query;\n    const token = rawToken && typeof rawToken !== 'string' ? rawToken.toString() : rawToken;\n\n    if (!config) {\n      this.invalidRequest();\n    }\n\n    if (!token || !username) {\n      return this.goToPage(req, pages.emailVerificationLinkInvalid);\n    }\n\n    const userController = config.userController;\n    return userController.verifyEmail(username, token).then(\n      () => {\n        const params = {\n          [pageParams.username]: username,\n        };\n        return this.goToPage(req, pages.emailVerificationSuccess, params);\n      },\n      () => {\n        const params = {\n          [pageParams.username]: username,\n        };\n        return this.goToPage(req, pages.emailVerificationLinkExpired, params);\n      }\n    );\n  }\n\n  resendVerificationEmail(req) {\n    const config = req.config;\n    const username = req.body.username;\n\n    if (!config) {\n      this.invalidRequest();\n    }\n\n    if (!username) {\n      return this.goToPage(req, pages.emailVerificationLinkInvalid);\n    }\n\n    const userController = config.userController;\n\n    return userController.resendVerificationEmail(username).then(\n      () => {\n        return this.goToPage(req, pages.emailVerificationSendSuccess);\n      },\n      () => {\n        return this.goToPage(req, pages.emailVerificationSendFail);\n      }\n    );\n  }\n\n  passwordReset(req) {\n    const config = req.config;\n    const params = {\n      [pageParams.appId]: req.params.appId,\n      [pageParams.appName]: config.appName,\n      [pageParams.token]: req.query.token,\n      [pageParams.username]: req.query.username,\n      [pageParams.publicServerUrl]: config.publicServerURL,\n    };\n    return this.goToPage(req, pages.passwordReset, params);\n  }\n\n  requestResetPassword(req) {\n    const config = req.config;\n\n    if (!config) {\n      this.invalidRequest();\n    }\n\n    const { username, token: rawToken } = req.query;\n    const token = rawToken && typeof rawToken !== 'string' ? rawToken.toString() : rawToken;\n\n    if (!username || !token) {\n      return this.goToPage(req, pages.passwordResetLinkInvalid);\n    }\n\n    return config.userController.checkResetTokenValidity(username, token).then(\n      () => {\n        const params = {\n          [pageParams.token]: token,\n          [pageParams.username]: username,\n          [pageParams.appId]: config.applicationId,\n          [pageParams.appName]: config.appName,\n        };\n        return this.goToPage(req, pages.passwordReset, params);\n      },\n      () => {\n        const params = {\n          [pageParams.username]: username,\n        };\n        return this.goToPage(req, pages.passwordResetLinkInvalid, params);\n      }\n    );\n  }\n\n  resetPassword(req) {\n    const config = req.config;\n\n    if (!config) {\n      this.invalidRequest();\n    }\n\n    const { username, new_password, token: rawToken } = req.body;\n    const token = rawToken && typeof rawToken !== 'string' ? rawToken.toString() : rawToken;\n\n    if ((!username || !token || !new_password) && req.xhr === false) {\n      return this.goToPage(req, pages.passwordResetLinkInvalid);\n    }\n\n    if (!username) {\n      throw new Parse.Error(Parse.Error.USERNAME_MISSING, 'Missing username');\n    }\n\n    if (!token) {\n      throw new Parse.Error(Parse.Error.OTHER_CAUSE, 'Missing token');\n    }\n\n    if (!new_password) {\n      throw new Parse.Error(Parse.Error.PASSWORD_MISSING, 'Missing password');\n    }\n\n    return config.userController\n      .updatePassword(username, token, new_password)\n      .then(\n        () => {\n          return Promise.resolve({\n            success: true,\n          });\n        },\n        err => {\n          return Promise.resolve({\n            success: false,\n            err,\n          });\n        }\n      )\n      .then(result => {\n        if (req.xhr) {\n          if (result.success) {\n            return Promise.resolve({\n              status: 200,\n              response: 'Password successfully reset',\n            });\n          }\n          if (result.err) {\n            throw new Parse.Error(Parse.Error.OTHER_CAUSE, `${result.err}`);\n          }\n        }\n\n        const query = result.success\n          ? {\n            [pageParams.username]: username,\n          }\n          : {\n            [pageParams.username]: username,\n            [pageParams.token]: token,\n            [pageParams.appId]: config.applicationId,\n            [pageParams.error]: result.err,\n            [pageParams.appName]: config.appName,\n          };\n        const page = result.success ? pages.passwordResetSuccess : pages.passwordReset;\n\n        return this.goToPage(req, page, query, false);\n      });\n  }\n\n  /**\n   * Returns page content if the page is a local file or returns a\n   * redirect to a custom page.\n   * @param {Object} req The express request.\n   * @param {Page} page The page to go to.\n   * @param {Object} [params={}] The query parameters to attach to the URL in case of\n   * HTTP redirect responses for POST requests, or the placeholders to fill into\n   * the response content in case of HTTP content responses for GET requests.\n   * @param {Boolean} [responseType] Is true if a redirect response should be forced,\n   * false if a content response should be forced, undefined if the response type\n   * should depend on the request type by default:\n   * - GET request -> content response\n   * - POST request -> redirect response (PRG pattern)\n   * @returns {Promise<Object>} The PromiseRouter response.\n   */\n  goToPage(req, page, params = {}, responseType) {\n    const config = req.config;\n\n    // Determine redirect either by force, response setting or request method\n    const redirect = config.pages.forceRedirect\n      ? true\n      : responseType !== undefined\n        ? responseType\n        : req.method == 'POST';\n\n    // Include default parameters\n    const defaultParams = this.getDefaultParams(config);\n    if (Object.values(defaultParams).includes(undefined)) {\n      return this.notFound();\n    }\n    params = Object.assign(params, defaultParams);\n\n    // Add locale to params to ensure it is passed on with every request;\n    // that means, once a locale is set, it is passed on to any follow-up page,\n    // e.g. request_password_reset -> password_reset -> password_reset_success\n    const locale = this.getLocale(req);\n    params[pageParams.locale] = locale;\n\n    // Compose paths and URLs\n    const defaultFile = page.defaultFile;\n    const defaultPath = this.defaultPagePath(defaultFile);\n    const defaultUrl = this.composePageUrl(defaultFile, config.publicServerURL);\n\n    // If custom URL is set redirect to it without localization\n    const customUrl = config.pages.customUrls[page.id];\n    if (customUrl && !Utils.isPath(customUrl)) {\n      return this.redirectResponse(customUrl, params);\n    }\n\n    // Get JSON placeholders\n    let placeholders = {};\n    if (config.pages.enableLocalization && config.pages.localizationJsonPath) {\n      placeholders = this.getJsonPlaceholders(locale, params);\n    }\n\n    // Send response\n    if (config.pages.enableLocalization && locale) {\n      return Utils.getLocalizedPath(defaultPath, locale).then(({ path, subdir }) =>\n        redirect\n          ? this.redirectResponse(\n            this.composePageUrl(defaultFile, config.publicServerURL, subdir),\n            params\n          )\n          : this.pageResponse(path, params, placeholders)\n      );\n    } else {\n      return redirect\n        ? this.redirectResponse(defaultUrl, params)\n        : this.pageResponse(defaultPath, params, placeholders);\n    }\n  }\n\n  /**\n   * Serves a request to a static resource and localizes the resource if it\n   * is a HTML file.\n   * @param {Object} req The request object.\n   * @returns {Promise<Object>} The response.\n   */\n  staticRoute(req) {\n    // Get requested path\n    const relativePath = req.params[0];\n\n    // Resolve requested path to absolute path\n    const absolutePath = path.resolve(this.pagesPath, relativePath);\n\n    // If the requested file is not a HTML file send its raw content\n    if (!absolutePath || !absolutePath.endsWith('.html')) {\n      return this.fileResponse(absolutePath);\n    }\n\n    // Get parameters\n    const params = this.getDefaultParams(req.config);\n    const locale = this.getLocale(req);\n    if (locale) {\n      params.locale = locale;\n    }\n\n    // Get JSON placeholders\n    const placeholders = this.getJsonPlaceholders(locale, params);\n\n    return this.pageResponse(absolutePath, params, placeholders);\n  }\n\n  /**\n   * Returns a translation from the JSON resource for a given locale. The JSON\n   * resource is parsed according to i18next syntax.\n   *\n   * Example JSON content:\n   * ```js\n   *  {\n   *    \"en\": {               // resource for language `en` (English)\n   *      \"translation\": {\n   *        \"greeting\": \"Hello!\"\n   *      }\n   *    },\n   *    \"de\": {               // resource for language `de` (German)\n   *      \"translation\": {\n   *        \"greeting\": \"Hallo!\"\n   *      }\n   *    }\n   *    \"de-CH\": {            // resource for locale `de-CH` (Swiss German)\n   *      \"translation\": {\n   *        \"greeting\": \"Grüezi!\"\n   *      }\n   *    }\n   *  }\n   * ```\n   * @param {String} locale The locale to translate to.\n   * @returns {Object} The translation or an empty object if no matching\n   * translation was found.\n   */\n  getJsonTranslation(locale) {\n    // If there is no JSON resource\n    if (this.jsonParameters === undefined) {\n      return {};\n    }\n\n    // If locale is not set use the fallback locale\n    locale = locale || this.pagesConfig.localizationFallbackLocale;\n\n    // Get matching translation by locale, language or fallback locale\n    const language = locale.split('-')[0];\n    const resource =\n      this.jsonParameters[locale] ||\n      this.jsonParameters[language] ||\n      this.jsonParameters[this.pagesConfig.localizationFallbackLocale] ||\n      {};\n    const translation = resource.translation || {};\n    return translation;\n  }\n\n  /**\n   * Returns a translation from the JSON resource for a given locale with\n   * placeholders filled in by given parameters.\n   * @param {String} locale The locale to translate to.\n   * @param {Object} params The parameters to fill into any placeholders\n   * within the translations.\n   * @returns {Object} The translation or an empty object if no matching\n   * translation was found.\n   */\n  getJsonPlaceholders(locale, params = {}) {\n    // If localization is disabled or there is no JSON resource\n    if (!this.pagesConfig.enableLocalization || !this.pagesConfig.localizationJsonPath) {\n      return {};\n    }\n\n    // Get JSON placeholders\n    let placeholders = this.getJsonTranslation(locale);\n\n    // Fill in any placeholders in the translation; this allows a translation\n    // to contain default placeholders like {{appName}} which are filled here\n    placeholders = JSON.stringify(placeholders);\n    placeholders = mustache.render(placeholders, params);\n    placeholders = JSON.parse(placeholders);\n\n    return placeholders;\n  }\n\n  /**\n   * Creates a response with file content.\n   * @param {String} path The path of the file to return.\n   * @param {Object} [params={}] The parameters to be included in the response\n   * header. These will also be used to fill placeholders.\n   * @param {Object} [placeholders={}] The placeholders to fill in the content.\n   * These will not be included in the response header.\n   * @returns {Object} The Promise Router response.\n   */\n  async pageResponse(path, params = {}, placeholders = {}) {\n    // Get file content\n    let data;\n    try {\n      data = await this.readFile(path);\n    } catch (e) {\n      return this.notFound();\n    }\n\n    // Get config placeholders; can be an object, a function or an async function\n    let configPlaceholders =\n      typeof this.pagesConfig.placeholders === 'function'\n        ? this.pagesConfig.placeholders(params)\n        : Object.prototype.toString.call(this.pagesConfig.placeholders) === '[object Object]'\n          ? this.pagesConfig.placeholders\n          : {};\n    if (configPlaceholders instanceof Promise) {\n      configPlaceholders = await configPlaceholders;\n    }\n\n    // Fill placeholders\n    const allPlaceholders = Object.assign({}, configPlaceholders, placeholders);\n    const paramsAndPlaceholders = Object.assign({}, params, allPlaceholders);\n    data = mustache.render(data, paramsAndPlaceholders);\n\n    // Add placeholders in header to allow parsing for programmatic use\n    // of response, instead of having to parse the HTML content.\n    const headers = Object.entries(params).reduce((m, p) => {\n      if (p[1] !== undefined) {\n        m[`${pageParamHeaderPrefix}${p[0].toLowerCase()}`] = p[1];\n      }\n      return m;\n    }, {});\n\n    return { text: data, headers: headers };\n  }\n\n  /**\n   * Creates a response with file content.\n   * @param {String} path The path of the file to return.\n   * @returns {Object} The PromiseRouter response.\n   */\n  async fileResponse(path) {\n    // Get file content\n    let data;\n    try {\n      data = await this.readFile(path);\n    } catch (e) {\n      return this.notFound();\n    }\n\n    return { text: data };\n  }\n\n  /**\n   * Reads and returns the content of a file at a given path. File reading to\n   * serve content on the static route is only allowed from the pages\n   * directory on downwards.\n   * -----------------------------------------------------------------------\n   * **WARNING:** All file reads in the PagesRouter must be executed by this\n   * wrapper because it also detects and prevents common exploits.\n   * -----------------------------------------------------------------------\n   * @param {String} filePath The path to the file to read.\n   * @returns {Promise<String>} The file content.\n   */\n  async readFile(filePath) {\n    // Normalize path to prevent it from containing any directory changing\n    // UNIX patterns which could expose the whole file system, e.g.\n    // `http://example.com/parse/apps/../file.txt` requests a file outside\n    // of the pages directory scope.\n    const normalizedPath = path.normalize(filePath);\n\n    // Abort if the path is outside of the path directory scope\n    if (!normalizedPath.startsWith(this.pagesPath)) {\n      throw errors.fileOutsideAllowedScope;\n    }\n\n    return await fs.readFile(normalizedPath, 'utf-8');\n  }\n\n  /**\n   * Loads a language resource JSON file that is used for translations.\n   */\n  loadJsonResource() {\n    if (this.pagesConfig.localizationJsonPath === undefined) {\n      return;\n    }\n    try {\n      const json = require(path.resolve('./', this.pagesConfig.localizationJsonPath));\n      this.jsonParameters = json;\n    } catch (e) {\n      throw errors.jsonFailedFileLoading;\n    }\n  }\n\n  /**\n   * Extracts and returns the page default parameters from the Parse Server\n   * configuration. These parameters are made accessible in every page served\n   * by this router.\n   * @param {Object} config The Parse Server configuration.\n   * @returns {Object} The default parameters.\n   */\n  getDefaultParams(config) {\n    return config\n      ? {\n        [pageParams.appId]: config.appId,\n        [pageParams.appName]: config.appName,\n        [pageParams.publicServerUrl]: config.publicServerURL,\n      }\n      : {};\n  }\n\n  /**\n   * Extracts and returns the locale from an express request.\n   * @param {Object} req The express request.\n   * @returns {String|undefined} The locale, or undefined if no locale was set.\n   */\n  getLocale(req) {\n    const locale =\n      (req.query || {})[pageParams.locale] ||\n      (req.body || {})[pageParams.locale] ||\n      (req.params || {})[pageParams.locale] ||\n      (req.headers || {})[pageParamHeaderPrefix + pageParams.locale];\n    return locale;\n  }\n\n  /**\n   * Creates a response with http redirect.\n   * @param {Object} req The express request.\n   * @param {String} path The path of the file to return.\n   * @param {Object} params The query parameters to include.\n   * @returns {Object} The Promise Router response.\n   */\n  async redirectResponse(url, params) {\n    // Remove any parameters with undefined value\n    params = Object.entries(params).reduce((m, p) => {\n      if (p[1] !== undefined) {\n        m[p[0]] = p[1];\n      }\n      return m;\n    }, {});\n\n    // Compose URL with parameters in query\n    const location = new URL(url);\n    Object.entries(params).forEach(p => location.searchParams.set(p[0], p[1]));\n    const locationString = location.toString();\n\n    // Add parameters to header to allow parsing for programmatic use\n    // of response, instead of having to parse the HTML content.\n    const headers = Object.entries(params).reduce((m, p) => {\n      if (p[1] !== undefined) {\n        m[`${pageParamHeaderPrefix}${p[0].toLowerCase()}`] = p[1];\n      }\n      return m;\n    }, {});\n\n    return {\n      status: 303,\n      location: locationString,\n      headers: headers,\n    };\n  }\n\n  defaultPagePath(file) {\n    return path.join(this.pagesPath, file);\n  }\n\n  composePageUrl(file, publicServerUrl, locale) {\n    let url = publicServerUrl;\n    url += url.endsWith('/') ? '' : '/';\n    url += this.pagesEndpoint + '/';\n    url += locale === undefined ? '' : locale + '/';\n    url += file;\n    return url;\n  }\n\n  notFound() {\n    return {\n      text: 'Not found.',\n      status: 404,\n    };\n  }\n\n  invalidRequest() {\n    const error = new Error();\n    error.status = 403;\n    error.message = 'unauthorized';\n    throw error;\n  }\n\n  /**\n   * Sets the Parse Server configuration in the request object to make it\n   * easily accessible throughtout request processing.\n   * @param {Object} req The request.\n   * @param {Boolean} failGracefully Is true if failing to set the config should\n   * not result in an invalid request response. Default is `false`.\n   */\n  setConfig(req, failGracefully = false) {\n    req.config = Config.get(req.params.appId || req.query.appId);\n    if (!req.config && !failGracefully) {\n      this.invalidRequest();\n    }\n    return Promise.resolve();\n  }\n\n  mountPagesRoutes() {\n    this.route(\n      'GET',\n      `/${this.pagesEndpoint}/:appId/verify_email`,\n      req => {\n        this.setConfig(req);\n      },\n      req => {\n        return this.verifyEmail(req);\n      }\n    );\n\n    this.route(\n      'POST',\n      `/${this.pagesEndpoint}/:appId/resend_verification_email`,\n      req => {\n        this.setConfig(req);\n      },\n      req => {\n        return this.resendVerificationEmail(req);\n      }\n    );\n\n    this.route(\n      'GET',\n      `/${this.pagesEndpoint}/choose_password`,\n      req => {\n        this.setConfig(req);\n      },\n      req => {\n        return this.passwordReset(req);\n      }\n    );\n\n    this.route(\n      'POST',\n      `/${this.pagesEndpoint}/:appId/request_password_reset`,\n      req => {\n        this.setConfig(req);\n      },\n      req => {\n        return this.resetPassword(req);\n      }\n    );\n\n    this.route(\n      'GET',\n      `/${this.pagesEndpoint}/:appId/request_password_reset`,\n      req => {\n        this.setConfig(req);\n      },\n      req => {\n        return this.requestResetPassword(req);\n      }\n    );\n  }\n\n  mountCustomRoutes() {\n    for (const route of this.pagesConfig.customRoutes || []) {\n      this.route(\n        route.method,\n        `/${this.pagesEndpoint}/:appId/${route.path}`,\n        req => {\n          this.setConfig(req);\n        },\n        async req => {\n          const { file, query = {} } = (await route.handler(req)) || {};\n\n          // If route handler did not return a page send 404 response\n          if (!file) {\n            return this.notFound();\n          }\n\n          // Send page response\n          const page = new Page({ id: file, defaultFile: file });\n          return this.goToPage(req, page, query, false);\n        }\n      );\n    }\n  }\n\n  mountStaticRoute() {\n    this.route(\n      'GET',\n      `/${this.pagesEndpoint}/(*)?`,\n      req => {\n        this.setConfig(req, true);\n      },\n      req => {\n        return this.staticRoute(req);\n      }\n    );\n  }\n\n  expressRouter() {\n    const router = express.Router();\n    router.use('/', super.expressRouter());\n    return router;\n  }\n}\n\nexport default PagesRouter;\nmodule.exports = {\n  PagesRouter,\n  pageParamHeaderPrefix,\n  pageParams,\n  pages,\n};\n"],"mappings":"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"}
|