parse-server 2.8.4 → 8.6.2

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.
Files changed (240) hide show
  1. package/LICENSE +167 -25
  2. package/NOTICE +10 -0
  3. package/README.md +929 -278
  4. package/lib/AccountLockout.js +47 -30
  5. package/lib/Adapters/AdapterLoader.js +21 -6
  6. package/lib/Adapters/Analytics/AnalyticsAdapter.js +15 -12
  7. package/lib/Adapters/Auth/AuthAdapter.js +116 -13
  8. package/lib/Adapters/Auth/BaseCodeAuthAdapter.js +99 -0
  9. package/lib/Adapters/Auth/OAuth1Client.js +27 -46
  10. package/lib/Adapters/Auth/apple.js +123 -0
  11. package/lib/Adapters/Auth/facebook.js +162 -35
  12. package/lib/Adapters/Auth/gcenter.js +217 -0
  13. package/lib/Adapters/Auth/github.js +118 -48
  14. package/lib/Adapters/Auth/google.js +160 -51
  15. package/lib/Adapters/Auth/gpgames.js +125 -0
  16. package/lib/Adapters/Auth/httpsRequest.js +6 -7
  17. package/lib/Adapters/Auth/index.js +170 -62
  18. package/lib/Adapters/Auth/instagram.js +114 -40
  19. package/lib/Adapters/Auth/janraincapture.js +52 -23
  20. package/lib/Adapters/Auth/janrainengage.js +19 -36
  21. package/lib/Adapters/Auth/keycloak.js +148 -0
  22. package/lib/Adapters/Auth/ldap.js +167 -0
  23. package/lib/Adapters/Auth/line.js +125 -0
  24. package/lib/Adapters/Auth/linkedin.js +111 -55
  25. package/lib/Adapters/Auth/meetup.js +24 -34
  26. package/lib/Adapters/Auth/mfa.js +324 -0
  27. package/lib/Adapters/Auth/microsoft.js +111 -0
  28. package/lib/Adapters/Auth/oauth2.js +97 -162
  29. package/lib/Adapters/Auth/phantauth.js +53 -0
  30. package/lib/Adapters/Auth/qq.js +108 -49
  31. package/lib/Adapters/Auth/spotify.js +107 -55
  32. package/lib/Adapters/Auth/twitter.js +188 -48
  33. package/lib/Adapters/Auth/utils.js +28 -0
  34. package/lib/Adapters/Auth/vkontakte.js +26 -39
  35. package/lib/Adapters/Auth/wechat.js +106 -44
  36. package/lib/Adapters/Auth/weibo.js +132 -58
  37. package/lib/Adapters/Cache/CacheAdapter.js +13 -8
  38. package/lib/Adapters/Cache/InMemoryCache.js +3 -13
  39. package/lib/Adapters/Cache/InMemoryCacheAdapter.js +5 -13
  40. package/lib/Adapters/Cache/LRUCache.js +13 -27
  41. package/lib/Adapters/Cache/NullCacheAdapter.js +3 -8
  42. package/lib/Adapters/Cache/RedisCacheAdapter.js +85 -76
  43. package/lib/Adapters/Cache/SchemaCache.js +25 -0
  44. package/lib/Adapters/Email/MailAdapter.js +10 -8
  45. package/lib/Adapters/Files/FilesAdapter.js +83 -25
  46. package/lib/Adapters/Files/GridFSBucketAdapter.js +231 -0
  47. package/lib/Adapters/Files/GridStoreAdapter.js +4 -91
  48. package/lib/Adapters/Logger/LoggerAdapter.js +18 -14
  49. package/lib/Adapters/Logger/WinstonLogger.js +69 -88
  50. package/lib/Adapters/Logger/WinstonLoggerAdapter.js +7 -16
  51. package/lib/Adapters/MessageQueue/EventEmitterMQ.js +8 -26
  52. package/lib/Adapters/PubSub/EventEmitterPubSub.js +12 -25
  53. package/lib/Adapters/PubSub/PubSubAdapter.js +34 -0
  54. package/lib/Adapters/PubSub/RedisPubSub.js +42 -19
  55. package/lib/Adapters/Push/PushAdapter.js +14 -7
  56. package/lib/Adapters/Storage/Mongo/MongoCollection.js +137 -45
  57. package/lib/Adapters/Storage/Mongo/MongoSchemaCollection.js +158 -63
  58. package/lib/Adapters/Storage/Mongo/MongoStorageAdapter.js +320 -168
  59. package/lib/Adapters/Storage/Mongo/MongoTransform.js +279 -306
  60. package/lib/Adapters/Storage/Postgres/PostgresClient.js +14 -10
  61. package/lib/Adapters/Storage/Postgres/PostgresConfigParser.js +47 -21
  62. package/lib/Adapters/Storage/Postgres/PostgresStorageAdapter.js +854 -468
  63. package/lib/Adapters/Storage/Postgres/sql/index.js +4 -6
  64. package/lib/Adapters/Storage/StorageAdapter.js +1 -1
  65. package/lib/Adapters/WebSocketServer/WSAdapter.js +35 -0
  66. package/lib/Adapters/WebSocketServer/WSSAdapter.js +66 -0
  67. package/lib/Auth.js +488 -125
  68. package/lib/ClientSDK.js +2 -6
  69. package/lib/Config.js +525 -94
  70. package/lib/Controllers/AdaptableController.js +5 -25
  71. package/lib/Controllers/AnalyticsController.js +22 -23
  72. package/lib/Controllers/CacheController.js +10 -31
  73. package/lib/Controllers/DatabaseController.js +767 -313
  74. package/lib/Controllers/FilesController.js +49 -54
  75. package/lib/Controllers/HooksController.js +80 -84
  76. package/lib/Controllers/LiveQueryController.js +35 -22
  77. package/lib/Controllers/LoggerController.js +22 -58
  78. package/lib/Controllers/ParseGraphQLController.js +293 -0
  79. package/lib/Controllers/PushController.js +58 -49
  80. package/lib/Controllers/SchemaController.js +916 -422
  81. package/lib/Controllers/UserController.js +265 -180
  82. package/lib/Controllers/index.js +90 -125
  83. package/lib/Controllers/types.js +1 -1
  84. package/lib/Deprecator/Deprecations.js +30 -0
  85. package/lib/Deprecator/Deprecator.js +127 -0
  86. package/lib/Error.js +48 -0
  87. package/lib/GraphQL/ParseGraphQLSchema.js +375 -0
  88. package/lib/GraphQL/ParseGraphQLServer.js +214 -0
  89. package/lib/GraphQL/helpers/objectsMutations.js +30 -0
  90. package/lib/GraphQL/helpers/objectsQueries.js +246 -0
  91. package/lib/GraphQL/loaders/configMutations.js +87 -0
  92. package/lib/GraphQL/loaders/configQueries.js +79 -0
  93. package/lib/GraphQL/loaders/defaultGraphQLMutations.js +21 -0
  94. package/lib/GraphQL/loaders/defaultGraphQLQueries.js +23 -0
  95. package/lib/GraphQL/loaders/defaultGraphQLTypes.js +1098 -0
  96. package/lib/GraphQL/loaders/defaultRelaySchema.js +53 -0
  97. package/lib/GraphQL/loaders/filesMutations.js +107 -0
  98. package/lib/GraphQL/loaders/functionsMutations.js +78 -0
  99. package/lib/GraphQL/loaders/parseClassMutations.js +268 -0
  100. package/lib/GraphQL/loaders/parseClassQueries.js +127 -0
  101. package/lib/GraphQL/loaders/parseClassTypes.js +493 -0
  102. package/lib/GraphQL/loaders/schemaDirectives.js +62 -0
  103. package/lib/GraphQL/loaders/schemaMutations.js +162 -0
  104. package/lib/GraphQL/loaders/schemaQueries.js +81 -0
  105. package/lib/GraphQL/loaders/schemaTypes.js +341 -0
  106. package/lib/GraphQL/loaders/usersMutations.js +433 -0
  107. package/lib/GraphQL/loaders/usersQueries.js +90 -0
  108. package/lib/GraphQL/parseGraphQLUtils.js +63 -0
  109. package/lib/GraphQL/transformers/className.js +14 -0
  110. package/lib/GraphQL/transformers/constraintType.js +53 -0
  111. package/lib/GraphQL/transformers/inputType.js +51 -0
  112. package/lib/GraphQL/transformers/mutation.js +274 -0
  113. package/lib/GraphQL/transformers/outputType.js +51 -0
  114. package/lib/GraphQL/transformers/query.js +237 -0
  115. package/lib/GraphQL/transformers/schemaFields.js +99 -0
  116. package/lib/KeyPromiseQueue.js +48 -0
  117. package/lib/LiveQuery/Client.js +25 -33
  118. package/lib/LiveQuery/Id.js +2 -5
  119. package/lib/LiveQuery/ParseCloudCodePublisher.js +26 -23
  120. package/lib/LiveQuery/ParseLiveQueryServer.js +560 -285
  121. package/lib/LiveQuery/ParsePubSub.js +7 -16
  122. package/lib/LiveQuery/ParseWebSocketServer.js +42 -39
  123. package/lib/LiveQuery/QueryTools.js +76 -15
  124. package/lib/LiveQuery/RequestSchema.js +111 -97
  125. package/lib/LiveQuery/SessionTokenCache.js +23 -36
  126. package/lib/LiveQuery/Subscription.js +8 -17
  127. package/lib/LiveQuery/equalObjects.js +2 -3
  128. package/lib/Options/Definitions.js +1355 -382
  129. package/lib/Options/docs.js +301 -62
  130. package/lib/Options/index.js +11 -1
  131. package/lib/Options/parsers.js +14 -10
  132. package/lib/Page.js +44 -0
  133. package/lib/ParseMessageQueue.js +6 -13
  134. package/lib/ParseServer.js +474 -235
  135. package/lib/ParseServerRESTController.js +102 -40
  136. package/lib/PromiseRouter.js +39 -50
  137. package/lib/Push/PushQueue.js +24 -30
  138. package/lib/Push/PushWorker.js +32 -56
  139. package/lib/Push/utils.js +22 -35
  140. package/lib/RestQuery.js +361 -139
  141. package/lib/RestWrite.js +713 -344
  142. package/lib/Routers/AggregateRouter.js +97 -71
  143. package/lib/Routers/AnalyticsRouter.js +8 -14
  144. package/lib/Routers/AudiencesRouter.js +16 -35
  145. package/lib/Routers/ClassesRouter.js +86 -72
  146. package/lib/Routers/CloudCodeRouter.js +28 -37
  147. package/lib/Routers/FeaturesRouter.js +22 -25
  148. package/lib/Routers/FilesRouter.js +266 -171
  149. package/lib/Routers/FunctionsRouter.js +87 -103
  150. package/lib/Routers/GlobalConfigRouter.js +94 -33
  151. package/lib/Routers/GraphQLRouter.js +41 -0
  152. package/lib/Routers/HooksRouter.js +43 -47
  153. package/lib/Routers/IAPValidationRouter.js +57 -70
  154. package/lib/Routers/InstallationsRouter.js +17 -25
  155. package/lib/Routers/LogsRouter.js +10 -25
  156. package/lib/Routers/PagesRouter.js +647 -0
  157. package/lib/Routers/PublicAPIRouter.js +104 -112
  158. package/lib/Routers/PurgeRouter.js +19 -29
  159. package/lib/Routers/PushRouter.js +14 -28
  160. package/lib/Routers/RolesRouter.js +7 -14
  161. package/lib/Routers/SchemasRouter.js +63 -42
  162. package/lib/Routers/SecurityRouter.js +34 -0
  163. package/lib/Routers/SessionsRouter.js +25 -38
  164. package/lib/Routers/UsersRouter.js +463 -190
  165. package/lib/SchemaMigrations/DefinedSchemas.js +379 -0
  166. package/lib/SchemaMigrations/Migrations.js +30 -0
  167. package/lib/Security/Check.js +109 -0
  168. package/lib/Security/CheckGroup.js +44 -0
  169. package/lib/Security/CheckGroups/CheckGroupDatabase.js +44 -0
  170. package/lib/Security/CheckGroups/CheckGroupServerConfig.js +96 -0
  171. package/lib/Security/CheckGroups/CheckGroups.js +21 -0
  172. package/lib/Security/CheckRunner.js +213 -0
  173. package/lib/SharedRest.js +29 -0
  174. package/lib/StatusHandler.js +96 -93
  175. package/lib/TestUtils.js +70 -14
  176. package/lib/Utils.js +468 -0
  177. package/lib/batch.js +74 -40
  178. package/lib/cache.js +8 -8
  179. package/lib/cli/definitions/parse-live-query-server.js +4 -3
  180. package/lib/cli/definitions/parse-server.js +4 -3
  181. package/lib/cli/parse-live-query-server.js +9 -17
  182. package/lib/cli/parse-server.js +49 -47
  183. package/lib/cli/utils/commander.js +20 -29
  184. package/lib/cli/utils/runner.js +31 -32
  185. package/lib/cloud-code/Parse.Cloud.js +711 -36
  186. package/lib/cloud-code/Parse.Server.js +21 -0
  187. package/lib/cryptoUtils.js +6 -11
  188. package/lib/defaults.js +21 -15
  189. package/lib/deprecated.js +1 -1
  190. package/lib/index.js +78 -67
  191. package/lib/logger.js +12 -20
  192. package/lib/middlewares.js +484 -160
  193. package/lib/password.js +10 -6
  194. package/lib/request.js +175 -0
  195. package/lib/requiredParameter.js +4 -3
  196. package/lib/rest.js +157 -82
  197. package/lib/triggers.js +627 -185
  198. package/lib/vendor/README.md +3 -3
  199. package/lib/vendor/mongodbUrl.js +224 -137
  200. package/package.json +135 -57
  201. package/postinstall.js +38 -50
  202. package/public_html/invalid_verification_link.html +3 -3
  203. package/types/@types/@parse/fs-files-adapter/index.d.ts +5 -0
  204. package/types/@types/deepcopy/index.d.ts +5 -0
  205. package/types/LiveQuery/ParseLiveQueryServer.d.ts +40 -0
  206. package/types/Options/index.d.ts +301 -0
  207. package/types/ParseServer.d.ts +65 -0
  208. package/types/eslint.config.mjs +30 -0
  209. package/types/index.d.ts +21 -0
  210. package/types/logger.d.ts +2 -0
  211. package/types/tests.ts +44 -0
  212. package/types/tsconfig.json +24 -0
  213. package/CHANGELOG.md +0 -1246
  214. package/PATENTS +0 -37
  215. package/bin/dev +0 -37
  216. package/lib/.DS_Store +0 -0
  217. package/lib/Adapters/Auth/common.js +0 -2
  218. package/lib/Adapters/Auth/facebookaccountkit.js +0 -69
  219. package/lib/Controllers/SchemaCache.js +0 -97
  220. package/lib/LiveQuery/.DS_Store +0 -0
  221. package/lib/cli/utils/parsers.js +0 -77
  222. package/lib/cloud-code/.DS_Store +0 -0
  223. package/lib/cloud-code/HTTPResponse.js +0 -57
  224. package/lib/cloud-code/Untitled-1 +0 -123
  225. package/lib/cloud-code/httpRequest.js +0 -102
  226. package/lib/cloud-code/team.html +0 -123
  227. package/lib/graphql/ParseClass.js +0 -234
  228. package/lib/graphql/Schema.js +0 -197
  229. package/lib/graphql/index.js +0 -1
  230. package/lib/graphql/types/ACL.js +0 -35
  231. package/lib/graphql/types/Date.js +0 -25
  232. package/lib/graphql/types/File.js +0 -24
  233. package/lib/graphql/types/GeoPoint.js +0 -35
  234. package/lib/graphql/types/JSONObject.js +0 -30
  235. package/lib/graphql/types/NumberInput.js +0 -43
  236. package/lib/graphql/types/NumberQuery.js +0 -42
  237. package/lib/graphql/types/Pointer.js +0 -35
  238. package/lib/graphql/types/QueryConstraint.js +0 -61
  239. package/lib/graphql/types/StringQuery.js +0 -39
  240. package/lib/graphql/types/index.js +0 -110
@@ -1,25 +1,29 @@
1
- 'use strict';
1
+ "use strict";
2
2
 
3
3
  Object.defineProperty(exports, "__esModule", {
4
4
  value: true
5
5
  });
6
+ exports.ParseServerRESTController = ParseServerRESTController;
7
+ exports.default = void 0;
8
+ var _RESTController = _interopRequireDefault(require("parse/lib/node/RESTController"));
9
+ function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
6
10
  const Config = require('./Config');
7
11
  const Auth = require('./Auth');
8
- const RESTController = require('parse/lib/node/RESTController');
9
- const URL = require('url');
10
12
  const Parse = require('parse/node');
11
-
12
13
  function getSessionToken(options) {
13
14
  if (options && typeof options.sessionToken === 'string') {
14
- return Parse.Promise.as(options.sessionToken);
15
+ return Promise.resolve(options.sessionToken);
15
16
  }
16
- return Parse.Promise.as(null);
17
+ return Promise.resolve(null);
17
18
  }
18
-
19
19
  function getAuth(options = {}, config) {
20
20
  const installationId = options.installationId || 'cloud';
21
21
  if (options.useMasterKey) {
22
- return Parse.Promise.as(new Auth.Auth({ config, isMaster: true, installationId }));
22
+ return Promise.resolve(new Auth.Auth({
23
+ config,
24
+ isMaster: true,
25
+ installationId
26
+ }));
23
27
  }
24
28
  return getSessionToken(options).then(sessionToken => {
25
29
  if (sessionToken) {
@@ -30,43 +34,88 @@ function getAuth(options = {}, config) {
30
34
  installationId
31
35
  });
32
36
  } else {
33
- return Parse.Promise.as(new Auth.Auth({ config, installationId }));
37
+ return Promise.resolve(new Auth.Auth({
38
+ config,
39
+ installationId
40
+ }));
34
41
  }
35
42
  });
36
43
  }
37
-
38
44
  function ParseServerRESTController(applicationId, router) {
39
- function handleRequest(method, path, data = {}, options = {}) {
45
+ function handleRequest(method, path, data = {}, options = {}, config) {
40
46
  // Store the arguments, for later use if internal fails
41
47
  const args = arguments;
42
-
43
- const config = Config.get(applicationId);
44
- const serverURL = URL.parse(config.serverURL);
45
- if (path.indexOf(serverURL.path) === 0) {
46
- path = path.slice(serverURL.path.length, path.length);
48
+ if (!config) {
49
+ config = Config.get(applicationId);
47
50
  }
48
-
49
- if (path[0] !== "/") {
50
- path = "/" + path;
51
+ const serverURL = new URL(config.serverURL);
52
+ if (path.indexOf(serverURL.pathname) === 0) {
53
+ path = path.slice(serverURL.pathname.length, path.length);
54
+ }
55
+ if (path[0] !== '/') {
56
+ path = '/' + path;
51
57
  }
52
-
53
58
  if (path === '/batch') {
54
- const promises = data.requests.map(request => {
55
- return handleRequest(request.method, request.path, request.body, options).then(response => {
56
- return Parse.Promise.as({ success: response });
57
- }, error => {
58
- return Parse.Promise.as({ error: { code: error.code, error: error.message } });
59
+ const batch = transactionRetries => {
60
+ let initialPromise = Promise.resolve();
61
+ if (data.transaction === true) {
62
+ initialPromise = config.database.createTransactionalSession();
63
+ }
64
+ return initialPromise.then(() => {
65
+ const promises = data.requests.map(request => {
66
+ return handleRequest(request.method, request.path, request.body, options, config).then(response => {
67
+ if (options.returnStatus) {
68
+ const status = response._status;
69
+ const headers = response._headers;
70
+ delete response._status;
71
+ delete response._headers;
72
+ return {
73
+ success: response,
74
+ _status: status,
75
+ _headers: headers
76
+ };
77
+ }
78
+ return {
79
+ success: response
80
+ };
81
+ }, error => {
82
+ return {
83
+ error: {
84
+ code: error.code,
85
+ error: error.message
86
+ }
87
+ };
88
+ });
89
+ });
90
+ return Promise.all(promises).then(result => {
91
+ if (data.transaction === true) {
92
+ if (result.find(resultItem => typeof resultItem.error === 'object')) {
93
+ return config.database.abortTransactionalSession().then(() => {
94
+ return Promise.reject(result);
95
+ });
96
+ } else {
97
+ return config.database.commitTransactionalSession().then(() => {
98
+ return result;
99
+ });
100
+ }
101
+ } else {
102
+ return result;
103
+ }
104
+ }).catch(error => {
105
+ if (error && error.find(errorItem => typeof errorItem.error === 'object' && errorItem.error.code === 251) && transactionRetries > 0) {
106
+ return batch(transactionRetries - 1);
107
+ }
108
+ throw error;
109
+ });
59
110
  });
60
- });
61
- return Parse.Promise.all(promises);
111
+ };
112
+ return batch(5);
62
113
  }
63
-
64
114
  let query;
65
115
  if (method === 'GET') {
66
116
  query = data;
67
117
  }
68
-
69
- return new Parse.Promise((resolve, reject) => {
118
+ return new Promise((resolve, reject) => {
70
119
  getAuth(options, config).then(auth => {
71
120
  const request = {
72
121
  body: data,
@@ -74,17 +123,32 @@ function ParseServerRESTController(applicationId, router) {
74
123
  auth,
75
124
  info: {
76
125
  applicationId: applicationId,
77
- sessionToken: options.sessionToken
126
+ sessionToken: options.sessionToken,
127
+ installationId: options.installationId,
128
+ context: options.context || {}
78
129
  },
79
130
  query
80
131
  };
81
132
  return Promise.resolve().then(() => {
82
133
  return router.tryRouteRequest(method, path, request);
83
- }).then(response => {
84
- resolve(response.response, response.status, response);
134
+ }).then(resp => {
135
+ const {
136
+ response,
137
+ status,
138
+ headers = {}
139
+ } = resp;
140
+ if (options.returnStatus) {
141
+ resolve({
142
+ ...response,
143
+ _status: status,
144
+ _headers: headers
145
+ });
146
+ } else {
147
+ resolve(response);
148
+ }
85
149
  }, err => {
86
150
  if (err instanceof Parse.Error && err.code == Parse.Error.INVALID_JSON && err.message == `cannot route ${method} ${path}`) {
87
- RESTController.request.apply(null, args).then(resolve, reject);
151
+ _RESTController.default.request.apply(null, args).then(resolve, reject);
88
152
  } else {
89
153
  reject(err);
90
154
  }
@@ -92,13 +156,11 @@ function ParseServerRESTController(applicationId, router) {
92
156
  }, reject);
93
157
  });
94
158
  }
95
-
96
159
  return {
97
160
  request: handleRequest,
98
- ajax: RESTController.ajax
161
+ ajax: _RESTController.default.ajax,
162
+ handleError: _RESTController.default.handleError
99
163
  };
100
164
  }
101
-
102
- exports.default = ParseServerRESTController;
103
- exports.ParseServerRESTController = ParseServerRESTController;
104
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../src/ParseServerRESTController.js"],"names":["Config","require","Auth","RESTController","URL","Parse","getSessionToken","options","sessionToken","Promise","as","getAuth","config","installationId","useMasterKey","isMaster","then","getAuthForSessionToken","ParseServerRESTController","applicationId","router","handleRequest","method","path","data","args","arguments","get","serverURL","parse","indexOf","slice","length","promises","requests","map","request","body","response","success","error","code","message","all","query","resolve","reject","auth","info","tryRouteRequest","status","err","Error","INVALID_JSON","apply","ajax"],"mappings":";;;;;AAAA,MAAMA,SAASC,QAAQ,UAAR,CAAf;AACA,MAAMC,OAAOD,QAAQ,QAAR,CAAb;AACA,MAAME,iBAAiBF,QAAQ,+BAAR,CAAvB;AACA,MAAMG,MAAMH,QAAQ,KAAR,CAAZ;AACA,MAAMI,QAAQJ,QAAQ,YAAR,CAAd;;AAEA,SAASK,eAAT,CAAyBC,OAAzB,EAAkC;AAChC,MAAIA,WAAW,OAAOA,QAAQC,YAAf,KAAgC,QAA/C,EAAyD;AACvD,WAAOH,MAAMI,OAAN,CAAcC,EAAd,CAAiBH,QAAQC,YAAzB,CAAP;AACD;AACD,SAAOH,MAAMI,OAAN,CAAcC,EAAd,CAAiB,IAAjB,CAAP;AACD;;AAED,SAASC,OAAT,CAAiBJ,UAAU,EAA3B,EAA+BK,MAA/B,EAAuC;AACrC,QAAMC,iBAAiBN,QAAQM,cAAR,IAA0B,OAAjD;AACA,MAAIN,QAAQO,YAAZ,EAA0B;AACxB,WAAOT,MAAMI,OAAN,CAAcC,EAAd,CAAiB,IAAIR,KAAKA,IAAT,CAAc,EAACU,MAAD,EAASG,UAAU,IAAnB,EAAyBF,cAAzB,EAAd,CAAjB,CAAP;AACD;AACD,SAAOP,gBAAgBC,OAAhB,EAAyBS,IAAzB,CAA+BR,YAAD,IAAkB;AACrD,QAAIA,YAAJ,EAAkB;AAChBD,cAAQC,YAAR,GAAuBA,YAAvB;AACA,aAAON,KAAKe,sBAAL,CAA4B;AACjCL,cADiC;AAEjCJ,sBAAcA,YAFmB;AAGjCK;AAHiC,OAA5B,CAAP;AAKD,KAPD,MAOO;AACL,aAAOR,MAAMI,OAAN,CAAcC,EAAd,CAAiB,IAAIR,KAAKA,IAAT,CAAc,EAAEU,MAAF,EAAUC,cAAV,EAAd,CAAjB,CAAP;AACD;AACF,GAXM,CAAP;AAYD;;AAED,SAASK,yBAAT,CAAmCC,aAAnC,EAAkDC,MAAlD,EAA0D;AACxD,WAASC,aAAT,CAAuBC,MAAvB,EAA+BC,IAA/B,EAAqCC,OAAO,EAA5C,EAAgDjB,UAAU,EAA1D,EAA8D;AAC5D;AACA,UAAMkB,OAAOC,SAAb;;AAEA,UAAMd,SAASZ,OAAO2B,GAAP,CAAWR,aAAX,CAAf;AACA,UAAMS,YAAYxB,IAAIyB,KAAJ,CAAUjB,OAAOgB,SAAjB,CAAlB;AACA,QAAIL,KAAKO,OAAL,CAAaF,UAAUL,IAAvB,MAAiC,CAArC,EAAwC;AACtCA,aAAOA,KAAKQ,KAAL,CAAWH,UAAUL,IAAV,CAAeS,MAA1B,EAAkCT,KAAKS,MAAvC,CAAP;AACD;;AAED,QAAIT,KAAK,CAAL,MAAY,GAAhB,EAAqB;AACnBA,aAAO,MAAMA,IAAb;AACD;;AAED,QAAIA,SAAS,QAAb,EAAuB;AACrB,YAAMU,WAAWT,KAAKU,QAAL,CAAcC,GAAd,CAAmBC,OAAD,IAAa;AAC9C,eAAOf,cAAce,QAAQd,MAAtB,EAA8Bc,QAAQb,IAAtC,EAA4Ca,QAAQC,IAApD,EAA0D9B,OAA1D,EAAmES,IAAnE,CAAyEsB,QAAD,IAAc;AAC3F,iBAAOjC,MAAMI,OAAN,CAAcC,EAAd,CAAiB,EAAC6B,SAASD,QAAV,EAAjB,CAAP;AACD,SAFM,EAEHE,KAAD,IAAW;AACZ,iBAAOnC,MAAMI,OAAN,CAAcC,EAAd,CAAiB,EAAC8B,OAAO,EAACC,MAAMD,MAAMC,IAAb,EAAmBD,OAAOA,MAAME,OAAhC,EAAR,EAAjB,CAAP;AACD,SAJM,CAAP;AAKD,OANgB,CAAjB;AAOA,aAAOrC,MAAMI,OAAN,CAAckC,GAAd,CAAkBV,QAAlB,CAAP;AACD;;AAED,QAAIW,KAAJ;AACA,QAAItB,WAAW,KAAf,EAAsB;AACpBsB,cAAQpB,IAAR;AACD;;AAED,WAAO,IAAInB,MAAMI,OAAV,CAAkB,CAACoC,OAAD,EAAUC,MAAV,KAAqB;AAC5CnC,cAAQJ,OAAR,EAAiBK,MAAjB,EAAyBI,IAAzB,CAA+B+B,IAAD,IAAU;AACtC,cAAMX,UAAU;AACdC,gBAAMb,IADQ;AAEdZ,gBAFc;AAGdmC,cAHc;AAIdC,gBAAM;AACJ7B,2BAAeA,aADX;AAEJX,0BAAcD,QAAQC;AAFlB,WAJQ;AAQdoC;AARc,SAAhB;AAUA,eAAOnC,QAAQoC,OAAR,GAAkB7B,IAAlB,CAAuB,MAAM;AAClC,iBAAOI,OAAO6B,eAAP,CAAuB3B,MAAvB,EAA+BC,IAA/B,EAAqCa,OAArC,CAAP;AACD,SAFM,EAEJpB,IAFI,CAEEsB,QAAD,IAAc;AACpBO,kBAAQP,SAASA,QAAjB,EAA2BA,SAASY,MAApC,EAA4CZ,QAA5C;AACD,SAJM,EAIHa,GAAD,IAAS;AACV,cAAIA,eAAe9C,MAAM+C,KAArB,IACAD,IAAIV,IAAJ,IAAYpC,MAAM+C,KAAN,CAAYC,YADxB,IAEAF,IAAIT,OAAJ,IAAgB,gBAAepB,MAAO,IAAGC,IAAK,EAFlD,EAEqD;AACnDpB,2BAAeiC,OAAf,CAAuBkB,KAAvB,CAA6B,IAA7B,EAAmC7B,IAAnC,EAAyCT,IAAzC,CAA8C6B,OAA9C,EAAuDC,MAAvD;AACD,WAJD,MAIO;AACLA,mBAAOK,GAAP;AACD;AACF,SAZM,CAAP;AAaD,OAxBD,EAwBGL,MAxBH;AAyBD,KA1BM,CAAP;AA2BD;;AAED,SAAQ;AACNV,aAASf,aADH;AAENkC,UAAMpD,eAAeoD;AAFf,GAAR;AAID;;kBAEcrC,yB;QACNA,yB,GAAAA,yB","file":"ParseServerRESTController.js","sourcesContent":["const Config = require('./Config');\nconst Auth = require('./Auth');\nconst RESTController = require('parse/lib/node/RESTController');\nconst URL = require('url');\nconst Parse = require('parse/node');\n\nfunction getSessionToken(options) {\n  if (options && typeof options.sessionToken === 'string') {\n    return Parse.Promise.as(options.sessionToken);\n  }\n  return Parse.Promise.as(null);\n}\n\nfunction getAuth(options = {}, config) {\n  const installationId = options.installationId || 'cloud';\n  if (options.useMasterKey) {\n    return Parse.Promise.as(new Auth.Auth({config, isMaster: true, installationId }));\n  }\n  return getSessionToken(options).then((sessionToken) => {\n    if (sessionToken) {\n      options.sessionToken = sessionToken;\n      return Auth.getAuthForSessionToken({\n        config,\n        sessionToken: sessionToken,\n        installationId\n      });\n    } else {\n      return Parse.Promise.as(new Auth.Auth({ config, installationId }));\n    }\n  })\n}\n\nfunction ParseServerRESTController(applicationId, router) {\n  function handleRequest(method, path, data = {}, options = {}) {\n    // Store the arguments, for later use if internal fails\n    const args = arguments;\n\n    const config = Config.get(applicationId);\n    const serverURL = URL.parse(config.serverURL);\n    if (path.indexOf(serverURL.path) === 0) {\n      path = path.slice(serverURL.path.length, path.length);\n    }\n\n    if (path[0] !== \"/\") {\n      path = \"/\" + path;\n    }\n\n    if (path === '/batch') {\n      const promises = data.requests.map((request) => {\n        return handleRequest(request.method, request.path, request.body, options).then((response) => {\n          return Parse.Promise.as({success: response});\n        }, (error) => {\n          return Parse.Promise.as({error: {code: error.code, error: error.message}});\n        });\n      });\n      return Parse.Promise.all(promises);\n    }\n\n    let query;\n    if (method === 'GET') {\n      query = data;\n    }\n\n    return new Parse.Promise((resolve, reject) => {\n      getAuth(options, config).then((auth) => {\n        const request = {\n          body: data,\n          config,\n          auth,\n          info: {\n            applicationId: applicationId,\n            sessionToken: options.sessionToken\n          },\n          query\n        };\n        return Promise.resolve().then(() => {\n          return router.tryRouteRequest(method, path, request);\n        }).then((response) => {\n          resolve(response.response, response.status, response);\n        }, (err) => {\n          if (err instanceof Parse.Error &&\n              err.code == Parse.Error.INVALID_JSON &&\n              err.message == `cannot route ${method} ${path}`) {\n            RESTController.request.apply(null, args).then(resolve, reject);\n          } else {\n            reject(err);\n          }\n        });\n      }, reject);\n    });\n  }\n\n  return  {\n    request: handleRequest,\n    ajax: RESTController.ajax\n  };\n}\n\nexport default ParseServerRESTController;\nexport { ParseServerRESTController };\n"]}
165
+ var _default = exports.default = ParseServerRESTController;
166
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_RESTController","_interopRequireDefault","require","e","__esModule","default","Config","Auth","Parse","getSessionToken","options","sessionToken","Promise","resolve","getAuth","config","installationId","useMasterKey","isMaster","then","getAuthForSessionToken","ParseServerRESTController","applicationId","router","handleRequest","method","path","data","args","arguments","get","serverURL","URL","indexOf","pathname","slice","length","batch","transactionRetries","initialPromise","transaction","database","createTransactionalSession","promises","requests","map","request","body","response","returnStatus","status","_status","headers","_headers","success","error","code","message","all","result","find","resultItem","abortTransactionalSession","reject","commitTransactionalSession","catch","errorItem","query","auth","info","context","tryRouteRequest","resp","err","Error","INVALID_JSON","RESTController","apply","ajax","handleError","_default","exports"],"sources":["../src/ParseServerRESTController.js"],"sourcesContent":["const Config = require('./Config');\nconst Auth = require('./Auth');\nimport RESTController from 'parse/lib/node/RESTController';\nconst Parse = require('parse/node');\n\nfunction getSessionToken(options) {\n  if (options && typeof options.sessionToken === 'string') {\n    return Promise.resolve(options.sessionToken);\n  }\n  return Promise.resolve(null);\n}\n\nfunction getAuth(options = {}, config) {\n  const installationId = options.installationId || 'cloud';\n  if (options.useMasterKey) {\n    return Promise.resolve(new Auth.Auth({ config, isMaster: true, installationId }));\n  }\n  return getSessionToken(options).then(sessionToken => {\n    if (sessionToken) {\n      options.sessionToken = sessionToken;\n      return Auth.getAuthForSessionToken({\n        config,\n        sessionToken: sessionToken,\n        installationId,\n      });\n    } else {\n      return Promise.resolve(new Auth.Auth({ config, installationId }));\n    }\n  });\n}\n\nfunction ParseServerRESTController(applicationId, router) {\n  function handleRequest(method, path, data = {}, options = {}, config) {\n    // Store the arguments, for later use if internal fails\n    const args = arguments;\n\n    if (!config) {\n      config = Config.get(applicationId);\n    }\n    const serverURL = new URL(config.serverURL);\n    if (path.indexOf(serverURL.pathname) === 0) {\n      path = path.slice(serverURL.pathname.length, path.length);\n    }\n\n    if (path[0] !== '/') {\n      path = '/' + path;\n    }\n\n    if (path === '/batch') {\n      const batch = transactionRetries => {\n        let initialPromise = Promise.resolve();\n        if (data.transaction === true) {\n          initialPromise = config.database.createTransactionalSession();\n        }\n        return initialPromise.then(() => {\n          const promises = data.requests.map(request => {\n            return handleRequest(request.method, request.path, request.body, options, config).then(\n              response => {\n                if (options.returnStatus) {\n                  const status = response._status;\n                  const headers = response._headers;\n                  delete response._status;\n                  delete response._headers;\n                  return { success: response, _status: status, _headers: headers };\n                }\n                return { success: response };\n              },\n              error => {\n                return {\n                  error: { code: error.code, error: error.message },\n                };\n              }\n            );\n          });\n          return Promise.all(promises)\n            .then(result => {\n              if (data.transaction === true) {\n                if (result.find(resultItem => typeof resultItem.error === 'object')) {\n                  return config.database.abortTransactionalSession().then(() => {\n                    return Promise.reject(result);\n                  });\n                } else {\n                  return config.database.commitTransactionalSession().then(() => {\n                    return result;\n                  });\n                }\n              } else {\n                return result;\n              }\n            })\n            .catch(error => {\n              if (\n                error &&\n                error.find(\n                  errorItem => typeof errorItem.error === 'object' && errorItem.error.code === 251\n                ) &&\n                transactionRetries > 0\n              ) {\n                return batch(transactionRetries - 1);\n              }\n              throw error;\n            });\n        });\n      };\n      return batch(5);\n    }\n\n    let query;\n    if (method === 'GET') {\n      query = data;\n    }\n\n    return new Promise((resolve, reject) => {\n      getAuth(options, config).then(auth => {\n        const request = {\n          body: data,\n          config,\n          auth,\n          info: {\n            applicationId: applicationId,\n            sessionToken: options.sessionToken,\n            installationId: options.installationId,\n            context: options.context || {},\n          },\n          query,\n        };\n        return Promise.resolve()\n          .then(() => {\n            return router.tryRouteRequest(method, path, request);\n          })\n          .then(\n            resp => {\n              const { response, status, headers = {} } = resp;\n              if (options.returnStatus) {\n                resolve({ ...response, _status: status, _headers: headers });\n              } else {\n                resolve(response);\n              }\n            },\n            err => {\n              if (\n                err instanceof Parse.Error &&\n                err.code == Parse.Error.INVALID_JSON &&\n                err.message == `cannot route ${method} ${path}`\n              ) {\n                RESTController.request.apply(null, args).then(resolve, reject);\n              } else {\n                reject(err);\n              }\n            }\n          );\n      }, reject);\n    });\n  }\n\n  return {\n    request: handleRequest,\n    ajax: RESTController.ajax,\n    handleError: RESTController.handleError,\n  };\n}\n\nexport default ParseServerRESTController;\nexport { ParseServerRESTController };\n"],"mappings":"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","ignoreList":[]}
@@ -1,25 +1,14 @@
1
- 'use strict';
1
+ "use strict";
2
2
 
3
3
  Object.defineProperty(exports, "__esModule", {
4
4
  value: true
5
5
  });
6
-
7
- var _node = require('parse/node');
8
-
9
- var _node2 = _interopRequireDefault(_node);
10
-
11
- var _express = require('express');
12
-
13
- var _express2 = _interopRequireDefault(_express);
14
-
15
- var _logger = require('./logger');
16
-
17
- var _logger2 = _interopRequireDefault(_logger);
18
-
19
- var _util = require('util');
20
-
21
- function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
22
-
6
+ exports.default = void 0;
7
+ var _node = _interopRequireDefault(require("parse/node"));
8
+ var _express = _interopRequireDefault(require("express"));
9
+ var _logger = _interopRequireDefault(require("./logger"));
10
+ var _util = require("util");
11
+ function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
23
12
  // A router that is based on promises rather than req/res/next.
24
13
  // This is intended to replace the use of express.Router to handle
25
14
  // subsections of the API surface.
@@ -27,8 +16,7 @@ function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { de
27
16
  // themselves use our routing information, without disturbing express
28
17
  // components that external developers may be modifying.
29
18
 
30
- const Layer = require('express/lib/router/layer');
31
-
19
+ const Layer = require('router/lib/layer');
32
20
  function validateParameter(key, value) {
33
21
  if (key == 'className') {
34
22
  if (value.match(/_?[A-Za-z][A-Za-z_0-9]*/)) {
@@ -42,7 +30,6 @@ function validateParameter(key, value) {
42
30
  return value;
43
31
  }
44
32
  }
45
-
46
33
  class PromiseRouter {
47
34
  // Each entry should be an object with:
48
35
  // path: the path to route, in express format
@@ -69,7 +56,6 @@ class PromiseRouter {
69
56
  this.routes.push(route);
70
57
  }
71
58
  }
72
-
73
59
  route(method, path, ...handlers) {
74
60
  switch (method) {
75
61
  case 'POST':
@@ -80,9 +66,7 @@ class PromiseRouter {
80
66
  default:
81
67
  throw 'cannot route method: ' + method;
82
68
  }
83
-
84
69
  let handler = handlers[0];
85
-
86
70
  if (handlers.length > 1) {
87
71
  handler = function (req) {
88
72
  return handlers.reduce((promise, handler) => {
@@ -92,7 +76,6 @@ class PromiseRouter {
92
76
  }, Promise.resolve());
93
77
  };
94
78
  }
95
-
96
79
  this.routes.push({
97
80
  path: path,
98
81
  method: method,
@@ -117,7 +100,10 @@ class PromiseRouter {
117
100
  Object.keys(params).forEach(key => {
118
101
  params[key] = validateParameter(key, params[key]);
119
102
  });
120
- return { params: params, handler: route.handler };
103
+ return {
104
+ params: params,
105
+ handler: route.handler
106
+ };
121
107
  }
122
108
  }
123
109
  }
@@ -131,15 +117,13 @@ class PromiseRouter {
131
117
  });
132
118
  return expressApp;
133
119
  }
134
-
135
120
  expressRouter() {
136
- return this.mountOnto(_express2.default.Router());
121
+ return this.mountOnto(_express.default.Router());
137
122
  }
138
-
139
123
  tryRouteRequest(method, path, request) {
140
124
  var match = this.match(method, path);
141
125
  if (!match) {
142
- throw new _node2.default.Error(_node2.default.Error.INVALID_JSON, 'cannot route ' + method + ' ' + path);
126
+ throw new _node.default.Error(_node.default.Error.INVALID_JSON, 'cannot route ' + method + ' ' + path);
143
127
  }
144
128
  request.params = match.params;
145
129
  return new Promise((resolve, reject) => {
@@ -148,11 +132,11 @@ class PromiseRouter {
148
132
  }
149
133
  }
150
134
 
151
- exports.default = PromiseRouter; // A helper function to make an express handler out of a a promise
135
+ // A helper function to make an express handler out of a a promise
152
136
  // handler.
153
137
  // Express handlers should never throw; if a promise handler throws we
154
138
  // just treat it like it resolved to an error.
155
-
139
+ exports.default = PromiseRouter;
156
140
  function makeExpressHandler(appId, promiseHandler) {
157
141
  return function (req, res, next) {
158
142
  try {
@@ -160,7 +144,7 @@ function makeExpressHandler(appId, promiseHandler) {
160
144
  const body = Object.assign({}, req.body);
161
145
  const method = req.method;
162
146
  const headers = req.headers;
163
- _logger2.default.logRequest({
147
+ _logger.default.logRequest({
164
148
  method,
165
149
  url,
166
150
  headers,
@@ -168,20 +152,25 @@ function makeExpressHandler(appId, promiseHandler) {
168
152
  });
169
153
  promiseHandler(req).then(result => {
170
154
  if (!result.response && !result.location && !result.text) {
171
- _logger2.default.error('the handler did not include a "response" or a "location" field');
155
+ _logger.default.error('the handler did not include a "response" or a "location" field');
172
156
  throw 'control should not get here';
173
157
  }
174
-
175
- _logger2.default.logResponse({ method, url, result });
176
-
158
+ _logger.default.logResponse({
159
+ method,
160
+ url,
161
+ result
162
+ });
177
163
  var status = result.status || 200;
178
164
  res.status(status);
179
-
165
+ if (result.headers) {
166
+ Object.keys(result.headers).forEach(header => {
167
+ res.set(header, result.headers[header]);
168
+ });
169
+ }
180
170
  if (result.text) {
181
171
  res.send(result.text);
182
172
  return;
183
173
  }
184
-
185
174
  if (result.location) {
186
175
  res.set('Location', result.location);
187
176
  // Override the default expressjs response
@@ -191,29 +180,29 @@ function makeExpressHandler(appId, promiseHandler) {
191
180
  return;
192
181
  }
193
182
  }
194
- if (result.headers) {
195
- Object.keys(result.headers).forEach(header => {
196
- res.set(header, result.headers[header]);
197
- });
198
- }
199
183
  res.json(result.response);
200
- }, e => {
201
- _logger2.default.error(`Error generating response. ${(0, _util.inspect)(e)}`, { error: e });
184
+ }, error => {
185
+ next(error);
186
+ }).catch(e => {
187
+ _logger.default.error(`Error generating response. ${(0, _util.inspect)(e)}`, {
188
+ error: e
189
+ });
202
190
  next(e);
203
191
  });
204
192
  } catch (e) {
205
- _logger2.default.error(`Error handling request: ${(0, _util.inspect)(e)}`, { error: e });
193
+ _logger.default.error(`Error handling request: ${(0, _util.inspect)(e)}`, {
194
+ error: e
195
+ });
206
196
  next(e);
207
197
  }
208
198
  };
209
199
  }
210
-
211
200
  function maskSensitiveUrl(req) {
212
201
  let maskUrl = req.originalUrl.toString();
213
202
  const shouldMaskUrl = req.method === 'GET' && req.originalUrl.includes('/login') && !req.originalUrl.includes('classes');
214
203
  if (shouldMaskUrl) {
215
- maskUrl = _logger2.default.maskSensitiveUrl(maskUrl);
204
+ maskUrl = _logger.default.maskSensitiveUrl(maskUrl);
216
205
  }
217
206
  return maskUrl;
218
207
  }
219
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../src/PromiseRouter.js"],"names":["Layer","require","validateParameter","key","value","match","PromiseRouter","constructor","routes","appId","mountRoutes","merge","router","route","push","method","path","handlers","handler","length","req","reduce","promise","then","Promise","resolve","layer","params","Object","keys","forEach","mountOnto","expressApp","toLowerCase","makeExpressHandler","call","expressRouter","express","Router","tryRouteRequest","request","Parse","Error","INVALID_JSON","reject","promiseHandler","res","next","url","maskSensitiveUrl","body","assign","headers","log","logRequest","result","response","location","text","error","logResponse","status","send","set","header","json","e","maskUrl","originalUrl","toString","shouldMaskUrl","includes"],"mappings":";;;;;;AAOA;;;;AACA;;;;AACA;;;;AACA;;;;AAVA;AACA;AACA;AACA;AACA;AACA;;AAMA,MAAMA,QAAQC,QAAQ,0BAAR,CAAd;;AAEA,SAASC,iBAAT,CAA2BC,GAA3B,EAAgCC,KAAhC,EAAuC;AACrC,MAAID,OAAO,WAAX,EAAwB;AACtB,QAAIC,MAAMC,KAAN,CAAY,yBAAZ,CAAJ,EAA4C;AAC1C,aAAOD,KAAP;AACD;AACF,GAJD,MAIO,IAAID,OAAO,UAAX,EAAuB;AAC5B,QAAIC,MAAMC,KAAN,CAAY,cAAZ,CAAJ,EAAiC;AAC/B,aAAOD,KAAP;AACD;AACF,GAJM,MAIA;AACL,WAAOA,KAAP;AACD;AACF;;AAGc,MAAME,aAAN,CAAoB;AACjC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACAC,cAAYC,SAAS,EAArB,EAAyBC,KAAzB,EAAgC;AAC9B,SAAKD,MAAL,GAAcA,MAAd;AACA,SAAKC,KAAL,GAAaA,KAAb;AACA,SAAKC,WAAL;AACD;;AAED;AACA;AACAA,gBAAc,CAAE;;AAEhB;AACAC,QAAMC,MAAN,EAAc;AACZ,SAAK,IAAIC,KAAT,IAAkBD,OAAOJ,MAAzB,EAAiC;AAC/B,WAAKA,MAAL,CAAYM,IAAZ,CAAiBD,KAAjB;AACD;AACF;;AAEDA,QAAME,MAAN,EAAcC,IAAd,EAAoB,GAAGC,QAAvB,EAAiC;AAC/B,YAAOF,MAAP;AACA,WAAK,MAAL;AACA,WAAK,KAAL;AACA,WAAK,KAAL;AACA,WAAK,QAAL;AACE;AACF;AACE,cAAM,0BAA0BA,MAAhC;AAPF;;AAUA,QAAIG,UAAUD,SAAS,CAAT,CAAd;;AAEA,QAAIA,SAASE,MAAT,GAAkB,CAAtB,EAAyB;AACvBD,gBAAU,UAASE,GAAT,EAAc;AACtB,eAAOH,SAASI,MAAT,CAAgB,CAACC,OAAD,EAAUJ,OAAV,KAAsB;AAC3C,iBAAOI,QAAQC,IAAR,CAAa,MAAM;AACxB,mBAAOL,QAAQE,GAAR,CAAP;AACD,WAFM,CAAP;AAGD,SAJM,EAIJI,QAAQC,OAAR,EAJI,CAAP;AAKD,OAND;AAOD;;AAED,SAAKjB,MAAL,CAAYM,IAAZ,CAAiB;AACfE,YAAMA,IADS;AAEfD,cAAQA,MAFO;AAGfG,eAASA,OAHM;AAIfQ,aAAO,IAAI1B,KAAJ,CAAUgB,IAAV,EAAgB,IAAhB,EAAsBE,OAAtB;AAJQ,KAAjB;AAMD;;AAED;AACA;AACA;AACA;AACAb,QAAMU,MAAN,EAAcC,IAAd,EAAoB;AAClB,SAAK,IAAIH,KAAT,IAAkB,KAAKL,MAAvB,EAA+B;AAC7B,UAAIK,MAAME,MAAN,IAAgBA,MAApB,EAA4B;AAC1B;AACD;AACD,YAAMW,QAAQb,MAAMa,KAAN,IAAe,IAAI1B,KAAJ,CAAUa,MAAMG,IAAhB,EAAsB,IAAtB,EAA4BH,MAAMK,OAAlC,CAA7B;AACA,YAAMb,QAAQqB,MAAMrB,KAAN,CAAYW,IAAZ,CAAd;AACA,UAAIX,KAAJ,EAAW;AACT,cAAMsB,SAASD,MAAMC,MAArB;AACAC,eAAOC,IAAP,CAAYF,MAAZ,EAAoBG,OAApB,CAA6B3B,GAAD,IAAS;AACnCwB,iBAAOxB,GAAP,IAAcD,kBAAkBC,GAAlB,EAAuBwB,OAAOxB,GAAP,CAAvB,CAAd;AACD,SAFD;AAGA,eAAO,EAACwB,QAAQA,MAAT,EAAiBT,SAASL,MAAMK,OAAhC,EAAP;AACD;AACF;AACF;;AAED;AACAa,YAAUC,UAAV,EAAsB;AACpB,SAAKxB,MAAL,CAAYsB,OAAZ,CAAqBjB,KAAD,IAAW;AAC7B,YAAME,SAASF,MAAME,MAAN,CAAakB,WAAb,EAAf;AACA,YAAMf,UAAUgB,mBAAmB,KAAKzB,KAAxB,EAA+BI,MAAMK,OAArC,CAAhB;AACAc,iBAAWjB,MAAX,EAAmBoB,IAAnB,CAAwBH,UAAxB,EAAoCnB,MAAMG,IAA1C,EAAgDE,OAAhD;AACD,KAJD;AAKA,WAAOc,UAAP;AACD;;AAEDI,kBAAgB;AACd,WAAO,KAAKL,SAAL,CAAeM,kBAAQC,MAAR,EAAf,CAAP;AACD;;AAEDC,kBAAgBxB,MAAhB,EAAwBC,IAAxB,EAA8BwB,OAA9B,EAAuC;AACrC,QAAInC,QAAQ,KAAKA,KAAL,CAAWU,MAAX,EAAmBC,IAAnB,CAAZ;AACA,QAAI,CAACX,KAAL,EAAY;AACV,YAAM,IAAIoC,eAAMC,KAAV,CACJD,eAAMC,KAAN,CAAYC,YADR,EAEJ,kBAAkB5B,MAAlB,GAA2B,GAA3B,GAAiCC,IAF7B,CAAN;AAGD;AACDwB,YAAQb,MAAR,GAAiBtB,MAAMsB,MAAvB;AACA,WAAO,IAAIH,OAAJ,CAAY,CAACC,OAAD,EAAUmB,MAAV,KAAqB;AACtCvC,YAAMa,OAAN,CAAcsB,OAAd,EAAuBjB,IAAvB,CAA4BE,OAA5B,EAAqCmB,MAArC;AACD,KAFM,CAAP;AAGD;AAxGgC;;kBAAdtC,a,EA2GrB;AACA;AACA;AACA;;AACA,SAAS4B,kBAAT,CAA4BzB,KAA5B,EAAmCoC,cAAnC,EAAmD;AACjD,SAAO,UAASzB,GAAT,EAAc0B,GAAd,EAAmBC,IAAnB,EAAyB;AAC9B,QAAI;AACF,YAAMC,MAAMC,iBAAiB7B,GAAjB,CAAZ;AACA,YAAM8B,OAAOtB,OAAOuB,MAAP,CAAc,EAAd,EAAkB/B,IAAI8B,IAAtB,CAAb;AACA,YAAMnC,SAASK,IAAIL,MAAnB;AACA,YAAMqC,UAAUhC,IAAIgC,OAApB;AACAC,uBAAIC,UAAJ,CAAe;AACbvC,cADa;AAEbiC,WAFa;AAGbI,eAHa;AAIbF;AAJa,OAAf;AAMAL,qBAAezB,GAAf,EAAoBG,IAApB,CAA0BgC,MAAD,IAAY;AACnC,YAAI,CAACA,OAAOC,QAAR,IAAoB,CAACD,OAAOE,QAA5B,IAAwC,CAACF,OAAOG,IAApD,EAA0D;AACxDL,2BAAIM,KAAJ,CAAU,gEAAV;AACA,gBAAM,6BAAN;AACD;;AAEDN,yBAAIO,WAAJ,CAAgB,EAAE7C,MAAF,EAAUiC,GAAV,EAAeO,MAAf,EAAhB;;AAEA,YAAIM,SAASN,OAAOM,MAAP,IAAiB,GAA9B;AACAf,YAAIe,MAAJ,CAAWA,MAAX;;AAEA,YAAIN,OAAOG,IAAX,EAAiB;AACfZ,cAAIgB,IAAJ,CAASP,OAAOG,IAAhB;AACA;AACD;;AAED,YAAIH,OAAOE,QAAX,EAAqB;AACnBX,cAAIiB,GAAJ,CAAQ,UAAR,EAAoBR,OAAOE,QAA3B;AACA;AACA;AACA,cAAI,CAACF,OAAOC,QAAZ,EAAsB;AACpBV,gBAAIgB,IAAJ,CAAS,2BAA2BP,OAAOE,QAA3C;AACA;AACD;AACF;AACD,YAAIF,OAAOH,OAAX,EAAoB;AAClBxB,iBAAOC,IAAP,CAAY0B,OAAOH,OAAnB,EAA4BtB,OAA5B,CAAqCkC,MAAD,IAAY;AAC9ClB,gBAAIiB,GAAJ,CAAQC,MAAR,EAAgBT,OAAOH,OAAP,CAAeY,MAAf,CAAhB;AACD,WAFD;AAGD;AACDlB,YAAImB,IAAJ,CAASV,OAAOC,QAAhB;AACD,OA/BD,EA+BIU,CAAD,IAAO;AACRb,yBAAIM,KAAJ,CAAW,8BAA6B,mBAAQO,CAAR,CAAW,EAAnD,EAAsD,EAACP,OAAOO,CAAR,EAAtD;AACAnB,aAAKmB,CAAL;AACD,OAlCD;AAmCD,KA9CD,CA8CE,OAAOA,CAAP,EAAU;AACVb,uBAAIM,KAAJ,CAAW,2BAA0B,mBAAQO,CAAR,CAAW,EAAhD,EAAmD,EAACP,OAAOO,CAAR,EAAnD;AACAnB,WAAKmB,CAAL;AACD;AACF,GAnDD;AAoDD;;AAGD,SAASjB,gBAAT,CAA0B7B,GAA1B,EAA+B;AAC7B,MAAI+C,UAAU/C,IAAIgD,WAAJ,CAAgBC,QAAhB,EAAd;AACA,QAAMC,gBAAgBlD,IAAIL,MAAJ,KAAe,KAAf,IAAwBK,IAAIgD,WAAJ,CAAgBG,QAAhB,CAAyB,QAAzB,CAAxB,IACC,CAACnD,IAAIgD,WAAJ,CAAgBG,QAAhB,CAAyB,SAAzB,CADxB;AAEA,MAAID,aAAJ,EAAmB;AACjBH,cAAUd,iBAAIJ,gBAAJ,CAAqBkB,OAArB,CAAV;AACD;AACD,SAAOA,OAAP;AACD","file":"PromiseRouter.js","sourcesContent":["// A router that is based on promises rather than req/res/next.\n// This is intended to replace the use of express.Router to handle\n// subsections of the API surface.\n// This will make it easier to have methods like 'batch' that\n// themselves use our routing information, without disturbing express\n// components that external developers may be modifying.\n\nimport Parse     from 'parse/node';\nimport express   from 'express';\nimport log       from './logger';\nimport {inspect} from 'util';\nconst Layer = require('express/lib/router/layer');\n\nfunction validateParameter(key, value) {\n  if (key == 'className') {\n    if (value.match(/_?[A-Za-z][A-Za-z_0-9]*/)) {\n      return value;\n    }\n  } else if (key == 'objectId') {\n    if (value.match(/[A-Za-z0-9]+/)) {\n      return value;\n    }\n  } else {\n    return value;\n  }\n}\n\n\nexport default class PromiseRouter {\n  // Each entry should be an object with:\n  // path: the path to route, in express format\n  // method: the HTTP method that this route handles.\n  //   Must be one of: POST, GET, PUT, DELETE\n  // handler: a function that takes request, and returns a promise.\n  //   Successful handlers should resolve to an object with fields:\n  //     status: optional. the http status code. defaults to 200\n  //     response: a json object with the content of the response\n  //     location: optional. a location header\n  constructor(routes = [], appId) {\n    this.routes = routes;\n    this.appId = appId;\n    this.mountRoutes();\n  }\n\n  // Leave the opportunity to\n  // subclasses to mount their routes by overriding\n  mountRoutes() {}\n\n  // Merge the routes into this one\n  merge(router) {\n    for (var route of router.routes) {\n      this.routes.push(route);\n    }\n  }\n\n  route(method, path, ...handlers) {\n    switch(method) {\n    case 'POST':\n    case 'GET':\n    case 'PUT':\n    case 'DELETE':\n      break;\n    default:\n      throw 'cannot route method: ' + method;\n    }\n\n    let handler = handlers[0];\n\n    if (handlers.length > 1) {\n      handler = function(req) {\n        return handlers.reduce((promise, handler) => {\n          return promise.then(() => {\n            return handler(req);\n          });\n        }, Promise.resolve());\n      }\n    }\n\n    this.routes.push({\n      path: path,\n      method: method,\n      handler: handler,\n      layer: new Layer(path, null, handler)\n    });\n  }\n\n  // Returns an object with:\n  //   handler: the handler that should deal with this request\n  //   params: any :-params that got parsed from the path\n  // Returns undefined if there is no match.\n  match(method, path) {\n    for (var route of this.routes) {\n      if (route.method != method) {\n        continue;\n      }\n      const layer = route.layer || new Layer(route.path, null, route.handler);\n      const match = layer.match(path);\n      if (match) {\n        const params = layer.params;\n        Object.keys(params).forEach((key) => {\n          params[key] = validateParameter(key, params[key]);\n        });\n        return {params: params, handler: route.handler};\n      }\n    }\n  }\n\n  // Mount the routes on this router onto an express app (or express router)\n  mountOnto(expressApp) {\n    this.routes.forEach((route) => {\n      const method = route.method.toLowerCase();\n      const handler = makeExpressHandler(this.appId, route.handler);\n      expressApp[method].call(expressApp, route.path, handler);\n    });\n    return expressApp;\n  }\n\n  expressRouter() {\n    return this.mountOnto(express.Router());\n  }\n\n  tryRouteRequest(method, path, request) {\n    var match = this.match(method, path);\n    if (!match) {\n      throw new Parse.Error(\n        Parse.Error.INVALID_JSON,\n        'cannot route ' + method + ' ' + path);\n    }\n    request.params = match.params;\n    return new Promise((resolve, reject) => {\n      match.handler(request).then(resolve, reject);\n    });\n  }\n}\n\n// A helper function to make an express handler out of a a promise\n// handler.\n// Express handlers should never throw; if a promise handler throws we\n// just treat it like it resolved to an error.\nfunction makeExpressHandler(appId, promiseHandler) {\n  return function(req, res, next) {\n    try {\n      const url = maskSensitiveUrl(req);\n      const body = Object.assign({}, req.body);\n      const method = req.method;\n      const headers = req.headers;\n      log.logRequest({\n        method,\n        url,\n        headers,\n        body\n      });\n      promiseHandler(req).then((result) => {\n        if (!result.response && !result.location && !result.text) {\n          log.error('the handler did not include a \"response\" or a \"location\" field');\n          throw 'control should not get here';\n        }\n\n        log.logResponse({ method, url, result });\n\n        var status = result.status || 200;\n        res.status(status);\n\n        if (result.text) {\n          res.send(result.text);\n          return;\n        }\n\n        if (result.location) {\n          res.set('Location', result.location);\n          // Override the default expressjs response\n          // as it double encodes %encoded chars in URL\n          if (!result.response) {\n            res.send('Found. Redirecting to ' + result.location);\n            return;\n          }\n        }\n        if (result.headers) {\n          Object.keys(result.headers).forEach((header) => {\n            res.set(header, result.headers[header]);\n          })\n        }\n        res.json(result.response);\n      }, (e) => {\n        log.error(`Error generating response. ${inspect(e)}`, {error: e});\n        next(e);\n      });\n    } catch (e) {\n      log.error(`Error handling request: ${inspect(e)}`, {error: e});\n      next(e);\n    }\n  }\n}\n\n\nfunction maskSensitiveUrl(req) {\n  let maskUrl = req.originalUrl.toString();\n  const shouldMaskUrl = req.method === 'GET' && req.originalUrl.includes('/login')\n                      && !req.originalUrl.includes('classes');\n  if (shouldMaskUrl) {\n    maskUrl = log.maskSensitiveUrl(maskUrl);\n  }\n  return maskUrl;\n}\n"]}
208
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_node","_interopRequireDefault","require","_express","_logger","_util","e","__esModule","default","Layer","validateParameter","key","value","match","PromiseRouter","constructor","routes","appId","mountRoutes","merge","router","route","push","method","path","handlers","handler","length","req","reduce","promise","then","Promise","resolve","layer","params","Object","keys","forEach","mountOnto","expressApp","toLowerCase","makeExpressHandler","call","expressRouter","express","Router","tryRouteRequest","request","Parse","Error","INVALID_JSON","reject","exports","promiseHandler","res","next","url","maskSensitiveUrl","body","assign","headers","log","logRequest","result","response","location","text","error","logResponse","status","header","set","send","json","catch","inspect","maskUrl","originalUrl","toString","shouldMaskUrl","includes"],"sources":["../src/PromiseRouter.js"],"sourcesContent":["// A router that is based on promises rather than req/res/next.\n// This is intended to replace the use of express.Router to handle\n// subsections of the API surface.\n// This will make it easier to have methods like 'batch' that\n// themselves use our routing information, without disturbing express\n// components that external developers may be modifying.\n\nimport Parse from 'parse/node';\nimport express from 'express';\nimport log from './logger';\nimport { inspect } from 'util';\nconst Layer = require('router/lib/layer');\n\nfunction validateParameter(key, value) {\n  if (key == 'className') {\n    if (value.match(/_?[A-Za-z][A-Za-z_0-9]*/)) {\n      return value;\n    }\n  } else if (key == 'objectId') {\n    if (value.match(/[A-Za-z0-9]+/)) {\n      return value;\n    }\n  } else {\n    return value;\n  }\n}\n\nexport default class PromiseRouter {\n  // Each entry should be an object with:\n  // path: the path to route, in express format\n  // method: the HTTP method that this route handles.\n  //   Must be one of: POST, GET, PUT, DELETE\n  // handler: a function that takes request, and returns a promise.\n  //   Successful handlers should resolve to an object with fields:\n  //     status: optional. the http status code. defaults to 200\n  //     response: a json object with the content of the response\n  //     location: optional. a location header\n  constructor(routes = [], appId) {\n    this.routes = routes;\n    this.appId = appId;\n    this.mountRoutes();\n  }\n\n  // Leave the opportunity to\n  // subclasses to mount their routes by overriding\n  mountRoutes() {}\n\n  // Merge the routes into this one\n  merge(router) {\n    for (var route of router.routes) {\n      this.routes.push(route);\n    }\n  }\n\n  route(method, path, ...handlers) {\n    switch (method) {\n      case 'POST':\n      case 'GET':\n      case 'PUT':\n      case 'DELETE':\n        break;\n      default:\n        throw 'cannot route method: ' + method;\n    }\n\n    let handler = handlers[0];\n\n    if (handlers.length > 1) {\n      handler = function (req) {\n        return handlers.reduce((promise, handler) => {\n          return promise.then(() => {\n            return handler(req);\n          });\n        }, Promise.resolve());\n      };\n    }\n\n    this.routes.push({\n      path: path,\n      method: method,\n      handler: handler,\n      layer: new Layer(path, null, handler),\n    });\n  }\n\n  // Returns an object with:\n  //   handler: the handler that should deal with this request\n  //   params: any :-params that got parsed from the path\n  // Returns undefined if there is no match.\n  match(method, path) {\n    for (var route of this.routes) {\n      if (route.method != method) {\n        continue;\n      }\n      const layer = route.layer || new Layer(route.path, null, route.handler);\n      const match = layer.match(path);\n      if (match) {\n        const params = layer.params;\n        Object.keys(params).forEach(key => {\n          params[key] = validateParameter(key, params[key]);\n        });\n        return { params: params, handler: route.handler };\n      }\n    }\n  }\n\n  // Mount the routes on this router onto an express app (or express router)\n  mountOnto(expressApp) {\n    this.routes.forEach(route => {\n      const method = route.method.toLowerCase();\n      const handler = makeExpressHandler(this.appId, route.handler);\n      expressApp[method].call(expressApp, route.path, handler);\n    });\n    return expressApp;\n  }\n\n  expressRouter() {\n    return this.mountOnto(express.Router());\n  }\n\n  tryRouteRequest(method, path, request) {\n    var match = this.match(method, path);\n    if (!match) {\n      throw new Parse.Error(Parse.Error.INVALID_JSON, 'cannot route ' + method + ' ' + path);\n    }\n    request.params = match.params;\n    return new Promise((resolve, reject) => {\n      match.handler(request).then(resolve, reject);\n    });\n  }\n}\n\n// A helper function to make an express handler out of a a promise\n// handler.\n// Express handlers should never throw; if a promise handler throws we\n// just treat it like it resolved to an error.\nfunction makeExpressHandler(appId, promiseHandler) {\n  return function (req, res, next) {\n    try {\n      const url = maskSensitiveUrl(req);\n      const body = Object.assign({}, req.body);\n      const method = req.method;\n      const headers = req.headers;\n      log.logRequest({\n        method,\n        url,\n        headers,\n        body,\n      });\n      promiseHandler(req)\n        .then(\n          result => {\n            if (!result.response && !result.location && !result.text) {\n              log.error('the handler did not include a \"response\" or a \"location\" field');\n              throw 'control should not get here';\n            }\n\n            log.logResponse({ method, url, result });\n\n            var status = result.status || 200;\n            res.status(status);\n\n            if (result.headers) {\n              Object.keys(result.headers).forEach(header => {\n                res.set(header, result.headers[header]);\n              });\n            }\n\n            if (result.text) {\n              res.send(result.text);\n              return;\n            }\n\n            if (result.location) {\n              res.set('Location', result.location);\n              // Override the default expressjs response\n              // as it double encodes %encoded chars in URL\n              if (!result.response) {\n                res.send('Found. Redirecting to ' + result.location);\n                return;\n              }\n            }\n            res.json(result.response);\n          },\n          error => {\n            next(error);\n          }\n        )\n        .catch(e => {\n          log.error(`Error generating response. ${inspect(e)}`, { error: e });\n          next(e);\n        });\n    } catch (e) {\n      log.error(`Error handling request: ${inspect(e)}`, { error: e });\n      next(e);\n    }\n  };\n}\n\nfunction maskSensitiveUrl(req) {\n  let maskUrl = req.originalUrl.toString();\n  const shouldMaskUrl =\n    req.method === 'GET' &&\n    req.originalUrl.includes('/login') &&\n    !req.originalUrl.includes('classes');\n  if (shouldMaskUrl) {\n    maskUrl = log.maskSensitiveUrl(maskUrl);\n  }\n  return maskUrl;\n}\n"],"mappings":"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","ignoreList":[]}
@@ -1,29 +1,17 @@
1
- 'use strict';
1
+ "use strict";
2
2
 
3
3
  Object.defineProperty(exports, "__esModule", {
4
4
  value: true
5
5
  });
6
- exports.PushQueue = undefined;
7
-
8
- var _ParseMessageQueue = require('../ParseMessageQueue');
9
-
10
- var _rest = require('../rest');
11
-
12
- var _rest2 = _interopRequireDefault(_rest);
13
-
14
- var _utils = require('./utils');
15
-
16
- var _node = require('parse/node');
17
-
18
- var _node2 = _interopRequireDefault(_node);
19
-
20
- function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
21
-
6
+ exports.PushQueue = void 0;
7
+ var _ParseMessageQueue = require("../ParseMessageQueue");
8
+ var _rest = _interopRequireDefault(require("../rest"));
9
+ var _utils = require("./utils");
10
+ var _node = _interopRequireDefault(require("parse/node"));
11
+ function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
22
12
  const PUSH_CHANNEL = 'parse-server-push';
23
13
  const DEFAULT_BATCH_SIZE = 100;
24
-
25
14
  class PushQueue {
26
-
27
15
  // config object of the publisher, right now it only contains the redisURL,
28
16
  // but we may extend it later.
29
17
  constructor(config = {}) {
@@ -31,36 +19,42 @@ class PushQueue {
31
19
  this.batchSize = config.batchSize || DEFAULT_BATCH_SIZE;
32
20
  this.parsePublisher = _ParseMessageQueue.ParseMessageQueue.createPublisher(config);
33
21
  }
34
-
35
22
  static defaultPushChannel() {
36
- return `${_node2.default.applicationId}-${PUSH_CHANNEL}`;
23
+ return `${_node.default.applicationId}-${PUSH_CHANNEL}`;
37
24
  }
38
-
39
25
  enqueue(body, where, config, auth, pushStatus) {
40
26
  const limit = this.batchSize;
41
-
42
27
  where = (0, _utils.applyDeviceTokenExists)(where);
43
28
 
44
29
  // Order by objectId so no impact on the DB
45
30
  const order = 'objectId';
46
31
  return Promise.resolve().then(() => {
47
- return _rest2.default.find(config, auth, '_Installation', where, { limit: 0, count: true });
48
- }).then(({ results, count }) => {
32
+ return _rest.default.find(config, auth, '_Installation', where, {
33
+ limit: 0,
34
+ count: true
35
+ });
36
+ }).then(({
37
+ results,
38
+ count
39
+ }) => {
49
40
  if (!results || count == 0) {
50
41
  return pushStatus.complete();
51
42
  }
52
43
  pushStatus.setRunning(Math.ceil(count / limit));
53
44
  let skip = 0;
54
45
  while (skip < count) {
55
- const query = { where,
46
+ const query = {
47
+ where,
56
48
  limit,
57
49
  skip,
58
- order };
59
-
50
+ order
51
+ };
60
52
  const pushWorkItem = {
61
53
  body,
62
54
  query,
63
- pushStatus: { objectId: pushStatus.objectId },
55
+ pushStatus: {
56
+ objectId: pushStatus.objectId
57
+ },
64
58
  applicationId: config.applicationId
65
59
  };
66
60
  this.parsePublisher.publish(this.channel, JSON.stringify(pushWorkItem));
@@ -70,4 +64,4 @@ class PushQueue {
70
64
  }
71
65
  }
72
66
  exports.PushQueue = PushQueue;
73
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9QdXNoL1B1c2hRdWV1ZS5qcyJdLCJuYW1lcyI6WyJQVVNIX0NIQU5ORUwiLCJERUZBVUxUX0JBVENIX1NJWkUiLCJQdXNoUXVldWUiLCJjb25zdHJ1Y3RvciIsImNvbmZpZyIsImNoYW5uZWwiLCJkZWZhdWx0UHVzaENoYW5uZWwiLCJiYXRjaFNpemUiLCJwYXJzZVB1Ymxpc2hlciIsIlBhcnNlTWVzc2FnZVF1ZXVlIiwiY3JlYXRlUHVibGlzaGVyIiwiUGFyc2UiLCJhcHBsaWNhdGlvbklkIiwiZW5xdWV1ZSIsImJvZHkiLCJ3aGVyZSIsImF1dGgiLCJwdXNoU3RhdHVzIiwibGltaXQiLCJvcmRlciIsIlByb21pc2UiLCJyZXNvbHZlIiwidGhlbiIsInJlc3QiLCJmaW5kIiwiY291bnQiLCJyZXN1bHRzIiwiY29tcGxldGUiLCJzZXRSdW5uaW5nIiwiTWF0aCIsImNlaWwiLCJza2lwIiwicXVlcnkiLCJwdXNoV29ya0l0ZW0iLCJvYmplY3RJZCIsInB1Ymxpc2giLCJKU09OIiwic3RyaW5naWZ5Il0sIm1hcHBpbmdzIjoiOzs7Ozs7O0FBQUE7O0FBQ0E7Ozs7QUFDQTs7QUFDQTs7Ozs7O0FBRUEsTUFBTUEsZUFBZSxtQkFBckI7QUFDQSxNQUFNQyxxQkFBcUIsR0FBM0I7O0FBRU8sTUFBTUMsU0FBTixDQUFnQjs7QUFLckI7QUFDQTtBQUNBQyxjQUFZQyxTQUFjLEVBQTFCLEVBQThCO0FBQzVCLFNBQUtDLE9BQUwsR0FBZUQsT0FBT0MsT0FBUCxJQUFrQkgsVUFBVUksa0JBQVYsRUFBakM7QUFDQSxTQUFLQyxTQUFMLEdBQWlCSCxPQUFPRyxTQUFQLElBQW9CTixrQkFBckM7QUFDQSxTQUFLTyxjQUFMLEdBQXNCQyxxQ0FBa0JDLGVBQWxCLENBQWtDTixNQUFsQyxDQUF0QjtBQUNEOztBQUVELFNBQU9FLGtCQUFQLEdBQTRCO0FBQzFCLFdBQVEsR0FBRUssZUFBTUMsYUFBYyxJQUFHWixZQUFhLEVBQTlDO0FBQ0Q7O0FBRURhLFVBQVFDLElBQVIsRUFBY0MsS0FBZCxFQUFxQlgsTUFBckIsRUFBNkJZLElBQTdCLEVBQW1DQyxVQUFuQyxFQUErQztBQUM3QyxVQUFNQyxRQUFRLEtBQUtYLFNBQW5COztBQUVBUSxZQUFRLG1DQUF1QkEsS0FBdkIsQ0FBUjs7QUFFQTtBQUNBLFVBQU1JLFFBQVEsVUFBZDtBQUNBLFdBQU9DLFFBQVFDLE9BQVIsR0FBa0JDLElBQWxCLENBQXVCLE1BQU07QUFDbEMsYUFBT0MsZUFBS0MsSUFBTCxDQUFVcEIsTUFBVixFQUNMWSxJQURLLEVBRUwsZUFGSyxFQUdMRCxLQUhLLEVBSUwsRUFBQ0csT0FBTyxDQUFSLEVBQVdPLE9BQU8sSUFBbEIsRUFKSyxDQUFQO0FBS0QsS0FOTSxFQU1KSCxJQU5JLENBTUMsQ0FBQyxFQUFDSSxPQUFELEVBQVVELEtBQVYsRUFBRCxLQUFzQjtBQUM1QixVQUFJLENBQUNDLE9BQUQsSUFBWUQsU0FBUyxDQUF6QixFQUE0QjtBQUMxQixlQUFPUixXQUFXVSxRQUFYLEVBQVA7QUFDRDtBQUNEVixpQkFBV1csVUFBWCxDQUFzQkMsS0FBS0MsSUFBTCxDQUFVTCxRQUFRUCxLQUFsQixDQUF0QjtBQUNBLFVBQUlhLE9BQU8sQ0FBWDtBQUNBLGFBQU9BLE9BQU9OLEtBQWQsRUFBcUI7QUFDbkIsY0FBTU8sUUFBUSxFQUFFakIsS0FBRjtBQUNaRyxlQURZO0FBRVphLGNBRlk7QUFHWlosZUFIWSxFQUFkOztBQUtBLGNBQU1jLGVBQWU7QUFDbkJuQixjQURtQjtBQUVuQmtCLGVBRm1CO0FBR25CZixzQkFBWSxFQUFFaUIsVUFBVWpCLFdBQVdpQixRQUF2QixFQUhPO0FBSW5CdEIseUJBQWVSLE9BQU9RO0FBSkgsU0FBckI7QUFNQSxhQUFLSixjQUFMLENBQW9CMkIsT0FBcEIsQ0FBNEIsS0FBSzlCLE9BQWpDLEVBQTBDK0IsS0FBS0MsU0FBTCxDQUFlSixZQUFmLENBQTFDO0FBQ0FGLGdCQUFRYixLQUFSO0FBQ0Q7QUFDRixLQTNCTSxDQUFQO0FBNEJEO0FBcERvQjtRQUFWaEIsUyxHQUFBQSxTIiwiZmlsZSI6IlB1c2hRdWV1ZS5qcyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IFBhcnNlTWVzc2FnZVF1ZXVlIH0gICAgICBmcm9tICcuLi9QYXJzZU1lc3NhZ2VRdWV1ZSc7XG5pbXBvcnQgcmVzdCAgICAgICAgICAgICAgICAgICAgICAgZnJvbSAnLi4vcmVzdCc7XG5pbXBvcnQgeyBhcHBseURldmljZVRva2VuRXhpc3RzIH0gZnJvbSAnLi91dGlscyc7XG5pbXBvcnQgUGFyc2UgZnJvbSAncGFyc2Uvbm9kZSc7XG5cbmNvbnN0IFBVU0hfQ0hBTk5FTCA9ICdwYXJzZS1zZXJ2ZXItcHVzaCc7XG5jb25zdCBERUZBVUxUX0JBVENIX1NJWkUgPSAxMDA7XG5cbmV4cG9ydCBjbGFzcyBQdXNoUXVldWUge1xuICBwYXJzZVB1Ymxpc2hlcjogT2JqZWN0O1xuICBjaGFubmVsOiBTdHJpbmc7XG4gIGJhdGNoU2l6ZTogTnVtYmVyO1xuXG4gIC8vIGNvbmZpZyBvYmplY3Qgb2YgdGhlIHB1Ymxpc2hlciwgcmlnaHQgbm93IGl0IG9ubHkgY29udGFpbnMgdGhlIHJlZGlzVVJMLFxuICAvLyBidXQgd2UgbWF5IGV4dGVuZCBpdCBsYXRlci5cbiAgY29uc3RydWN0b3IoY29uZmlnOiBhbnkgPSB7fSkge1xuICAgIHRoaXMuY2hhbm5lbCA9IGNvbmZpZy5jaGFubmVsIHx8IFB1c2hRdWV1ZS5kZWZhdWx0UHVzaENoYW5uZWwoKTtcbiAgICB0aGlzLmJhdGNoU2l6ZSA9IGNvbmZpZy5iYXRjaFNpemUgfHwgREVGQVVMVF9CQVRDSF9TSVpFO1xuICAgIHRoaXMucGFyc2VQdWJsaXNoZXIgPSBQYXJzZU1lc3NhZ2VRdWV1ZS5jcmVhdGVQdWJsaXNoZXIoY29uZmlnKTtcbiAgfVxuXG4gIHN0YXRpYyBkZWZhdWx0UHVzaENoYW5uZWwoKSB7XG4gICAgcmV0dXJuIGAke1BhcnNlLmFwcGxpY2F0aW9uSWR9LSR7UFVTSF9DSEFOTkVMfWA7XG4gIH1cblxuICBlbnF1ZXVlKGJvZHksIHdoZXJlLCBjb25maWcsIGF1dGgsIHB1c2hTdGF0dXMpIHtcbiAgICBjb25zdCBsaW1pdCA9IHRoaXMuYmF0Y2hTaXplO1xuXG4gICAgd2hlcmUgPSBhcHBseURldmljZVRva2VuRXhpc3RzKHdoZXJlKTtcblxuICAgIC8vIE9yZGVyIGJ5IG9iamVjdElkIHNvIG5vIGltcGFjdCBvbiB0aGUgREJcbiAgICBjb25zdCBvcmRlciA9ICdvYmplY3RJZCc7XG4gICAgcmV0dXJuIFByb21pc2UucmVzb2x2ZSgpLnRoZW4oKCkgPT4ge1xuICAgICAgcmV0dXJuIHJlc3QuZmluZChjb25maWcsXG4gICAgICAgIGF1dGgsXG4gICAgICAgICdfSW5zdGFsbGF0aW9uJyxcbiAgICAgICAgd2hlcmUsXG4gICAgICAgIHtsaW1pdDogMCwgY291bnQ6IHRydWV9KTtcbiAgICB9KS50aGVuKCh7cmVzdWx0cywgY291bnR9KSA9PiB7XG4gICAgICBpZiAoIXJlc3VsdHMgfHwgY291bnQgPT0gMCkge1xuICAgICAgICByZXR1cm4gcHVzaFN0YXR1cy5jb21wbGV0ZSgpO1xuICAgICAgfVxuICAgICAgcHVzaFN0YXR1cy5zZXRSdW5uaW5nKE1hdGguY2VpbChjb3VudCAvIGxpbWl0KSk7XG4gICAgICBsZXQgc2tpcCA9IDA7XG4gICAgICB3aGlsZSAoc2tpcCA8IGNvdW50KSB7XG4gICAgICAgIGNvbnN0IHF1ZXJ5ID0geyB3aGVyZSxcbiAgICAgICAgICBsaW1pdCxcbiAgICAgICAgICBza2lwLFxuICAgICAgICAgIG9yZGVyIH07XG5cbiAgICAgICAgY29uc3QgcHVzaFdvcmtJdGVtID0ge1xuICAgICAgICAgIGJvZHksXG4gICAgICAgICAgcXVlcnksXG4gICAgICAgICAgcHVzaFN0YXR1czogeyBvYmplY3RJZDogcHVzaFN0YXR1cy5vYmplY3RJZCB9LFxuICAgICAgICAgIGFwcGxpY2F0aW9uSWQ6IGNvbmZpZy5hcHBsaWNhdGlvbklkXG4gICAgICAgIH1cbiAgICAgICAgdGhpcy5wYXJzZVB1Ymxpc2hlci5wdWJsaXNoKHRoaXMuY2hhbm5lbCwgSlNPTi5zdHJpbmdpZnkocHVzaFdvcmtJdGVtKSk7XG4gICAgICAgIHNraXAgKz0gbGltaXQ7XG4gICAgICB9XG4gICAgfSk7XG4gIH1cbn1cbiJdfQ==
67
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJfUGFyc2VNZXNzYWdlUXVldWUiLCJyZXF1aXJlIiwiX3Jlc3QiLCJfaW50ZXJvcFJlcXVpcmVEZWZhdWx0IiwiX3V0aWxzIiwiX25vZGUiLCJlIiwiX19lc01vZHVsZSIsImRlZmF1bHQiLCJQVVNIX0NIQU5ORUwiLCJERUZBVUxUX0JBVENIX1NJWkUiLCJQdXNoUXVldWUiLCJjb25zdHJ1Y3RvciIsImNvbmZpZyIsImNoYW5uZWwiLCJkZWZhdWx0UHVzaENoYW5uZWwiLCJiYXRjaFNpemUiLCJwYXJzZVB1Ymxpc2hlciIsIlBhcnNlTWVzc2FnZVF1ZXVlIiwiY3JlYXRlUHVibGlzaGVyIiwiUGFyc2UiLCJhcHBsaWNhdGlvbklkIiwiZW5xdWV1ZSIsImJvZHkiLCJ3aGVyZSIsImF1dGgiLCJwdXNoU3RhdHVzIiwibGltaXQiLCJhcHBseURldmljZVRva2VuRXhpc3RzIiwib3JkZXIiLCJQcm9taXNlIiwicmVzb2x2ZSIsInRoZW4iLCJyZXN0IiwiZmluZCIsImNvdW50IiwicmVzdWx0cyIsImNvbXBsZXRlIiwic2V0UnVubmluZyIsIk1hdGgiLCJjZWlsIiwic2tpcCIsInF1ZXJ5IiwicHVzaFdvcmtJdGVtIiwib2JqZWN0SWQiLCJwdWJsaXNoIiwiSlNPTiIsInN0cmluZ2lmeSIsImV4cG9ydHMiXSwic291cmNlcyI6WyIuLi8uLi9zcmMvUHVzaC9QdXNoUXVldWUuanMiXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgUGFyc2VNZXNzYWdlUXVldWUgfSBmcm9tICcuLi9QYXJzZU1lc3NhZ2VRdWV1ZSc7XG5pbXBvcnQgcmVzdCBmcm9tICcuLi9yZXN0JztcbmltcG9ydCB7IGFwcGx5RGV2aWNlVG9rZW5FeGlzdHMgfSBmcm9tICcuL3V0aWxzJztcbmltcG9ydCBQYXJzZSBmcm9tICdwYXJzZS9ub2RlJztcblxuY29uc3QgUFVTSF9DSEFOTkVMID0gJ3BhcnNlLXNlcnZlci1wdXNoJztcbmNvbnN0IERFRkFVTFRfQkFUQ0hfU0laRSA9IDEwMDtcblxuZXhwb3J0IGNsYXNzIFB1c2hRdWV1ZSB7XG4gIHBhcnNlUHVibGlzaGVyOiBPYmplY3Q7XG4gIGNoYW5uZWw6IFN0cmluZztcbiAgYmF0Y2hTaXplOiBOdW1iZXI7XG5cbiAgLy8gY29uZmlnIG9iamVjdCBvZiB0aGUgcHVibGlzaGVyLCByaWdodCBub3cgaXQgb25seSBjb250YWlucyB0aGUgcmVkaXNVUkwsXG4gIC8vIGJ1dCB3ZSBtYXkgZXh0ZW5kIGl0IGxhdGVyLlxuICBjb25zdHJ1Y3Rvcihjb25maWc6IGFueSA9IHt9KSB7XG4gICAgdGhpcy5jaGFubmVsID0gY29uZmlnLmNoYW5uZWwgfHwgUHVzaFF1ZXVlLmRlZmF1bHRQdXNoQ2hhbm5lbCgpO1xuICAgIHRoaXMuYmF0Y2hTaXplID0gY29uZmlnLmJhdGNoU2l6ZSB8fCBERUZBVUxUX0JBVENIX1NJWkU7XG4gICAgdGhpcy5wYXJzZVB1Ymxpc2hlciA9IFBhcnNlTWVzc2FnZVF1ZXVlLmNyZWF0ZVB1Ymxpc2hlcihjb25maWcpO1xuICB9XG5cbiAgc3RhdGljIGRlZmF1bHRQdXNoQ2hhbm5lbCgpIHtcbiAgICByZXR1cm4gYCR7UGFyc2UuYXBwbGljYXRpb25JZH0tJHtQVVNIX0NIQU5ORUx9YDtcbiAgfVxuXG4gIGVucXVldWUoYm9keSwgd2hlcmUsIGNvbmZpZywgYXV0aCwgcHVzaFN0YXR1cykge1xuICAgIGNvbnN0IGxpbWl0ID0gdGhpcy5iYXRjaFNpemU7XG5cbiAgICB3aGVyZSA9IGFwcGx5RGV2aWNlVG9rZW5FeGlzdHMod2hlcmUpO1xuXG4gICAgLy8gT3JkZXIgYnkgb2JqZWN0SWQgc28gbm8gaW1wYWN0IG9uIHRoZSBEQlxuICAgIGNvbnN0IG9yZGVyID0gJ29iamVjdElkJztcbiAgICByZXR1cm4gUHJvbWlzZS5yZXNvbHZlKClcbiAgICAgIC50aGVuKCgpID0+IHtcbiAgICAgICAgcmV0dXJuIHJlc3QuZmluZChjb25maWcsIGF1dGgsICdfSW5zdGFsbGF0aW9uJywgd2hlcmUsIHtcbiAgICAgICAgICBsaW1pdDogMCxcbiAgICAgICAgICBjb3VudDogdHJ1ZSxcbiAgICAgICAgfSk7XG4gICAgICB9KVxuICAgICAgLnRoZW4oKHsgcmVzdWx0cywgY291bnQgfSkgPT4ge1xuICAgICAgICBpZiAoIXJlc3VsdHMgfHwgY291bnQgPT0gMCkge1xuICAgICAgICAgIHJldHVybiBwdXNoU3RhdHVzLmNvbXBsZXRlKCk7XG4gICAgICAgIH1cbiAgICAgICAgcHVzaFN0YXR1cy5zZXRSdW5uaW5nKE1hdGguY2VpbChjb3VudCAvIGxpbWl0KSk7XG4gICAgICAgIGxldCBza2lwID0gMDtcbiAgICAgICAgd2hpbGUgKHNraXAgPCBjb3VudCkge1xuICAgICAgICAgIGNvbnN0IHF1ZXJ5ID0ge1xuICAgICAgICAgICAgd2hlcmUsXG4gICAgICAgICAgICBsaW1pdCxcbiAgICAgICAgICAgIHNraXAsXG4gICAgICAgICAgICBvcmRlcixcbiAgICAgICAgICB9O1xuXG4gICAgICAgICAgY29uc3QgcHVzaFdvcmtJdGVtID0ge1xuICAgICAgICAgICAgYm9keSxcbiAgICAgICAgICAgIHF1ZXJ5LFxuICAgICAgICAgICAgcHVzaFN0YXR1czogeyBvYmplY3RJZDogcHVzaFN0YXR1cy5vYmplY3RJZCB9LFxuICAgICAgICAgICAgYXBwbGljYXRpb25JZDogY29uZmlnLmFwcGxpY2F0aW9uSWQsXG4gICAgICAgICAgfTtcbiAgICAgICAgICB0aGlzLnBhcnNlUHVibGlzaGVyLnB1Ymxpc2godGhpcy5jaGFubmVsLCBKU09OLnN0cmluZ2lmeShwdXNoV29ya0l0ZW0pKTtcbiAgICAgICAgICBza2lwICs9IGxpbWl0O1xuICAgICAgICB9XG4gICAgICB9KTtcbiAgfVxufVxuIl0sIm1hcHBpbmdzIjoiOzs7Ozs7QUFBQSxJQUFBQSxrQkFBQSxHQUFBQyxPQUFBO0FBQ0EsSUFBQUMsS0FBQSxHQUFBQyxzQkFBQSxDQUFBRixPQUFBO0FBQ0EsSUFBQUcsTUFBQSxHQUFBSCxPQUFBO0FBQ0EsSUFBQUksS0FBQSxHQUFBRixzQkFBQSxDQUFBRixP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