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
package/lib/RestQuery.js CHANGED
@@ -1,4 +1,4 @@
1
- 'use strict';
1
+ "use strict";
2
2
 
3
3
  // An object that encapsulates everything we need to run a 'find'
4
4
  // operation, encoded in the REST API format.
@@ -6,8 +6,17 @@
6
6
  var SchemaController = require('./Controllers/SchemaController');
7
7
  var Parse = require('parse/node').Parse;
8
8
  const triggers = require('./triggers');
9
+ const {
10
+ continueWhile
11
+ } = require('parse/lib/node/promiseUtils');
12
+ const AlwaysSelectedKeys = ['objectId', 'createdAt', 'updatedAt', 'ACL'];
13
+ const {
14
+ enforceRoleSecurity
15
+ } = require('./SharedRest');
16
+ const {
17
+ createSanitizedError
18
+ } = require('./Error');
9
19
 
10
- const AlwaysSelectedKeys = ['objectId', 'createdAt', 'updatedAt'];
11
20
  // restOptions can include:
12
21
  // skip
13
22
  // limit
@@ -15,27 +24,86 @@ const AlwaysSelectedKeys = ['objectId', 'createdAt', 'updatedAt'];
15
24
  // count
16
25
  // include
17
26
  // keys
27
+ // excludeKeys
18
28
  // redirectClassNameForKey
19
- function RestQuery(config, auth, className, restWhere = {}, restOptions = {}, clientSDK) {
20
-
29
+ // readPreference
30
+ // includeReadPreference
31
+ // subqueryReadPreference
32
+ /**
33
+ * Use to perform a query on a class. It will run security checks and triggers.
34
+ * @param options
35
+ * @param options.method {RestQuery.Method} The type of query to perform
36
+ * @param options.config {ParseServerConfiguration} The server configuration
37
+ * @param options.auth {Auth} The auth object for the request
38
+ * @param options.className {string} The name of the class to query
39
+ * @param options.restWhere {object} The where object for the query
40
+ * @param options.restOptions {object} The options object for the query
41
+ * @param options.clientSDK {string} The client SDK that is performing the query
42
+ * @param options.runAfterFind {boolean} Whether to run the afterFind trigger
43
+ * @param options.runBeforeFind {boolean} Whether to run the beforeFind trigger
44
+ * @param options.context {object} The context object for the query
45
+ * @returns {Promise<_UnsafeRestQuery>} A promise that is resolved with the _UnsafeRestQuery object
46
+ */
47
+ async function RestQuery({
48
+ method,
49
+ config,
50
+ auth,
51
+ className,
52
+ restWhere = {},
53
+ restOptions = {},
54
+ clientSDK,
55
+ runAfterFind = true,
56
+ runBeforeFind = true,
57
+ context
58
+ }) {
59
+ if (![RestQuery.Method.find, RestQuery.Method.get].includes(method)) {
60
+ throw new Parse.Error(Parse.Error.INVALID_QUERY, 'bad query type');
61
+ }
62
+ const isGet = method === RestQuery.Method.get;
63
+ enforceRoleSecurity(method, className, auth, config);
64
+ const result = runBeforeFind ? await triggers.maybeRunQueryTrigger(triggers.Types.beforeFind, className, restWhere, restOptions, config, auth, context, isGet) : Promise.resolve({
65
+ restWhere,
66
+ restOptions
67
+ });
68
+ return new _UnsafeRestQuery(config, auth, className, result.restWhere || restWhere, result.restOptions || restOptions, clientSDK, runAfterFind, context, isGet);
69
+ }
70
+ RestQuery.Method = Object.freeze({
71
+ get: 'get',
72
+ find: 'find'
73
+ });
74
+
75
+ /**
76
+ * _UnsafeRestQuery is meant for specific internal usage only. When you need to skip security checks or some triggers.
77
+ * Don't use it if you don't know what you are doing.
78
+ * @param config
79
+ * @param auth
80
+ * @param className
81
+ * @param restWhere
82
+ * @param restOptions
83
+ * @param clientSDK
84
+ * @param runAfterFind
85
+ * @param context
86
+ */
87
+ function _UnsafeRestQuery(config, auth, className, restWhere = {}, restOptions = {}, clientSDK, runAfterFind = true, context, isGet) {
21
88
  this.config = config;
22
89
  this.auth = auth;
23
90
  this.className = className;
24
91
  this.restWhere = restWhere;
25
92
  this.restOptions = restOptions;
26
93
  this.clientSDK = clientSDK;
94
+ this.runAfterFind = runAfterFind;
27
95
  this.response = null;
28
96
  this.findOptions = {};
29
- this.isWrite = false;
30
-
97
+ this.context = context || {};
98
+ this.isGet = isGet;
31
99
  if (!this.auth.isMaster) {
32
100
  if (this.className == '_Session') {
33
101
  if (!this.auth.user) {
34
- throw new Parse.Error(Parse.Error.INVALID_SESSION_TOKEN, 'Invalid session token');
102
+ throw createSanitizedError(Parse.Error.INVALID_SESSION_TOKEN, 'Invalid session token', config);
35
103
  }
36
104
  this.restWhere = {
37
- '$and': [this.restWhere, {
38
- 'user': {
105
+ $and: [this.restWhere, {
106
+ user: {
39
107
  __type: 'Pointer',
40
108
  className: '_User',
41
109
  objectId: this.auth.user.id
@@ -44,7 +112,6 @@ function RestQuery(config, auth, className, restWhere = {}, restOptions = {}, cl
44
112
  };
45
113
  }
46
114
  }
47
-
48
115
  this.doCount = false;
49
116
  this.includeAll = false;
50
117
 
@@ -55,17 +122,27 @@ function RestQuery(config, auth, className, restWhere = {}, restOptions = {}, cl
55
122
  // For example, passing an arg of include=foo.bar,foo.baz could lead to
56
123
  // this.include = [['foo'], ['foo', 'baz'], ['foo', 'bar']]
57
124
  this.include = [];
125
+ let keysForInclude = '';
58
126
 
59
127
  // If we have keys, we probably want to force some includes (n-1 level)
60
128
  // See issue: https://github.com/parse-community/parse-server/issues/3185
61
- if (restOptions.hasOwnProperty('keys')) {
62
- const keysForInclude = restOptions.keys.split(',').filter(key => {
129
+ if (Object.prototype.hasOwnProperty.call(restOptions, 'keys')) {
130
+ keysForInclude = restOptions.keys;
131
+ }
132
+
133
+ // If we have keys, we probably want to force some includes (n-1 level)
134
+ // in order to exclude specific keys.
135
+ if (Object.prototype.hasOwnProperty.call(restOptions, 'excludeKeys')) {
136
+ keysForInclude += ',' + restOptions.excludeKeys;
137
+ }
138
+ if (keysForInclude.length > 0) {
139
+ keysForInclude = keysForInclude.split(',').filter(key => {
63
140
  // At least 2 components
64
- return key.split(".").length > 1;
141
+ return key.split('.').length > 1;
65
142
  }).map(key => {
66
143
  // Slice the last component (a.b.c -> a.b)
67
144
  // Otherwise we'll include one level too much.
68
- return key.slice(0, key.lastIndexOf("."));
145
+ return key.slice(0, key.lastIndexOf('.'));
69
146
  }).join(',');
70
147
 
71
148
  // Concat the possibly present include string with the one from the keys
@@ -74,38 +151,48 @@ function RestQuery(config, auth, className, restWhere = {}, restOptions = {}, cl
74
151
  if (!restOptions.include || restOptions.include.length == 0) {
75
152
  restOptions.include = keysForInclude;
76
153
  } else {
77
- restOptions.include += "," + keysForInclude;
154
+ restOptions.include += ',' + keysForInclude;
78
155
  }
79
156
  }
80
157
  }
81
-
82
158
  for (var option in restOptions) {
83
159
  switch (option) {
84
160
  case 'keys':
85
161
  {
86
- const keys = restOptions.keys.split(',').concat(AlwaysSelectedKeys);
162
+ const keys = restOptions.keys.split(',').filter(key => key.length > 0).concat(AlwaysSelectedKeys);
87
163
  this.keys = Array.from(new Set(keys));
88
164
  break;
89
165
  }
166
+ case 'excludeKeys':
167
+ {
168
+ const exclude = restOptions.excludeKeys.split(',').filter(k => AlwaysSelectedKeys.indexOf(k) < 0);
169
+ this.excludeKeys = Array.from(new Set(exclude));
170
+ break;
171
+ }
90
172
  case 'count':
91
173
  this.doCount = true;
92
174
  break;
93
175
  case 'includeAll':
94
176
  this.includeAll = true;
95
177
  break;
178
+ case 'explain':
179
+ case 'hint':
96
180
  case 'distinct':
97
181
  case 'pipeline':
98
182
  case 'skip':
99
183
  case 'limit':
100
184
  case 'readPreference':
185
+ case 'comment':
101
186
  this.findOptions[option] = restOptions[option];
102
187
  break;
103
188
  case 'order':
104
189
  var fields = restOptions.order.split(',');
105
190
  this.findOptions.sort = fields.reduce((sortMap, field) => {
106
191
  field = field.trim();
107
- if (field === '$score') {
108
- sortMap.score = { $meta: 'textScore' };
192
+ if (field === '$score' || field === '-$score') {
193
+ sortMap.score = {
194
+ $meta: 'textScore'
195
+ };
109
196
  } else if (field[0] == '-') {
110
197
  sortMap[field.slice(1)] = -1;
111
198
  } else {
@@ -117,6 +204,10 @@ function RestQuery(config, auth, className, restWhere = {}, restOptions = {}, cl
117
204
  case 'include':
118
205
  {
119
206
  const paths = restOptions.include.split(',');
207
+ if (paths.includes('*')) {
208
+ this.includeAll = true;
209
+ break;
210
+ }
120
211
  // Load the existing includes (from keys)
121
212
  const pathSet = paths.reduce((memo, path) => {
122
213
  // Split each paths on . (a.b.c -> [a,b,c])
@@ -127,7 +218,6 @@ function RestQuery(config, auth, className, restWhere = {}, restOptions = {}, cl
127
218
  return memo;
128
219
  }, memo);
129
220
  }, {});
130
-
131
221
  this.include = Object.keys(pathSet).map(s => {
132
222
  return s.split('.');
133
223
  }).sort((a, b) => {
@@ -153,11 +243,15 @@ function RestQuery(config, auth, className, restWhere = {}, restOptions = {}, cl
153
243
  // Returns a promise for the response - an object with optional keys
154
244
  // 'results' and 'count'.
155
245
  // TODO: consolidate the replaceX functions
156
- RestQuery.prototype.execute = function (executeOptions) {
246
+ _UnsafeRestQuery.prototype.execute = function (executeOptions) {
157
247
  return Promise.resolve().then(() => {
158
248
  return this.buildRestWhere();
249
+ }).then(() => {
250
+ return this.denyProtectedFields();
159
251
  }).then(() => {
160
252
  return this.handleIncludeAll();
253
+ }).then(() => {
254
+ return this.handleExcludeKeys();
161
255
  }).then(() => {
162
256
  return this.runFind(executeOptions);
163
257
  }).then(() => {
@@ -166,12 +260,44 @@ RestQuery.prototype.execute = function (executeOptions) {
166
260
  return this.handleInclude();
167
261
  }).then(() => {
168
262
  return this.runAfterFindTrigger();
263
+ }).then(() => {
264
+ return this.handleAuthAdapters();
169
265
  }).then(() => {
170
266
  return this.response;
171
267
  });
172
268
  };
173
-
174
- RestQuery.prototype.buildRestWhere = function () {
269
+ _UnsafeRestQuery.prototype.each = function (callback) {
270
+ const {
271
+ config,
272
+ auth,
273
+ className,
274
+ restWhere,
275
+ restOptions,
276
+ clientSDK
277
+ } = this;
278
+ // if the limit is set, use it
279
+ restOptions.limit = restOptions.limit || 100;
280
+ restOptions.order = 'objectId';
281
+ let finished = false;
282
+ return continueWhile(() => {
283
+ return !finished;
284
+ }, async () => {
285
+ // Safe here to use _UnsafeRestQuery because the security was already
286
+ // checked during "await RestQuery()"
287
+ const query = new _UnsafeRestQuery(config, auth, className, restWhere, restOptions, clientSDK, this.runAfterFind, this.context);
288
+ const {
289
+ results
290
+ } = await query.execute();
291
+ results.forEach(callback);
292
+ finished = results.length < restOptions.limit;
293
+ if (!finished) {
294
+ restWhere.objectId = Object.assign({}, restWhere.objectId, {
295
+ $gt: results[results.length - 1].objectId
296
+ });
297
+ }
298
+ });
299
+ };
300
+ _UnsafeRestQuery.prototype.buildRestWhere = function () {
175
301
  return Promise.resolve().then(() => {
176
302
  return this.getUserAndRoleACL();
177
303
  }).then(() => {
@@ -191,20 +317,12 @@ RestQuery.prototype.buildRestWhere = function () {
191
317
  });
192
318
  };
193
319
 
194
- // Marks the query for a write attempt, so we read the proper ACL (write instead of read)
195
- RestQuery.prototype.forWrite = function () {
196
- this.isWrite = true;
197
- return this;
198
- };
199
-
200
320
  // Uses the Auth object to get the list of roles, adds the user id
201
- RestQuery.prototype.getUserAndRoleACL = function () {
321
+ _UnsafeRestQuery.prototype.getUserAndRoleACL = function () {
202
322
  if (this.auth.isMaster) {
203
323
  return Promise.resolve();
204
324
  }
205
-
206
325
  this.findOptions.acl = ['*'];
207
-
208
326
  if (this.auth.user) {
209
327
  return this.auth.getUserRoles().then(roles => {
210
328
  this.findOptions.acl = this.findOptions.acl.concat(roles, [this.auth.user.id]);
@@ -217,7 +335,7 @@ RestQuery.prototype.getUserAndRoleACL = function () {
217
335
 
218
336
  // Changes the className if redirectClassNameForKey is set.
219
337
  // Returns a promise.
220
- RestQuery.prototype.redirectClassNameForKey = function () {
338
+ _UnsafeRestQuery.prototype.redirectClassNameForKey = function () {
221
339
  if (!this.redirectKey) {
222
340
  return Promise.resolve();
223
341
  }
@@ -230,18 +348,17 @@ RestQuery.prototype.redirectClassNameForKey = function () {
230
348
  };
231
349
 
232
350
  // Validates this operation against the allowClientClassCreation config.
233
- RestQuery.prototype.validateClientClassCreation = function () {
351
+ _UnsafeRestQuery.prototype.validateClientClassCreation = function () {
234
352
  if (this.config.allowClientClassCreation === false && !this.auth.isMaster && SchemaController.systemClasses.indexOf(this.className) === -1) {
235
353
  return this.config.database.loadSchema().then(schemaController => schemaController.hasClass(this.className)).then(hasClass => {
236
354
  if (hasClass !== true) {
237
- throw new Parse.Error(Parse.Error.OPERATION_FORBIDDEN, 'This user is not allowed to access ' + 'non-existent class: ' + this.className);
355
+ throw createSanitizedError(Parse.Error.OPERATION_FORBIDDEN, 'This user is not allowed to access ' + 'non-existent class: ' + this.className, this.config);
238
356
  }
239
357
  });
240
358
  } else {
241
359
  return Promise.resolve();
242
360
  }
243
361
  };
244
-
245
362
  function transformInQuery(inQueryObject, className, results) {
246
363
  var values = [];
247
364
  for (var result of results) {
@@ -263,7 +380,7 @@ function transformInQuery(inQueryObject, className, results) {
263
380
  // $inQuery clause.
264
381
  // The $inQuery clause turns into an $in with values that are just
265
382
  // pointers to the objects returned in the subquery.
266
- RestQuery.prototype.replaceInQuery = function () {
383
+ _UnsafeRestQuery.prototype.replaceInQuery = async function () {
267
384
  var inQueryObject = findObjectWithKey(this.restWhere, '$inQuery');
268
385
  if (!inQueryObject) {
269
386
  return;
@@ -274,24 +391,30 @@ RestQuery.prototype.replaceInQuery = function () {
274
391
  if (!inQueryValue.where || !inQueryValue.className) {
275
392
  throw new Parse.Error(Parse.Error.INVALID_QUERY, 'improper usage of $inQuery');
276
393
  }
277
-
278
394
  const additionalOptions = {
279
395
  redirectClassNameForKey: inQueryValue.redirectClassNameForKey
280
396
  };
281
-
282
397
  if (this.restOptions.subqueryReadPreference) {
283
398
  additionalOptions.readPreference = this.restOptions.subqueryReadPreference;
284
399
  additionalOptions.subqueryReadPreference = this.restOptions.subqueryReadPreference;
400
+ } else if (this.restOptions.readPreference) {
401
+ additionalOptions.readPreference = this.restOptions.readPreference;
285
402
  }
286
-
287
- var subquery = new RestQuery(this.config, this.auth, inQueryValue.className, inQueryValue.where, additionalOptions);
403
+ const subquery = await RestQuery({
404
+ method: RestQuery.Method.find,
405
+ config: this.config,
406
+ auth: this.auth,
407
+ className: inQueryValue.className,
408
+ restWhere: inQueryValue.where,
409
+ restOptions: additionalOptions,
410
+ context: this.context
411
+ });
288
412
  return subquery.execute().then(response => {
289
413
  transformInQuery(inQueryObject, subquery.className, response.results);
290
414
  // Recurse to repeat
291
415
  return this.replaceInQuery();
292
416
  });
293
417
  };
294
-
295
418
  function transformNotInQuery(notInQueryObject, className, results) {
296
419
  var values = [];
297
420
  for (var result of results) {
@@ -313,7 +436,7 @@ function transformNotInQuery(notInQueryObject, className, results) {
313
436
  // $notInQuery clause.
314
437
  // The $notInQuery clause turns into a $nin with values that are just
315
438
  // pointers to the objects returned in the subquery.
316
- RestQuery.prototype.replaceNotInQuery = function () {
439
+ _UnsafeRestQuery.prototype.replaceNotInQuery = async function () {
317
440
  var notInQueryObject = findObjectWithKey(this.restWhere, '$notInQuery');
318
441
  if (!notInQueryObject) {
319
442
  return;
@@ -324,17 +447,24 @@ RestQuery.prototype.replaceNotInQuery = function () {
324
447
  if (!notInQueryValue.where || !notInQueryValue.className) {
325
448
  throw new Parse.Error(Parse.Error.INVALID_QUERY, 'improper usage of $notInQuery');
326
449
  }
327
-
328
450
  const additionalOptions = {
329
451
  redirectClassNameForKey: notInQueryValue.redirectClassNameForKey
330
452
  };
331
-
332
453
  if (this.restOptions.subqueryReadPreference) {
333
454
  additionalOptions.readPreference = this.restOptions.subqueryReadPreference;
334
455
  additionalOptions.subqueryReadPreference = this.restOptions.subqueryReadPreference;
456
+ } else if (this.restOptions.readPreference) {
457
+ additionalOptions.readPreference = this.restOptions.readPreference;
335
458
  }
336
-
337
- var subquery = new RestQuery(this.config, this.auth, notInQueryValue.className, notInQueryValue.where, additionalOptions);
459
+ const subquery = await RestQuery({
460
+ method: RestQuery.Method.find,
461
+ config: this.config,
462
+ auth: this.auth,
463
+ className: notInQueryValue.className,
464
+ restWhere: notInQueryValue.where,
465
+ restOptions: additionalOptions,
466
+ context: this.context
467
+ });
338
468
  return subquery.execute().then(response => {
339
469
  transformNotInQuery(notInQueryObject, subquery.className, response.results);
340
470
  // Recurse to repeat
@@ -342,10 +472,17 @@ RestQuery.prototype.replaceNotInQuery = function () {
342
472
  });
343
473
  };
344
474
 
475
+ // Used to get the deepest object from json using dot notation.
476
+ const getDeepestObjectFromKey = (json, key, idx, src) => {
477
+ if (key in json) {
478
+ return json[key];
479
+ }
480
+ src.splice(1); // Exit Early
481
+ };
345
482
  const transformSelect = (selectObject, key, objects) => {
346
483
  var values = [];
347
484
  for (var result of objects) {
348
- values.push(key.split('.').reduce((o, i) => o[i], result));
485
+ values.push(key.split('.').reduce(getDeepestObjectFromKey, result));
349
486
  }
350
487
  delete selectObject['$select'];
351
488
  if (Array.isArray(selectObject['$in'])) {
@@ -360,7 +497,7 @@ const transformSelect = (selectObject, key, objects) => {
360
497
  // The $select clause turns into an $in with values selected out of
361
498
  // the subquery.
362
499
  // Returns a possible-promise.
363
- RestQuery.prototype.replaceSelect = function () {
500
+ _UnsafeRestQuery.prototype.replaceSelect = async function () {
364
501
  var selectObject = findObjectWithKey(this.restWhere, '$select');
365
502
  if (!selectObject) {
366
503
  return;
@@ -372,28 +509,34 @@ RestQuery.prototype.replaceSelect = function () {
372
509
  if (!selectValue.query || !selectValue.key || typeof selectValue.query !== 'object' || !selectValue.query.className || Object.keys(selectValue).length !== 2) {
373
510
  throw new Parse.Error(Parse.Error.INVALID_QUERY, 'improper usage of $select');
374
511
  }
375
-
376
512
  const additionalOptions = {
377
513
  redirectClassNameForKey: selectValue.query.redirectClassNameForKey
378
514
  };
379
-
380
515
  if (this.restOptions.subqueryReadPreference) {
381
516
  additionalOptions.readPreference = this.restOptions.subqueryReadPreference;
382
517
  additionalOptions.subqueryReadPreference = this.restOptions.subqueryReadPreference;
518
+ } else if (this.restOptions.readPreference) {
519
+ additionalOptions.readPreference = this.restOptions.readPreference;
383
520
  }
384
-
385
- var subquery = new RestQuery(this.config, this.auth, selectValue.query.className, selectValue.query.where, additionalOptions);
521
+ const subquery = await RestQuery({
522
+ method: RestQuery.Method.find,
523
+ config: this.config,
524
+ auth: this.auth,
525
+ className: selectValue.query.className,
526
+ restWhere: selectValue.query.where,
527
+ restOptions: additionalOptions,
528
+ context: this.context
529
+ });
386
530
  return subquery.execute().then(response => {
387
531
  transformSelect(selectObject, selectValue.key, response.results);
388
532
  // Keep replacing $select clauses
389
533
  return this.replaceSelect();
390
534
  });
391
535
  };
392
-
393
536
  const transformDontSelect = (dontSelectObject, key, objects) => {
394
537
  var values = [];
395
538
  for (var result of objects) {
396
- values.push(key.split('.').reduce((o, i) => o[i], result));
539
+ values.push(key.split('.').reduce(getDeepestObjectFromKey, result));
397
540
  }
398
541
  delete dontSelectObject['$dontSelect'];
399
542
  if (Array.isArray(dontSelectObject['$nin'])) {
@@ -408,7 +551,7 @@ const transformDontSelect = (dontSelectObject, key, objects) => {
408
551
  // The $dontSelect clause turns into an $nin with values selected out of
409
552
  // the subquery.
410
553
  // Returns a possible-promise.
411
- RestQuery.prototype.replaceDontSelect = function () {
554
+ _UnsafeRestQuery.prototype.replaceDontSelect = async function () {
412
555
  var dontSelectObject = findObjectWithKey(this.restWhere, '$dontSelect');
413
556
  if (!dontSelectObject) {
414
557
  return;
@@ -422,46 +565,40 @@ RestQuery.prototype.replaceDontSelect = function () {
422
565
  const additionalOptions = {
423
566
  redirectClassNameForKey: dontSelectValue.query.redirectClassNameForKey
424
567
  };
425
-
426
568
  if (this.restOptions.subqueryReadPreference) {
427
569
  additionalOptions.readPreference = this.restOptions.subqueryReadPreference;
428
570
  additionalOptions.subqueryReadPreference = this.restOptions.subqueryReadPreference;
571
+ } else if (this.restOptions.readPreference) {
572
+ additionalOptions.readPreference = this.restOptions.readPreference;
429
573
  }
430
-
431
- var subquery = new RestQuery(this.config, this.auth, dontSelectValue.query.className, dontSelectValue.query.where, additionalOptions);
574
+ const subquery = await RestQuery({
575
+ method: RestQuery.Method.find,
576
+ config: this.config,
577
+ auth: this.auth,
578
+ className: dontSelectValue.query.className,
579
+ restWhere: dontSelectValue.query.where,
580
+ restOptions: additionalOptions,
581
+ context: this.context
582
+ });
432
583
  return subquery.execute().then(response => {
433
584
  transformDontSelect(dontSelectObject, dontSelectValue.key, response.results);
434
585
  // Keep replacing $dontSelect clauses
435
586
  return this.replaceDontSelect();
436
587
  });
437
588
  };
438
-
439
- const cleanResultOfSensitiveUserInfo = function (result, auth, config) {
589
+ _UnsafeRestQuery.prototype.cleanResultAuthData = function (result) {
440
590
  delete result.password;
441
-
442
- if (auth.isMaster || auth.user && auth.user.id === result.objectId) {
443
- return;
444
- }
445
-
446
- for (const field of config.userSensitiveFields) {
447
- delete result[field];
448
- }
449
- };
450
-
451
- const cleanResultAuthData = function (result) {
452
591
  if (result.authData) {
453
592
  Object.keys(result.authData).forEach(provider => {
454
593
  if (result.authData[provider] === null) {
455
594
  delete result.authData[provider];
456
595
  }
457
596
  });
458
-
459
597
  if (Object.keys(result.authData).length == 0) {
460
598
  delete result.authData;
461
599
  }
462
600
  }
463
601
  };
464
-
465
602
  const replaceEqualityConstraint = constraint => {
466
603
  if (typeof constraint !== 'object') {
467
604
  return constraint;
@@ -485,8 +622,7 @@ const replaceEqualityConstraint = constraint => {
485
622
  }
486
623
  return constraint;
487
624
  };
488
-
489
- RestQuery.prototype.replaceEquality = function () {
625
+ _UnsafeRestQuery.prototype.replaceEquality = function () {
490
626
  if (typeof this.restWhere !== 'object') {
491
627
  return;
492
628
  }
@@ -497,9 +633,11 @@ RestQuery.prototype.replaceEquality = function () {
497
633
 
498
634
  // Returns a promise for whether it was successful.
499
635
  // Populates this.response with an object that only has 'results'.
500
- RestQuery.prototype.runFind = function (options = {}) {
636
+ _UnsafeRestQuery.prototype.runFind = async function (options = {}) {
501
637
  if (this.findOptions.limit === 0) {
502
- this.response = { results: [] };
638
+ this.response = {
639
+ results: []
640
+ };
503
641
  return Promise.resolve();
504
642
  }
505
643
  const findOptions = Object.assign({}, this.findOptions);
@@ -511,31 +649,26 @@ RestQuery.prototype.runFind = function (options = {}) {
511
649
  if (options.op) {
512
650
  findOptions.op = options.op;
513
651
  }
514
- if (this.isWrite) {
515
- findOptions.isWrite = true;
516
- }
517
- return this.config.database.find(this.className, this.restWhere, findOptions).then(results => {
518
- if (this.className === '_User') {
519
- for (var result of results) {
520
- cleanResultOfSensitiveUserInfo(result, this.auth, this.config);
521
- cleanResultAuthData(result);
522
- }
652
+ const results = await this.config.database.find(this.className, this.restWhere, findOptions, this.auth);
653
+ if (this.className === '_User' && !findOptions.explain) {
654
+ for (var result of results) {
655
+ this.cleanResultAuthData(result);
523
656
  }
524
-
525
- this.config.filesController.expandFilesInObject(this.config, results);
526
-
527
- if (this.redirectClassName) {
528
- for (var r of results) {
529
- r.className = this.redirectClassName;
530
- }
657
+ }
658
+ await this.config.filesController.expandFilesInObject(this.config, results);
659
+ if (this.redirectClassName) {
660
+ for (var r of results) {
661
+ r.className = this.redirectClassName;
531
662
  }
532
- this.response = { results: results };
533
- });
663
+ }
664
+ this.response = {
665
+ results: results
666
+ };
534
667
  };
535
668
 
536
669
  // Returns a promise for whether it was successful.
537
670
  // Populates this.response.count with the count
538
- RestQuery.prototype.runCount = function () {
671
+ _UnsafeRestQuery.prototype.runCount = function () {
539
672
  if (!this.doCount) {
540
673
  return;
541
674
  }
@@ -546,9 +679,21 @@ RestQuery.prototype.runCount = function () {
546
679
  this.response.count = c;
547
680
  });
548
681
  };
682
+ _UnsafeRestQuery.prototype.denyProtectedFields = async function () {
683
+ if (this.auth.isMaster) {
684
+ return;
685
+ }
686
+ const schemaController = await this.config.database.loadSchema();
687
+ const protectedFields = this.config.database.addProtectedFields(schemaController, this.className, this.restWhere, this.findOptions.acl, this.auth, this.findOptions) || [];
688
+ for (const key of protectedFields) {
689
+ if (this.restWhere[key]) {
690
+ throw createSanitizedError(Parse.Error.OPERATION_FORBIDDEN, `This user is not allowed to query ${key} on class ${this.className}`, this.config);
691
+ }
692
+ }
693
+ };
549
694
 
550
695
  // Augments this.response with all pointers on an object
551
- RestQuery.prototype.handleIncludeAll = function () {
696
+ _UnsafeRestQuery.prototype.handleIncludeAll = function () {
552
697
  if (!this.includeAll) {
553
698
  return;
554
699
  }
@@ -556,7 +701,7 @@ RestQuery.prototype.handleIncludeAll = function () {
556
701
  const includeFields = [];
557
702
  const keyFields = [];
558
703
  for (const field in schema.fields) {
559
- if (schema.fields[field].type && schema.fields[field].type === 'Pointer') {
704
+ if (schema.fields[field].type && schema.fields[field].type === 'Pointer' || schema.fields[field].type && schema.fields[field].type === 'Array') {
560
705
  includeFields.push([field]);
561
706
  keyFields.push(field);
562
707
  }
@@ -570,32 +715,72 @@ RestQuery.prototype.handleIncludeAll = function () {
570
715
  });
571
716
  };
572
717
 
718
+ // Updates property `this.keys` to contain all keys but the ones unselected.
719
+ _UnsafeRestQuery.prototype.handleExcludeKeys = function () {
720
+ if (!this.excludeKeys) {
721
+ return;
722
+ }
723
+ if (this.keys) {
724
+ this.keys = this.keys.filter(k => !this.excludeKeys.includes(k));
725
+ return;
726
+ }
727
+ return this.config.database.loadSchema().then(schemaController => schemaController.getOneSchema(this.className)).then(schema => {
728
+ const fields = Object.keys(schema.fields);
729
+ this.keys = fields.filter(k => !this.excludeKeys.includes(k));
730
+ });
731
+ };
732
+
573
733
  // Augments this.response with data at the paths provided in this.include.
574
- RestQuery.prototype.handleInclude = function () {
734
+ _UnsafeRestQuery.prototype.handleInclude = async function () {
575
735
  if (this.include.length == 0) {
576
736
  return;
577
737
  }
578
-
579
- var pathResponse = includePath(this.config, this.auth, this.response, this.include[0], this.restOptions);
580
- if (pathResponse.then) {
581
- return pathResponse.then(newResponse => {
582
- this.response = newResponse;
583
- this.include = this.include.slice(1);
584
- return this.handleInclude();
738
+ const indexedResults = this.response.results.reduce((indexed, result, i) => {
739
+ indexed[result.objectId] = i;
740
+ return indexed;
741
+ }, {});
742
+
743
+ // Build the execution tree
744
+ const executionTree = {};
745
+ this.include.forEach(path => {
746
+ let current = executionTree;
747
+ path.forEach(node => {
748
+ if (!current[node]) {
749
+ current[node] = {
750
+ path,
751
+ children: {}
752
+ };
753
+ }
754
+ current = current[node].children;
585
755
  });
586
- } else if (this.include.length > 0) {
587
- this.include = this.include.slice(1);
588
- return this.handleInclude();
589
- }
590
-
591
- return pathResponse;
756
+ });
757
+ const recursiveExecutionTree = async treeNode => {
758
+ const {
759
+ path,
760
+ children
761
+ } = treeNode;
762
+ const pathResponse = includePath(this.config, this.auth, this.response, path, this.context, this.restOptions, this);
763
+ if (pathResponse.then) {
764
+ const newResponse = await pathResponse;
765
+ newResponse.results.forEach(newObject => {
766
+ // We hydrate the root of each result with sub results
767
+ this.response.results[indexedResults[newObject.objectId]][path[0]] = newObject[path[0]];
768
+ });
769
+ }
770
+ return Promise.all(Object.values(children).map(recursiveExecutionTree));
771
+ };
772
+ await Promise.all(Object.values(executionTree).map(recursiveExecutionTree));
773
+ this.include = [];
592
774
  };
593
775
 
594
776
  //Returns a promise of a processed set of results
595
- RestQuery.prototype.runAfterFindTrigger = function () {
777
+ _UnsafeRestQuery.prototype.runAfterFindTrigger = function () {
596
778
  if (!this.response) {
597
779
  return;
598
780
  }
781
+ if (!this.runAfterFind) {
782
+ return;
783
+ }
599
784
  // Avoid doing any setup for triggers if there is no 'afterFind' trigger for this class.
600
785
  const hasAfterFindHook = triggers.triggerExists(this.className, triggers.Types.afterFind, this.config.applicationId);
601
786
  if (!hasAfterFindHook) {
@@ -605,8 +790,12 @@ RestQuery.prototype.runAfterFindTrigger = function () {
605
790
  if (this.findOptions.pipeline || this.findOptions.distinct) {
606
791
  return Promise.resolve();
607
792
  }
793
+ const json = Object.assign({}, this.restOptions);
794
+ json.where = this.restWhere;
795
+ const parseQuery = new Parse.Query(this.className);
796
+ parseQuery.withJSON(json);
608
797
  // Run afterFind trigger and set the new results
609
- return triggers.maybeRunAfterFindTrigger(triggers.Types.afterFind, this.auth, this.className, this.response.results, this.config).then(results => {
798
+ return triggers.maybeRunAfterFindTrigger(triggers.Types.afterFind, this.auth, this.className, this.response.results, this.config, parseQuery, this.context, this.isGet).then(results => {
610
799
  // Ensure we properly set the className back
611
800
  if (this.redirectClassName) {
612
801
  this.response.results = results.map(object => {
@@ -621,11 +810,20 @@ RestQuery.prototype.runAfterFindTrigger = function () {
621
810
  }
622
811
  });
623
812
  };
813
+ _UnsafeRestQuery.prototype.handleAuthAdapters = async function () {
814
+ if (this.className !== '_User' || this.findOptions.explain) {
815
+ return;
816
+ }
817
+ await Promise.all(this.response.results.map(result => this.config.authDataManager.runAfterFind({
818
+ config: this.config,
819
+ auth: this.auth
820
+ }, result.authData)));
821
+ };
624
822
 
625
823
  // Adds included values to the response.
626
824
  // Path is a list of field names.
627
825
  // Returns a promise for an augmented response.
628
- function includePath(config, auth, response, path, restOptions = {}) {
826
+ function includePath(config, auth, response, path, context, restOptions = {}) {
629
827
  var pointers = findPointers(response.results, path);
630
828
  if (pointers.length == 0) {
631
829
  return response;
@@ -662,22 +860,57 @@ function includePath(config, auth, response, path, restOptions = {}) {
662
860
  includeRestOptions.keys = Array.from(keySet).join(',');
663
861
  }
664
862
  }
665
-
863
+ if (restOptions.excludeKeys) {
864
+ const excludeKeys = new Set(restOptions.excludeKeys.split(','));
865
+ const excludeKeySet = Array.from(excludeKeys).reduce((set, key) => {
866
+ const keyPath = key.split('.');
867
+ let i = 0;
868
+ for (i; i < path.length; i++) {
869
+ if (path[i] != keyPath[i]) {
870
+ return set;
871
+ }
872
+ }
873
+ if (i == keyPath.length - 1) {
874
+ set.add(keyPath[i]);
875
+ }
876
+ return set;
877
+ }, new Set());
878
+ if (excludeKeySet.size > 0) {
879
+ includeRestOptions.excludeKeys = Array.from(excludeKeySet).join(',');
880
+ }
881
+ }
666
882
  if (restOptions.includeReadPreference) {
667
883
  includeRestOptions.readPreference = restOptions.includeReadPreference;
668
884
  includeRestOptions.includeReadPreference = restOptions.includeReadPreference;
885
+ } else if (restOptions.readPreference) {
886
+ includeRestOptions.readPreference = restOptions.readPreference;
669
887
  }
670
-
671
- const queryPromises = Object.keys(pointersHash).map(className => {
888
+ const queryPromises = Object.keys(pointersHash).map(async className => {
672
889
  const objectIds = Array.from(pointersHash[className]);
673
890
  let where;
674
891
  if (objectIds.length === 1) {
675
- where = { 'objectId': objectIds[0] };
892
+ where = {
893
+ objectId: objectIds[0]
894
+ };
676
895
  } else {
677
- where = { 'objectId': { '$in': objectIds } };
896
+ where = {
897
+ objectId: {
898
+ $in: objectIds
899
+ }
900
+ };
678
901
  }
679
- var query = new RestQuery(config, auth, className, where, includeRestOptions);
680
- return query.execute({ op: 'get' }).then(results => {
902
+ const query = await RestQuery({
903
+ method: objectIds.length === 1 ? RestQuery.Method.get : RestQuery.Method.find,
904
+ config,
905
+ auth,
906
+ className,
907
+ restWhere: where,
908
+ restOptions: includeRestOptions,
909
+ context: context
910
+ });
911
+ return query.execute({
912
+ op: 'get'
913
+ }).then(results => {
681
914
  results.className = className;
682
915
  return Promise.resolve(results);
683
916
  });
@@ -689,8 +922,7 @@ function includePath(config, auth, response, path, restOptions = {}) {
689
922
  for (var obj of includeResponse.results) {
690
923
  obj.__type = 'Object';
691
924
  obj.className = includeResponse.className;
692
-
693
- if (obj.className == "_User" && !auth.isMaster) {
925
+ if (obj.className == '_User' && !auth.isMaster) {
694
926
  delete obj.sessionToken;
695
927
  delete obj.authData;
696
928
  }
@@ -698,7 +930,6 @@ function includePath(config, auth, response, path, restOptions = {}) {
698
930
  }
699
931
  return replace;
700
932
  }, {});
701
-
702
933
  var resp = {
703
934
  results: replacePointers(response.results, path, replace)
704
935
  };
@@ -716,24 +947,17 @@ function includePath(config, auth, response, path, restOptions = {}) {
716
947
  // Returns a list of pointers in REST format.
717
948
  function findPointers(object, path) {
718
949
  if (object instanceof Array) {
719
- var answer = [];
720
- for (var x of object) {
721
- answer = answer.concat(findPointers(x, path));
722
- }
723
- return answer;
950
+ return object.map(x => findPointers(x, path)).flat();
724
951
  }
725
-
726
952
  if (typeof object !== 'object' || !object) {
727
953
  return [];
728
954
  }
729
-
730
955
  if (path.length == 0) {
731
956
  if (object === null || object.__type == 'Pointer') {
732
957
  return [object];
733
958
  }
734
959
  return [];
735
960
  }
736
-
737
961
  var subobject = object[path[0]];
738
962
  if (!subobject) {
739
963
  return [];
@@ -751,18 +975,15 @@ function replacePointers(object, path, replace) {
751
975
  if (object instanceof Array) {
752
976
  return object.map(obj => replacePointers(obj, path, replace)).filter(obj => typeof obj !== 'undefined');
753
977
  }
754
-
755
978
  if (typeof object !== 'object' || !object) {
756
979
  return object;
757
980
  }
758
-
759
981
  if (path.length === 0) {
760
982
  if (object && object.__type === 'Pointer') {
761
983
  return replace[object.objectId];
762
984
  }
763
985
  return object;
764
986
  }
765
-
766
987
  var subobject = object[path[0]];
767
988
  if (!subobject) {
768
989
  return object;
@@ -803,6 +1024,7 @@ function findObjectWithKey(root, key) {
803
1024
  }
804
1025
  }
805
1026
  }
806
-
807
1027
  module.exports = RestQuery;
808
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../src/RestQuery.js"],"names":["SchemaController","require","Parse","triggers","AlwaysSelectedKeys","RestQuery","config","auth","className","restWhere","restOptions","clientSDK","response","findOptions","isWrite","isMaster","user","Error","INVALID_SESSION_TOKEN","__type","objectId","id","doCount","includeAll","include","hasOwnProperty","keysForInclude","keys","split","filter","key","length","map","slice","lastIndexOf","join","option","concat","Array","from","Set","fields","order","sort","reduce","sortMap","field","trim","score","$meta","paths","pathSet","memo","path","index","parts","Object","s","a","b","redirectKey","redirectClassNameForKey","redirectClassName","INVALID_JSON","prototype","execute","executeOptions","Promise","resolve","then","buildRestWhere","handleIncludeAll","runFind","runCount","handleInclude","runAfterFindTrigger","getUserAndRoleACL","validateClientClassCreation","replaceSelect","replaceDontSelect","replaceInQuery","replaceNotInQuery","replaceEquality","forWrite","acl","getUserRoles","roles","database","newClassName","allowClientClassCreation","systemClasses","indexOf","loadSchema","schemaController","hasClass","OPERATION_FORBIDDEN","transformInQuery","inQueryObject","results","values","result","push","isArray","findObjectWithKey","inQueryValue","where","INVALID_QUERY","additionalOptions","subqueryReadPreference","readPreference","subquery","transformNotInQuery","notInQueryObject","notInQueryValue","transformSelect","selectObject","objects","o","i","selectValue","query","transformDontSelect","dontSelectObject","dontSelectValue","cleanResultOfSensitiveUserInfo","password","userSensitiveFields","cleanResultAuthData","authData","forEach","provider","replaceEqualityConstraint","constraint","equalToObject","hasDirectConstraint","hasOperatorConstraint","options","limit","assign","op","find","filesController","expandFilesInObject","r","count","skip","c","getOneSchema","schema","includeFields","keyFields","type","pathResponse","includePath","newResponse","hasAfterFindHook","triggerExists","Types","afterFind","applicationId","pipeline","distinct","maybeRunAfterFindTrigger","object","toJSON","pointers","findPointers","pointersHash","pointer","add","includeRestOptions","keySet","set","keyPath","size","includeReadPreference","queryPromises","objectIds","all","responses","replace","includeResponse","obj","sessionToken","resp","replacePointers","answer","x","subobject","newsub","root","item","subkey","module","exports"],"mappings":"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","file":"RestQuery.js","sourcesContent":["// An object that encapsulates everything we need to run a 'find'\n// operation, encoded in the REST API format.\n\nvar SchemaController = require('./Controllers/SchemaController');\nvar Parse = require('parse/node').Parse;\nconst triggers = require('./triggers');\n\nconst AlwaysSelectedKeys = ['objectId', 'createdAt', 'updatedAt'];\n// restOptions can include:\n//   skip\n//   limit\n//   order\n//   count\n//   include\n//   keys\n//   redirectClassNameForKey\nfunction RestQuery(config, auth, className, restWhere = {}, restOptions = {}, clientSDK) {\n\n  this.config = config;\n  this.auth = auth;\n  this.className = className;\n  this.restWhere = restWhere;\n  this.restOptions = restOptions;\n  this.clientSDK = clientSDK;\n  this.response = null;\n  this.findOptions = {};\n  this.isWrite = false;\n\n  if (!this.auth.isMaster) {\n    if (this.className == '_Session') {\n      if (!this.auth.user) {\n        throw new Parse.Error(Parse.Error.INVALID_SESSION_TOKEN,\n          'Invalid session token');\n      }\n      this.restWhere = {\n        '$and': [this.restWhere, {\n          'user': {\n            __type: 'Pointer',\n            className: '_User',\n            objectId: this.auth.user.id\n          }\n        }]\n      };\n    }\n  }\n\n  this.doCount = false;\n  this.includeAll = false;\n\n  // The format for this.include is not the same as the format for the\n  // include option - it's the paths we should include, in order,\n  // stored as arrays, taking into account that we need to include foo\n  // before including foo.bar. Also it should dedupe.\n  // For example, passing an arg of include=foo.bar,foo.baz could lead to\n  // this.include = [['foo'], ['foo', 'baz'], ['foo', 'bar']]\n  this.include = [];\n\n  // If we have keys, we probably want to force some includes (n-1 level)\n  // See issue: https://github.com/parse-community/parse-server/issues/3185\n  if (restOptions.hasOwnProperty('keys')) {\n    const keysForInclude = restOptions.keys.split(',').filter((key) => {\n      // At least 2 components\n      return key.split(\".\").length > 1;\n    }).map((key) => {\n      // Slice the last component (a.b.c -> a.b)\n      // Otherwise we'll include one level too much.\n      return key.slice(0, key.lastIndexOf(\".\"));\n    }).join(',');\n\n    // Concat the possibly present include string with the one from the keys\n    // Dedup / sorting is handle in 'include' case.\n    if (keysForInclude.length > 0) {\n      if (!restOptions.include || restOptions.include.length == 0) {\n        restOptions.include = keysForInclude;\n      } else {\n        restOptions.include += \",\" + keysForInclude;\n      }\n    }\n  }\n\n  for (var option in restOptions) {\n    switch(option) {\n    case 'keys': {\n      const keys = restOptions.keys.split(',').concat(AlwaysSelectedKeys);\n      this.keys = Array.from(new Set(keys));\n      break;\n    }\n    case 'count':\n      this.doCount = true;\n      break;\n    case 'includeAll':\n      this.includeAll = true;\n      break;\n    case 'distinct':\n    case 'pipeline':\n    case 'skip':\n    case 'limit':\n    case 'readPreference':\n      this.findOptions[option] = restOptions[option];\n      break;\n    case 'order':\n      var fields = restOptions.order.split(',');\n      this.findOptions.sort = fields.reduce((sortMap, field) => {\n        field = field.trim();\n        if (field === '$score') {\n          sortMap.score = {$meta: 'textScore'};\n        } else if (field[0] == '-') {\n          sortMap[field.slice(1)] = -1;\n        } else {\n          sortMap[field] = 1;\n        }\n        return sortMap;\n      }, {});\n      break;\n    case 'include': {\n      const paths = restOptions.include.split(',');\n      // Load the existing includes (from keys)\n      const pathSet = paths.reduce((memo, path) => {\n        // Split each paths on . (a.b.c -> [a,b,c])\n        // reduce to create all paths\n        // ([a,b,c] -> {a: true, 'a.b': true, 'a.b.c': true})\n        return path.split('.').reduce((memo, path, index, parts) => {\n          memo[parts.slice(0, index + 1).join('.')] = true;\n          return memo;\n        }, memo);\n      }, {});\n\n      this.include = Object.keys(pathSet).map((s) => {\n        return s.split('.');\n      }).sort((a, b) => {\n        return a.length - b.length; // Sort by number of components\n      });\n      break;\n    }\n    case 'redirectClassNameForKey':\n      this.redirectKey = restOptions.redirectClassNameForKey;\n      this.redirectClassName = null;\n      break;\n    case 'includeReadPreference':\n    case 'subqueryReadPreference':\n      break;\n    default:\n      throw new Parse.Error(Parse.Error.INVALID_JSON,\n        'bad option: ' + option);\n    }\n  }\n}\n\n// A convenient method to perform all the steps of processing a query\n// in order.\n// Returns a promise for the response - an object with optional keys\n// 'results' and 'count'.\n// TODO: consolidate the replaceX functions\nRestQuery.prototype.execute = function(executeOptions) {\n  return Promise.resolve().then(() => {\n    return this.buildRestWhere();\n  }).then(() => {\n    return this.handleIncludeAll();\n  }).then(() => {\n    return this.runFind(executeOptions);\n  }).then(() => {\n    return this.runCount();\n  }).then(() => {\n    return this.handleInclude();\n  }).then(() => {\n    return this.runAfterFindTrigger();\n  }).then(() => {\n    return this.response;\n  });\n};\n\nRestQuery.prototype.buildRestWhere = function() {\n  return Promise.resolve().then(() => {\n    return this.getUserAndRoleACL();\n  }).then(() => {\n    return this.redirectClassNameForKey();\n  }).then(() => {\n    return this.validateClientClassCreation();\n  }).then(() => {\n    return this.replaceSelect();\n  }).then(() => {\n    return this.replaceDontSelect();\n  }).then(() => {\n    return this.replaceInQuery();\n  }).then(() => {\n    return this.replaceNotInQuery();\n  }).then(() => {\n    return this.replaceEquality();\n  });\n}\n\n// Marks the query for a write attempt, so we read the proper ACL (write instead of read)\nRestQuery.prototype.forWrite = function() {\n  this.isWrite = true;\n  return this;\n}\n\n// Uses the Auth object to get the list of roles, adds the user id\nRestQuery.prototype.getUserAndRoleACL = function() {\n  if (this.auth.isMaster) {\n    return Promise.resolve();\n  }\n\n  this.findOptions.acl = ['*'];\n\n  if (this.auth.user) {\n    return this.auth.getUserRoles().then((roles) => {\n      this.findOptions.acl = this.findOptions.acl.concat(roles, [this.auth.user.id]);\n      return;\n    });\n  } else {\n    return Promise.resolve();\n  }\n};\n\n// Changes the className if redirectClassNameForKey is set.\n// Returns a promise.\nRestQuery.prototype.redirectClassNameForKey = function() {\n  if (!this.redirectKey) {\n    return Promise.resolve();\n  }\n\n  // We need to change the class name based on the schema\n  return this.config.database.redirectClassNameForKey(this.className, this.redirectKey)\n    .then((newClassName) => {\n      this.className = newClassName;\n      this.redirectClassName = newClassName;\n    });\n};\n\n// Validates this operation against the allowClientClassCreation config.\nRestQuery.prototype.validateClientClassCreation = function() {\n  if (this.config.allowClientClassCreation === false && !this.auth.isMaster\n      && SchemaController.systemClasses.indexOf(this.className) === -1) {\n    return this.config.database.loadSchema()\n      .then(schemaController => schemaController.hasClass(this.className))\n      .then(hasClass => {\n        if (hasClass !== true) {\n          throw new Parse.Error(Parse.Error.OPERATION_FORBIDDEN,\n            'This user is not allowed to access ' +\n                                'non-existent class: ' + this.className);\n        }\n      });\n  } else {\n    return Promise.resolve();\n  }\n};\n\nfunction transformInQuery(inQueryObject, className, results) {\n  var values = [];\n  for (var result of results) {\n    values.push({\n      __type: 'Pointer',\n      className: className,\n      objectId: result.objectId\n    });\n  }\n  delete inQueryObject['$inQuery'];\n  if (Array.isArray(inQueryObject['$in'])) {\n    inQueryObject['$in'] = inQueryObject['$in'].concat(values);\n  } else {\n    inQueryObject['$in'] = values;\n  }\n}\n\n// Replaces a $inQuery clause by running the subquery, if there is an\n// $inQuery clause.\n// The $inQuery clause turns into an $in with values that are just\n// pointers to the objects returned in the subquery.\nRestQuery.prototype.replaceInQuery = function() {\n  var inQueryObject = findObjectWithKey(this.restWhere, '$inQuery');\n  if (!inQueryObject) {\n    return;\n  }\n\n  // The inQuery value must have precisely two keys - where and className\n  var inQueryValue = inQueryObject['$inQuery'];\n  if (!inQueryValue.where || !inQueryValue.className) {\n    throw new Parse.Error(Parse.Error.INVALID_QUERY,\n      'improper usage of $inQuery');\n  }\n\n  const additionalOptions = {\n    redirectClassNameForKey: inQueryValue.redirectClassNameForKey\n  };\n\n  if (this.restOptions.subqueryReadPreference) {\n    additionalOptions.readPreference = this.restOptions.subqueryReadPreference;\n    additionalOptions.subqueryReadPreference = this.restOptions.subqueryReadPreference;\n  }\n\n  var subquery = new RestQuery(\n    this.config, this.auth, inQueryValue.className,\n    inQueryValue.where, additionalOptions);\n  return subquery.execute().then((response) => {\n    transformInQuery(inQueryObject, subquery.className, response.results);\n    // Recurse to repeat\n    return this.replaceInQuery();\n  });\n};\n\nfunction transformNotInQuery(notInQueryObject, className, results) {\n  var values = [];\n  for (var result of results) {\n    values.push({\n      __type: 'Pointer',\n      className: className,\n      objectId: result.objectId\n    });\n  }\n  delete notInQueryObject['$notInQuery'];\n  if (Array.isArray(notInQueryObject['$nin'])) {\n    notInQueryObject['$nin'] = notInQueryObject['$nin'].concat(values);\n  } else {\n    notInQueryObject['$nin'] = values;\n  }\n}\n\n// Replaces a $notInQuery clause by running the subquery, if there is an\n// $notInQuery clause.\n// The $notInQuery clause turns into a $nin with values that are just\n// pointers to the objects returned in the subquery.\nRestQuery.prototype.replaceNotInQuery = function() {\n  var notInQueryObject = findObjectWithKey(this.restWhere, '$notInQuery');\n  if (!notInQueryObject) {\n    return;\n  }\n\n  // The notInQuery value must have precisely two keys - where and className\n  var notInQueryValue = notInQueryObject['$notInQuery'];\n  if (!notInQueryValue.where || !notInQueryValue.className) {\n    throw new Parse.Error(Parse.Error.INVALID_QUERY,\n      'improper usage of $notInQuery');\n  }\n\n  const additionalOptions = {\n    redirectClassNameForKey: notInQueryValue.redirectClassNameForKey\n  };\n\n  if (this.restOptions.subqueryReadPreference) {\n    additionalOptions.readPreference = this.restOptions.subqueryReadPreference;\n    additionalOptions.subqueryReadPreference = this.restOptions.subqueryReadPreference;\n  }\n\n  var subquery = new RestQuery(\n    this.config, this.auth, notInQueryValue.className,\n    notInQueryValue.where, additionalOptions);\n  return subquery.execute().then((response) => {\n    transformNotInQuery(notInQueryObject, subquery.className, response.results);\n    // Recurse to repeat\n    return this.replaceNotInQuery();\n  });\n};\n\nconst transformSelect = (selectObject, key ,objects) => {\n  var values = [];\n  for (var result of objects) {\n    values.push(key.split('.').reduce((o,i)=>o[i], result));\n  }\n  delete selectObject['$select'];\n  if (Array.isArray(selectObject['$in'])) {\n    selectObject['$in'] = selectObject['$in'].concat(values);\n  } else {\n    selectObject['$in'] = values;\n  }\n}\n\n// Replaces a $select clause by running the subquery, if there is a\n// $select clause.\n// The $select clause turns into an $in with values selected out of\n// the subquery.\n// Returns a possible-promise.\nRestQuery.prototype.replaceSelect = function() {\n  var selectObject = findObjectWithKey(this.restWhere, '$select');\n  if (!selectObject) {\n    return;\n  }\n\n  // The select value must have precisely two keys - query and key\n  var selectValue = selectObject['$select'];\n  // iOS SDK don't send where if not set, let it pass\n  if (!selectValue.query ||\n      !selectValue.key ||\n      typeof selectValue.query !== 'object' ||\n      !selectValue.query.className ||\n      Object.keys(selectValue).length !== 2) {\n    throw new Parse.Error(Parse.Error.INVALID_QUERY,\n      'improper usage of $select');\n  }\n\n  const additionalOptions = {\n    redirectClassNameForKey: selectValue.query.redirectClassNameForKey\n  };\n\n  if (this.restOptions.subqueryReadPreference) {\n    additionalOptions.readPreference = this.restOptions.subqueryReadPreference;\n    additionalOptions.subqueryReadPreference = this.restOptions.subqueryReadPreference;\n  }\n\n  var subquery = new RestQuery(\n    this.config, this.auth, selectValue.query.className,\n    selectValue.query.where, additionalOptions);\n  return subquery.execute().then((response) => {\n    transformSelect(selectObject, selectValue.key, response.results);\n    // Keep replacing $select clauses\n    return this.replaceSelect();\n  })\n};\n\nconst transformDontSelect = (dontSelectObject, key, objects) => {\n  var values = [];\n  for (var result of objects) {\n    values.push(key.split('.').reduce((o,i)=>o[i], result));\n  }\n  delete dontSelectObject['$dontSelect'];\n  if (Array.isArray(dontSelectObject['$nin'])) {\n    dontSelectObject['$nin'] = dontSelectObject['$nin'].concat(values);\n  } else {\n    dontSelectObject['$nin'] = values;\n  }\n}\n\n// Replaces a $dontSelect clause by running the subquery, if there is a\n// $dontSelect clause.\n// The $dontSelect clause turns into an $nin with values selected out of\n// the subquery.\n// Returns a possible-promise.\nRestQuery.prototype.replaceDontSelect = function() {\n  var dontSelectObject = findObjectWithKey(this.restWhere, '$dontSelect');\n  if (!dontSelectObject) {\n    return;\n  }\n\n  // The dontSelect value must have precisely two keys - query and key\n  var dontSelectValue = dontSelectObject['$dontSelect'];\n  if (!dontSelectValue.query ||\n      !dontSelectValue.key ||\n      typeof dontSelectValue.query !== 'object' ||\n      !dontSelectValue.query.className ||\n      Object.keys(dontSelectValue).length !== 2) {\n    throw new Parse.Error(Parse.Error.INVALID_QUERY,\n      'improper usage of $dontSelect');\n  }\n  const additionalOptions = {\n    redirectClassNameForKey: dontSelectValue.query.redirectClassNameForKey\n  };\n\n  if (this.restOptions.subqueryReadPreference) {\n    additionalOptions.readPreference = this.restOptions.subqueryReadPreference;\n    additionalOptions.subqueryReadPreference = this.restOptions.subqueryReadPreference;\n  }\n\n  var subquery = new RestQuery(\n    this.config, this.auth, dontSelectValue.query.className,\n    dontSelectValue.query.where, additionalOptions);\n  return subquery.execute().then((response) => {\n    transformDontSelect(dontSelectObject, dontSelectValue.key, response.results);\n    // Keep replacing $dontSelect clauses\n    return this.replaceDontSelect();\n  })\n};\n\nconst cleanResultOfSensitiveUserInfo = function (result, auth, config) {\n  delete result.password;\n\n  if (auth.isMaster || (auth.user && auth.user.id === result.objectId)) {\n    return;\n  }\n\n  for (const field of config.userSensitiveFields) {\n    delete result[field];\n  }\n};\n\nconst cleanResultAuthData = function (result) {\n  if (result.authData) {\n    Object.keys(result.authData).forEach((provider) => {\n      if (result.authData[provider] === null) {\n        delete result.authData[provider];\n      }\n    });\n\n    if (Object.keys(result.authData).length == 0) {\n      delete result.authData;\n    }\n  }\n};\n\nconst replaceEqualityConstraint = (constraint) => {\n  if (typeof constraint !== 'object') {\n    return constraint;\n  }\n  const equalToObject = {};\n  let hasDirectConstraint = false;\n  let hasOperatorConstraint = false;\n  for (const key in constraint) {\n    if (key.indexOf('$') !== 0) {\n      hasDirectConstraint = true;\n      equalToObject[key] = constraint[key];\n    } else {\n      hasOperatorConstraint = true;\n    }\n  }\n  if (hasDirectConstraint && hasOperatorConstraint) {\n    constraint['$eq'] = equalToObject;\n    Object.keys(equalToObject).forEach((key) => {\n      delete constraint[key];\n    });\n  }\n  return constraint;\n}\n\nRestQuery.prototype.replaceEquality = function() {\n  if (typeof this.restWhere !== 'object') {\n    return;\n  }\n  for (const key in this.restWhere) {\n    this.restWhere[key] = replaceEqualityConstraint(this.restWhere[key]);\n  }\n}\n\n// Returns a promise for whether it was successful.\n// Populates this.response with an object that only has 'results'.\nRestQuery.prototype.runFind = function(options = {}) {\n  if (this.findOptions.limit === 0) {\n    this.response = {results: []};\n    return Promise.resolve();\n  }\n  const findOptions = Object.assign({}, this.findOptions);\n  if (this.keys) {\n    findOptions.keys = this.keys.map((key) => {\n      return key.split('.')[0];\n    });\n  }\n  if (options.op) {\n    findOptions.op = options.op;\n  }\n  if (this.isWrite) {\n    findOptions.isWrite = true;\n  }\n  return this.config.database.find(this.className, this.restWhere, findOptions)\n    .then((results) => {\n      if (this.className === '_User') {\n        for (var result of results) {\n          cleanResultOfSensitiveUserInfo(result, this.auth, this.config);\n          cleanResultAuthData(result);\n        }\n      }\n\n      this.config.filesController.expandFilesInObject(this.config, results);\n\n      if (this.redirectClassName) {\n        for (var r of results) {\n          r.className = this.redirectClassName;\n        }\n      }\n      this.response = {results: results};\n    });\n};\n\n// Returns a promise for whether it was successful.\n// Populates this.response.count with the count\nRestQuery.prototype.runCount = function() {\n  if (!this.doCount) {\n    return;\n  }\n  this.findOptions.count = true;\n  delete this.findOptions.skip;\n  delete this.findOptions.limit;\n  return this.config.database.find(this.className, this.restWhere, this.findOptions)\n    .then((c) => {\n      this.response.count = c;\n    });\n};\n\n// Augments this.response with all pointers on an object\nRestQuery.prototype.handleIncludeAll = function() {\n  if (!this.includeAll) {\n    return;\n  }\n  return this.config.database.loadSchema()\n    .then(schemaController => schemaController.getOneSchema(this.className))\n    .then(schema => {\n      const includeFields = [];\n      const keyFields = [];\n      for (const field in schema.fields) {\n        if (schema.fields[field].type && schema.fields[field].type === 'Pointer') {\n          includeFields.push([field]);\n          keyFields.push(field);\n        }\n      }\n      // Add fields to include, keys, remove dups\n      this.include = [...new Set([...this.include, ...includeFields])];\n      // if this.keys not set, then all keys are already included\n      if (this.keys) {\n        this.keys = [...new Set([...this.keys, ...keyFields])];\n      }\n    });\n};\n\n// Augments this.response with data at the paths provided in this.include.\nRestQuery.prototype.handleInclude = function() {\n  if (this.include.length == 0) {\n    return;\n  }\n\n  var pathResponse = includePath(this.config, this.auth,\n    this.response, this.include[0], this.restOptions);\n  if (pathResponse.then) {\n    return pathResponse.then((newResponse) => {\n      this.response = newResponse;\n      this.include = this.include.slice(1);\n      return this.handleInclude();\n    });\n  } else if (this.include.length > 0) {\n    this.include = this.include.slice(1);\n    return this.handleInclude();\n  }\n\n  return pathResponse;\n};\n\n//Returns a promise of a processed set of results\nRestQuery.prototype.runAfterFindTrigger = function() {\n  if (!this.response) {\n    return;\n  }\n  // Avoid doing any setup for triggers if there is no 'afterFind' trigger for this class.\n  const hasAfterFindHook = triggers.triggerExists(this.className, triggers.Types.afterFind, this.config.applicationId);\n  if (!hasAfterFindHook) {\n    return Promise.resolve();\n  }\n  // Skip Aggregate and Distinct Queries\n  if (this.findOptions.pipeline || this.findOptions.distinct) {\n    return Promise.resolve();\n  }\n  // Run afterFind trigger and set the new results\n  return triggers.maybeRunAfterFindTrigger(triggers.Types.afterFind, this.auth, this.className,this.response.results, this.config).then((results) => {\n    // Ensure we properly set the className back\n    if (this.redirectClassName) {\n      this.response.results = results.map((object) => {\n        if (object instanceof Parse.Object) {\n          object = object.toJSON();\n        }\n        object.className = this.redirectClassName;\n        return object;\n      });\n    } else {\n      this.response.results = results;\n    }\n  });\n};\n\n// Adds included values to the response.\n// Path is a list of field names.\n// Returns a promise for an augmented response.\nfunction includePath(config, auth, response, path, restOptions = {}) {\n  var pointers = findPointers(response.results, path);\n  if (pointers.length == 0) {\n    return response;\n  }\n  const pointersHash = {};\n  for (var pointer of pointers) {\n    if (!pointer) {\n      continue;\n    }\n    const className = pointer.className;\n    // only include the good pointers\n    if (className) {\n      pointersHash[className] = pointersHash[className] || new Set();\n      pointersHash[className].add(pointer.objectId);\n    }\n  }\n  const includeRestOptions = {};\n  if (restOptions.keys) {\n    const keys = new Set(restOptions.keys.split(','));\n    const keySet = Array.from(keys).reduce((set, key) => {\n      const keyPath = key.split('.');\n      let i = 0;\n      for (i; i < path.length; i++) {\n        if (path[i] != keyPath[i]) {\n          return set;\n        }\n      }\n      if (i < keyPath.length) {\n        set.add(keyPath[i]);\n      }\n      return set;\n    }, new Set());\n    if (keySet.size > 0) {\n      includeRestOptions.keys = Array.from(keySet).join(',');\n    }\n  }\n\n  if (restOptions.includeReadPreference) {\n    includeRestOptions.readPreference = restOptions.includeReadPreference;\n    includeRestOptions.includeReadPreference = restOptions.includeReadPreference;\n  }\n\n  const queryPromises = Object.keys(pointersHash).map((className) => {\n    const objectIds = Array.from(pointersHash[className]);\n    let where;\n    if (objectIds.length === 1) {\n      where = {'objectId': objectIds[0]};\n    } else {\n      where = {'objectId': {'$in': objectIds}};\n    }\n    var query = new RestQuery(config, auth, className, where, includeRestOptions);\n    return query.execute({op: 'get'}).then((results) => {\n      results.className = className;\n      return Promise.resolve(results);\n    })\n  })\n\n  // Get the objects for all these object ids\n  return Promise.all(queryPromises).then((responses) => {\n    var replace = responses.reduce((replace, includeResponse) => {\n      for (var obj of includeResponse.results) {\n        obj.__type = 'Object';\n        obj.className = includeResponse.className;\n\n        if (obj.className == \"_User\" && !auth.isMaster) {\n          delete obj.sessionToken;\n          delete obj.authData;\n        }\n        replace[obj.objectId] = obj;\n      }\n      return replace;\n    }, {})\n\n    var resp = {\n      results: replacePointers(response.results, path, replace)\n    };\n    if (response.count) {\n      resp.count = response.count;\n    }\n    return resp;\n  });\n}\n\n// Object may be a list of REST-format object to find pointers in, or\n// it may be a single object.\n// If the path yields things that aren't pointers, this throws an error.\n// Path is a list of fields to search into.\n// Returns a list of pointers in REST format.\nfunction findPointers(object, path) {\n  if (object instanceof Array) {\n    var answer = [];\n    for (var x of object) {\n      answer = answer.concat(findPointers(x, path));\n    }\n    return answer;\n  }\n\n  if (typeof object !== 'object' || !object) {\n    return [];\n  }\n\n  if (path.length == 0) {\n    if (object === null || object.__type == 'Pointer') {\n      return [object];\n    }\n    return [];\n  }\n\n  var subobject = object[path[0]];\n  if (!subobject) {\n    return [];\n  }\n  return findPointers(subobject, path.slice(1));\n}\n\n// Object may be a list of REST-format objects to replace pointers\n// in, or it may be a single object.\n// Path is a list of fields to search into.\n// replace is a map from object id -> object.\n// Returns something analogous to object, but with the appropriate\n// pointers inflated.\nfunction replacePointers(object, path, replace) {\n  if (object instanceof Array) {\n    return object.map((obj) => replacePointers(obj, path, replace))\n      .filter((obj) => typeof obj !== 'undefined');\n  }\n\n  if (typeof object !== 'object' || !object) {\n    return object;\n  }\n\n  if (path.length === 0) {\n    if (object && object.__type === 'Pointer') {\n      return replace[object.objectId];\n    }\n    return object;\n  }\n\n  var subobject = object[path[0]];\n  if (!subobject) {\n    return object;\n  }\n  var newsub = replacePointers(subobject, path.slice(1), replace);\n  var answer = {};\n  for (var key in object) {\n    if (key == path[0]) {\n      answer[key] = newsub;\n    } else {\n      answer[key] = object[key];\n    }\n  }\n  return answer;\n}\n\n// Finds a subobject that has the given key, if there is one.\n// Returns undefined otherwise.\nfunction findObjectWithKey(root, key) {\n  if (typeof root !== 'object') {\n    return;\n  }\n  if (root instanceof Array) {\n    for (var item of root) {\n      const answer = findObjectWithKey(item, key);\n      if (answer) {\n        return answer;\n      }\n    }\n  }\n  if (root && root[key]) {\n    return root;\n  }\n  for (var subkey in root) {\n    const answer = findObjectWithKey(root[subkey], key);\n    if (answer) {\n      return answer;\n    }\n  }\n}\n\nmodule.exports = RestQuery;\n"]}
1028
+ // For tests
1029
+ module.exports._UnsafeRestQuery = _UnsafeRestQuery;
1030
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["SchemaController","require","Parse","triggers","continueWhile","AlwaysSelectedKeys","enforceRoleSecurity","createSanitizedError","RestQuery","method","config","auth","className","restWhere","restOptions","clientSDK","runAfterFind","runBeforeFind","context","Method","find","get","includes","Error","INVALID_QUERY","isGet","result","maybeRunQueryTrigger","Types","beforeFind","Promise","resolve","_UnsafeRestQuery","Object","freeze","response","findOptions","isMaster","user","INVALID_SESSION_TOKEN","$and","__type","objectId","id","doCount","includeAll","include","keysForInclude","prototype","hasOwnProperty","call","keys","excludeKeys","length","split","filter","key","map","slice","lastIndexOf","join","option","concat","Array","from","Set","exclude","k","indexOf","fields","order","sort","reduce","sortMap","field","trim","score","$meta","paths","pathSet","memo","path","index","parts","s","a","b","redirectKey","redirectClassNameForKey","redirectClassName","INVALID_JSON","execute","executeOptions","then","buildRestWhere","denyProtectedFields","handleIncludeAll","handleExcludeKeys","runFind","runCount","handleInclude","runAfterFindTrigger","handleAuthAdapters","each","callback","limit","finished","query","results","forEach","assign","$gt","getUserAndRoleACL","validateClientClassCreation","replaceSelect","replaceDontSelect","replaceInQuery","replaceNotInQuery","replaceEquality","acl","getUserRoles","roles","database","newClassName","allowClientClassCreation","systemClasses","loadSchema","schemaController","hasClass","OPERATION_FORBIDDEN","transformInQuery","inQueryObject","values","push","isArray","findObjectWithKey","inQueryValue","where","additionalOptions","subqueryReadPreference","readPreference","subquery","transformNotInQuery","notInQueryObject","notInQueryValue","getDeepestObjectFromKey","json","idx","src","splice","transformSelect","selectObject","objects","selectValue","transformDontSelect","dontSelectObject","dontSelectValue","cleanResultAuthData","password","authData","provider","replaceEqualityConstraint","constraint","equalToObject","hasDirectConstraint","hasOperatorConstraint","options","op","explain","filesController","expandFilesInObject","r","count","skip","c","protectedFields","addProtectedFields","getOneSchema","schema","includeFields","keyFields","type","indexedResults","indexed","i","executionTree","current","node","children","recursiveExecutionTree","treeNode","pathResponse","includePath","newResponse","newObject","all","hasAfterFindHook","triggerExists","afterFind","applicationId","pipeline","distinct","parseQuery","Query","withJSON","maybeRunAfterFindTrigger","object","toJSON","authDataManager","pointers","findPointers","pointersHash","pointer","add","includeRestOptions","keySet","set","keyPath","size","excludeKeySet","includeReadPreference","queryPromises","objectIds","$in","responses","replace","includeResponse","obj","sessionToken","resp","replacePointers","x","flat","subobject","newsub","answer","root","item","subkey","module","exports"],"sources":["../src/RestQuery.js"],"sourcesContent":["// An object that encapsulates everything we need to run a 'find'\n// operation, encoded in the REST API format.\n\nvar SchemaController = require('./Controllers/SchemaController');\nvar Parse = require('parse/node').Parse;\nconst triggers = require('./triggers');\nconst { continueWhile } = require('parse/lib/node/promiseUtils');\nconst AlwaysSelectedKeys = ['objectId', 'createdAt', 'updatedAt', 'ACL'];\nconst { enforceRoleSecurity } = require('./SharedRest');\nconst { createSanitizedError } = require('./Error');\n\n// restOptions can include:\n//   skip\n//   limit\n//   order\n//   count\n//   include\n//   keys\n//   excludeKeys\n//   redirectClassNameForKey\n//   readPreference\n//   includeReadPreference\n//   subqueryReadPreference\n/**\n * Use to perform a query on a class. It will run security checks and triggers.\n * @param options\n * @param options.method {RestQuery.Method} The type of query to perform\n * @param options.config {ParseServerConfiguration} The server configuration\n * @param options.auth {Auth} The auth object for the request\n * @param options.className {string} The name of the class to query\n * @param options.restWhere {object} The where object for the query\n * @param options.restOptions {object} The options object for the query\n * @param options.clientSDK {string} The client SDK that is performing the query\n * @param options.runAfterFind {boolean} Whether to run the afterFind trigger\n * @param options.runBeforeFind {boolean} Whether to run the beforeFind trigger\n * @param options.context {object} The context object for the query\n * @returns {Promise<_UnsafeRestQuery>} A promise that is resolved with the _UnsafeRestQuery object\n */\nasync function RestQuery({\n  method,\n  config,\n  auth,\n  className,\n  restWhere = {},\n  restOptions = {},\n  clientSDK,\n  runAfterFind = true,\n  runBeforeFind = true,\n  context,\n}) {\n  if (![RestQuery.Method.find, RestQuery.Method.get].includes(method)) {\n    throw new Parse.Error(Parse.Error.INVALID_QUERY, 'bad query type');\n  }\n  const isGet = method === RestQuery.Method.get;\n  enforceRoleSecurity(method, className, auth, config);\n  const result = runBeforeFind\n    ? await triggers.maybeRunQueryTrigger(\n      triggers.Types.beforeFind,\n      className,\n      restWhere,\n      restOptions,\n      config,\n      auth,\n      context,\n      isGet\n    )\n    : Promise.resolve({ restWhere, restOptions });\n\n  return new _UnsafeRestQuery(\n    config,\n    auth,\n    className,\n    result.restWhere || restWhere,\n    result.restOptions || restOptions,\n    clientSDK,\n    runAfterFind,\n    context,\n    isGet\n  );\n}\n\nRestQuery.Method = Object.freeze({\n  get: 'get',\n  find: 'find',\n});\n\n/**\n * _UnsafeRestQuery is meant for specific internal usage only. When you need to skip security checks or some triggers.\n * Don't use it if you don't know what you are doing.\n * @param config\n * @param auth\n * @param className\n * @param restWhere\n * @param restOptions\n * @param clientSDK\n * @param runAfterFind\n * @param context\n */\nfunction _UnsafeRestQuery(\n  config,\n  auth,\n  className,\n  restWhere = {},\n  restOptions = {},\n  clientSDK,\n  runAfterFind = true,\n  context,\n  isGet\n) {\n  this.config = config;\n  this.auth = auth;\n  this.className = className;\n  this.restWhere = restWhere;\n  this.restOptions = restOptions;\n  this.clientSDK = clientSDK;\n  this.runAfterFind = runAfterFind;\n  this.response = null;\n  this.findOptions = {};\n  this.context = context || {};\n  this.isGet = isGet;\n  if (!this.auth.isMaster) {\n    if (this.className == '_Session') {\n      if (!this.auth.user) {\n        throw createSanitizedError(Parse.Error.INVALID_SESSION_TOKEN, 'Invalid session token', config);\n      }\n      this.restWhere = {\n        $and: [\n          this.restWhere,\n          {\n            user: {\n              __type: 'Pointer',\n              className: '_User',\n              objectId: this.auth.user.id,\n            },\n          },\n        ],\n      };\n    }\n  }\n\n  this.doCount = false;\n  this.includeAll = false;\n\n  // The format for this.include is not the same as the format for the\n  // include option - it's the paths we should include, in order,\n  // stored as arrays, taking into account that we need to include foo\n  // before including foo.bar. Also it should dedupe.\n  // For example, passing an arg of include=foo.bar,foo.baz could lead to\n  // this.include = [['foo'], ['foo', 'baz'], ['foo', 'bar']]\n  this.include = [];\n  let keysForInclude = '';\n\n  // If we have keys, we probably want to force some includes (n-1 level)\n  // See issue: https://github.com/parse-community/parse-server/issues/3185\n  if (Object.prototype.hasOwnProperty.call(restOptions, 'keys')) {\n    keysForInclude = restOptions.keys;\n  }\n\n  // If we have keys, we probably want to force some includes (n-1 level)\n  // in order to exclude specific keys.\n  if (Object.prototype.hasOwnProperty.call(restOptions, 'excludeKeys')) {\n    keysForInclude += ',' + restOptions.excludeKeys;\n  }\n\n  if (keysForInclude.length > 0) {\n    keysForInclude = keysForInclude\n      .split(',')\n      .filter(key => {\n        // At least 2 components\n        return key.split('.').length > 1;\n      })\n      .map(key => {\n        // Slice the last component (a.b.c -> a.b)\n        // Otherwise we'll include one level too much.\n        return key.slice(0, key.lastIndexOf('.'));\n      })\n      .join(',');\n\n    // Concat the possibly present include string with the one from the keys\n    // Dedup / sorting is handle in 'include' case.\n    if (keysForInclude.length > 0) {\n      if (!restOptions.include || restOptions.include.length == 0) {\n        restOptions.include = keysForInclude;\n      } else {\n        restOptions.include += ',' + keysForInclude;\n      }\n    }\n  }\n\n  for (var option in restOptions) {\n    switch (option) {\n      case 'keys': {\n        const keys = restOptions.keys\n          .split(',')\n          .filter(key => key.length > 0)\n          .concat(AlwaysSelectedKeys);\n        this.keys = Array.from(new Set(keys));\n        break;\n      }\n      case 'excludeKeys': {\n        const exclude = restOptions.excludeKeys\n          .split(',')\n          .filter(k => AlwaysSelectedKeys.indexOf(k) < 0);\n        this.excludeKeys = Array.from(new Set(exclude));\n        break;\n      }\n      case 'count':\n        this.doCount = true;\n        break;\n      case 'includeAll':\n        this.includeAll = true;\n        break;\n      case 'explain':\n      case 'hint':\n      case 'distinct':\n      case 'pipeline':\n      case 'skip':\n      case 'limit':\n      case 'readPreference':\n      case 'comment':\n        this.findOptions[option] = restOptions[option];\n        break;\n      case 'order':\n        var fields = restOptions.order.split(',');\n        this.findOptions.sort = fields.reduce((sortMap, field) => {\n          field = field.trim();\n          if (field === '$score' || field === '-$score') {\n            sortMap.score = { $meta: 'textScore' };\n          } else if (field[0] == '-') {\n            sortMap[field.slice(1)] = -1;\n          } else {\n            sortMap[field] = 1;\n          }\n          return sortMap;\n        }, {});\n        break;\n      case 'include': {\n        const paths = restOptions.include.split(',');\n        if (paths.includes('*')) {\n          this.includeAll = true;\n          break;\n        }\n        // Load the existing includes (from keys)\n        const pathSet = paths.reduce((memo, path) => {\n          // Split each paths on . (a.b.c -> [a,b,c])\n          // reduce to create all paths\n          // ([a,b,c] -> {a: true, 'a.b': true, 'a.b.c': true})\n          return path.split('.').reduce((memo, path, index, parts) => {\n            memo[parts.slice(0, index + 1).join('.')] = true;\n            return memo;\n          }, memo);\n        }, {});\n\n        this.include = Object.keys(pathSet)\n          .map(s => {\n            return s.split('.');\n          })\n          .sort((a, b) => {\n            return a.length - b.length; // Sort by number of components\n          });\n        break;\n      }\n      case 'redirectClassNameForKey':\n        this.redirectKey = restOptions.redirectClassNameForKey;\n        this.redirectClassName = null;\n        break;\n      case 'includeReadPreference':\n      case 'subqueryReadPreference':\n        break;\n      default:\n        throw new Parse.Error(Parse.Error.INVALID_JSON, 'bad option: ' + option);\n    }\n  }\n}\n\n// A convenient method to perform all the steps of processing a query\n// in order.\n// Returns a promise for the response - an object with optional keys\n// 'results' and 'count'.\n// TODO: consolidate the replaceX functions\n_UnsafeRestQuery.prototype.execute = function (executeOptions) {\n  return Promise.resolve()\n    .then(() => {\n      return this.buildRestWhere();\n    })\n    .then(() => {\n      return this.denyProtectedFields();\n    })\n    .then(() => {\n      return this.handleIncludeAll();\n    })\n    .then(() => {\n      return this.handleExcludeKeys();\n    })\n    .then(() => {\n      return this.runFind(executeOptions);\n    })\n    .then(() => {\n      return this.runCount();\n    })\n    .then(() => {\n      return this.handleInclude();\n    })\n    .then(() => {\n      return this.runAfterFindTrigger();\n    })\n    .then(() => {\n      return this.handleAuthAdapters();\n    })\n    .then(() => {\n      return this.response;\n    });\n};\n\n_UnsafeRestQuery.prototype.each = function (callback) {\n  const { config, auth, className, restWhere, restOptions, clientSDK } = this;\n  // if the limit is set, use it\n  restOptions.limit = restOptions.limit || 100;\n  restOptions.order = 'objectId';\n  let finished = false;\n\n  return continueWhile(\n    () => {\n      return !finished;\n    },\n    async () => {\n      // Safe here to use _UnsafeRestQuery because the security was already\n      // checked during \"await RestQuery()\"\n      const query = new _UnsafeRestQuery(\n        config,\n        auth,\n        className,\n        restWhere,\n        restOptions,\n        clientSDK,\n        this.runAfterFind,\n        this.context\n      );\n      const { results } = await query.execute();\n      results.forEach(callback);\n      finished = results.length < restOptions.limit;\n      if (!finished) {\n        restWhere.objectId = Object.assign({}, restWhere.objectId, {\n          $gt: results[results.length - 1].objectId,\n        });\n      }\n    }\n  );\n};\n\n_UnsafeRestQuery.prototype.buildRestWhere = function () {\n  return Promise.resolve()\n    .then(() => {\n      return this.getUserAndRoleACL();\n    })\n    .then(() => {\n      return this.redirectClassNameForKey();\n    })\n    .then(() => {\n      return this.validateClientClassCreation();\n    })\n    .then(() => {\n      return this.replaceSelect();\n    })\n    .then(() => {\n      return this.replaceDontSelect();\n    })\n    .then(() => {\n      return this.replaceInQuery();\n    })\n    .then(() => {\n      return this.replaceNotInQuery();\n    })\n    .then(() => {\n      return this.replaceEquality();\n    });\n};\n\n// Uses the Auth object to get the list of roles, adds the user id\n_UnsafeRestQuery.prototype.getUserAndRoleACL = function () {\n  if (this.auth.isMaster) {\n    return Promise.resolve();\n  }\n\n  this.findOptions.acl = ['*'];\n\n  if (this.auth.user) {\n    return this.auth.getUserRoles().then(roles => {\n      this.findOptions.acl = this.findOptions.acl.concat(roles, [this.auth.user.id]);\n      return;\n    });\n  } else {\n    return Promise.resolve();\n  }\n};\n\n// Changes the className if redirectClassNameForKey is set.\n// Returns a promise.\n_UnsafeRestQuery.prototype.redirectClassNameForKey = function () {\n  if (!this.redirectKey) {\n    return Promise.resolve();\n  }\n\n  // We need to change the class name based on the schema\n  return this.config.database\n    .redirectClassNameForKey(this.className, this.redirectKey)\n    .then(newClassName => {\n      this.className = newClassName;\n      this.redirectClassName = newClassName;\n    });\n};\n\n// Validates this operation against the allowClientClassCreation config.\n_UnsafeRestQuery.prototype.validateClientClassCreation = function () {\n  if (\n    this.config.allowClientClassCreation === false &&\n    !this.auth.isMaster &&\n    SchemaController.systemClasses.indexOf(this.className) === -1\n  ) {\n    return this.config.database\n      .loadSchema()\n      .then(schemaController => schemaController.hasClass(this.className))\n      .then(hasClass => {\n        if (hasClass !== true) {\n          throw createSanitizedError(\n            Parse.Error.OPERATION_FORBIDDEN,\n            'This user is not allowed to access ' + 'non-existent class: ' + this.className,\n            this.config\n          );\n        }\n      });\n  } else {\n    return Promise.resolve();\n  }\n};\n\nfunction transformInQuery(inQueryObject, className, results) {\n  var values = [];\n  for (var result of results) {\n    values.push({\n      __type: 'Pointer',\n      className: className,\n      objectId: result.objectId,\n    });\n  }\n  delete inQueryObject['$inQuery'];\n  if (Array.isArray(inQueryObject['$in'])) {\n    inQueryObject['$in'] = inQueryObject['$in'].concat(values);\n  } else {\n    inQueryObject['$in'] = values;\n  }\n}\n\n// Replaces a $inQuery clause by running the subquery, if there is an\n// $inQuery clause.\n// The $inQuery clause turns into an $in with values that are just\n// pointers to the objects returned in the subquery.\n_UnsafeRestQuery.prototype.replaceInQuery = async function () {\n  var inQueryObject = findObjectWithKey(this.restWhere, '$inQuery');\n  if (!inQueryObject) {\n    return;\n  }\n\n  // The inQuery value must have precisely two keys - where and className\n  var inQueryValue = inQueryObject['$inQuery'];\n  if (!inQueryValue.where || !inQueryValue.className) {\n    throw new Parse.Error(Parse.Error.INVALID_QUERY, 'improper usage of $inQuery');\n  }\n\n  const additionalOptions = {\n    redirectClassNameForKey: inQueryValue.redirectClassNameForKey,\n  };\n\n  if (this.restOptions.subqueryReadPreference) {\n    additionalOptions.readPreference = this.restOptions.subqueryReadPreference;\n    additionalOptions.subqueryReadPreference = this.restOptions.subqueryReadPreference;\n  } else if (this.restOptions.readPreference) {\n    additionalOptions.readPreference = this.restOptions.readPreference;\n  }\n\n  const subquery = await RestQuery({\n    method: RestQuery.Method.find,\n    config: this.config,\n    auth: this.auth,\n    className: inQueryValue.className,\n    restWhere: inQueryValue.where,\n    restOptions: additionalOptions,\n    context: this.context,\n  });\n  return subquery.execute().then(response => {\n    transformInQuery(inQueryObject, subquery.className, response.results);\n    // Recurse to repeat\n    return this.replaceInQuery();\n  });\n};\n\nfunction transformNotInQuery(notInQueryObject, className, results) {\n  var values = [];\n  for (var result of results) {\n    values.push({\n      __type: 'Pointer',\n      className: className,\n      objectId: result.objectId,\n    });\n  }\n  delete notInQueryObject['$notInQuery'];\n  if (Array.isArray(notInQueryObject['$nin'])) {\n    notInQueryObject['$nin'] = notInQueryObject['$nin'].concat(values);\n  } else {\n    notInQueryObject['$nin'] = values;\n  }\n}\n\n// Replaces a $notInQuery clause by running the subquery, if there is an\n// $notInQuery clause.\n// The $notInQuery clause turns into a $nin with values that are just\n// pointers to the objects returned in the subquery.\n_UnsafeRestQuery.prototype.replaceNotInQuery = async function () {\n  var notInQueryObject = findObjectWithKey(this.restWhere, '$notInQuery');\n  if (!notInQueryObject) {\n    return;\n  }\n\n  // The notInQuery value must have precisely two keys - where and className\n  var notInQueryValue = notInQueryObject['$notInQuery'];\n  if (!notInQueryValue.where || !notInQueryValue.className) {\n    throw new Parse.Error(Parse.Error.INVALID_QUERY, 'improper usage of $notInQuery');\n  }\n\n  const additionalOptions = {\n    redirectClassNameForKey: notInQueryValue.redirectClassNameForKey,\n  };\n\n  if (this.restOptions.subqueryReadPreference) {\n    additionalOptions.readPreference = this.restOptions.subqueryReadPreference;\n    additionalOptions.subqueryReadPreference = this.restOptions.subqueryReadPreference;\n  } else if (this.restOptions.readPreference) {\n    additionalOptions.readPreference = this.restOptions.readPreference;\n  }\n\n  const subquery = await RestQuery({\n    method: RestQuery.Method.find,\n    config: this.config,\n    auth: this.auth,\n    className: notInQueryValue.className,\n    restWhere: notInQueryValue.where,\n    restOptions: additionalOptions,\n    context: this.context,\n  });\n\n  return subquery.execute().then(response => {\n    transformNotInQuery(notInQueryObject, subquery.className, response.results);\n    // Recurse to repeat\n    return this.replaceNotInQuery();\n  });\n};\n\n// Used to get the deepest object from json using dot notation.\nconst getDeepestObjectFromKey = (json, key, idx, src) => {\n  if (key in json) {\n    return json[key];\n  }\n  src.splice(1); // Exit Early\n};\n\nconst transformSelect = (selectObject, key, objects) => {\n  var values = [];\n  for (var result of objects) {\n    values.push(key.split('.').reduce(getDeepestObjectFromKey, result));\n  }\n  delete selectObject['$select'];\n  if (Array.isArray(selectObject['$in'])) {\n    selectObject['$in'] = selectObject['$in'].concat(values);\n  } else {\n    selectObject['$in'] = values;\n  }\n};\n\n// Replaces a $select clause by running the subquery, if there is a\n// $select clause.\n// The $select clause turns into an $in with values selected out of\n// the subquery.\n// Returns a possible-promise.\n_UnsafeRestQuery.prototype.replaceSelect = async function () {\n  var selectObject = findObjectWithKey(this.restWhere, '$select');\n  if (!selectObject) {\n    return;\n  }\n\n  // The select value must have precisely two keys - query and key\n  var selectValue = selectObject['$select'];\n  // iOS SDK don't send where if not set, let it pass\n  if (\n    !selectValue.query ||\n    !selectValue.key ||\n    typeof selectValue.query !== 'object' ||\n    !selectValue.query.className ||\n    Object.keys(selectValue).length !== 2\n  ) {\n    throw new Parse.Error(Parse.Error.INVALID_QUERY, 'improper usage of $select');\n  }\n\n  const additionalOptions = {\n    redirectClassNameForKey: selectValue.query.redirectClassNameForKey,\n  };\n\n  if (this.restOptions.subqueryReadPreference) {\n    additionalOptions.readPreference = this.restOptions.subqueryReadPreference;\n    additionalOptions.subqueryReadPreference = this.restOptions.subqueryReadPreference;\n  } else if (this.restOptions.readPreference) {\n    additionalOptions.readPreference = this.restOptions.readPreference;\n  }\n\n  const subquery = await RestQuery({\n    method: RestQuery.Method.find,\n    config: this.config,\n    auth: this.auth,\n    className: selectValue.query.className,\n    restWhere: selectValue.query.where,\n    restOptions: additionalOptions,\n    context: this.context,\n  });\n\n  return subquery.execute().then(response => {\n    transformSelect(selectObject, selectValue.key, response.results);\n    // Keep replacing $select clauses\n    return this.replaceSelect();\n  });\n};\n\nconst transformDontSelect = (dontSelectObject, key, objects) => {\n  var values = [];\n  for (var result of objects) {\n    values.push(key.split('.').reduce(getDeepestObjectFromKey, result));\n  }\n  delete dontSelectObject['$dontSelect'];\n  if (Array.isArray(dontSelectObject['$nin'])) {\n    dontSelectObject['$nin'] = dontSelectObject['$nin'].concat(values);\n  } else {\n    dontSelectObject['$nin'] = values;\n  }\n};\n\n// Replaces a $dontSelect clause by running the subquery, if there is a\n// $dontSelect clause.\n// The $dontSelect clause turns into an $nin with values selected out of\n// the subquery.\n// Returns a possible-promise.\n_UnsafeRestQuery.prototype.replaceDontSelect = async function () {\n  var dontSelectObject = findObjectWithKey(this.restWhere, '$dontSelect');\n  if (!dontSelectObject) {\n    return;\n  }\n\n  // The dontSelect value must have precisely two keys - query and key\n  var dontSelectValue = dontSelectObject['$dontSelect'];\n  if (\n    !dontSelectValue.query ||\n    !dontSelectValue.key ||\n    typeof dontSelectValue.query !== 'object' ||\n    !dontSelectValue.query.className ||\n    Object.keys(dontSelectValue).length !== 2\n  ) {\n    throw new Parse.Error(Parse.Error.INVALID_QUERY, 'improper usage of $dontSelect');\n  }\n  const additionalOptions = {\n    redirectClassNameForKey: dontSelectValue.query.redirectClassNameForKey,\n  };\n\n  if (this.restOptions.subqueryReadPreference) {\n    additionalOptions.readPreference = this.restOptions.subqueryReadPreference;\n    additionalOptions.subqueryReadPreference = this.restOptions.subqueryReadPreference;\n  } else if (this.restOptions.readPreference) {\n    additionalOptions.readPreference = this.restOptions.readPreference;\n  }\n\n  const subquery = await RestQuery({\n    method: RestQuery.Method.find,\n    config: this.config,\n    auth: this.auth,\n    className: dontSelectValue.query.className,\n    restWhere: dontSelectValue.query.where,\n    restOptions: additionalOptions,\n    context: this.context,\n  });\n\n  return subquery.execute().then(response => {\n    transformDontSelect(dontSelectObject, dontSelectValue.key, response.results);\n    // Keep replacing $dontSelect clauses\n    return this.replaceDontSelect();\n  });\n};\n\n_UnsafeRestQuery.prototype.cleanResultAuthData = function (result) {\n  delete result.password;\n  if (result.authData) {\n    Object.keys(result.authData).forEach(provider => {\n      if (result.authData[provider] === null) {\n        delete result.authData[provider];\n      }\n    });\n\n    if (Object.keys(result.authData).length == 0) {\n      delete result.authData;\n    }\n  }\n};\n\nconst replaceEqualityConstraint = constraint => {\n  if (typeof constraint !== 'object') {\n    return constraint;\n  }\n  const equalToObject = {};\n  let hasDirectConstraint = false;\n  let hasOperatorConstraint = false;\n  for (const key in constraint) {\n    if (key.indexOf('$') !== 0) {\n      hasDirectConstraint = true;\n      equalToObject[key] = constraint[key];\n    } else {\n      hasOperatorConstraint = true;\n    }\n  }\n  if (hasDirectConstraint && hasOperatorConstraint) {\n    constraint['$eq'] = equalToObject;\n    Object.keys(equalToObject).forEach(key => {\n      delete constraint[key];\n    });\n  }\n  return constraint;\n};\n\n_UnsafeRestQuery.prototype.replaceEquality = function () {\n  if (typeof this.restWhere !== 'object') {\n    return;\n  }\n  for (const key in this.restWhere) {\n    this.restWhere[key] = replaceEqualityConstraint(this.restWhere[key]);\n  }\n};\n\n// Returns a promise for whether it was successful.\n// Populates this.response with an object that only has 'results'.\n_UnsafeRestQuery.prototype.runFind = async function (options = {}) {\n  if (this.findOptions.limit === 0) {\n    this.response = { results: [] };\n    return Promise.resolve();\n  }\n  const findOptions = Object.assign({}, this.findOptions);\n  if (this.keys) {\n    findOptions.keys = this.keys.map(key => {\n      return key.split('.')[0];\n    });\n  }\n  if (options.op) {\n    findOptions.op = options.op;\n  }\n  const results = await this.config.database.find(this.className, this.restWhere, findOptions, this.auth);\n  if (this.className === '_User' && !findOptions.explain) {\n    for (var result of results) {\n      this.cleanResultAuthData(result);\n    }\n  }\n\n  await this.config.filesController.expandFilesInObject(this.config, results);\n\n  if (this.redirectClassName) {\n    for (var r of results) {\n      r.className = this.redirectClassName;\n    }\n  }\n  this.response = { results: results };\n};\n\n// Returns a promise for whether it was successful.\n// Populates this.response.count with the count\n_UnsafeRestQuery.prototype.runCount = function () {\n  if (!this.doCount) {\n    return;\n  }\n  this.findOptions.count = true;\n  delete this.findOptions.skip;\n  delete this.findOptions.limit;\n  return this.config.database.find(this.className, this.restWhere, this.findOptions).then(c => {\n    this.response.count = c;\n  });\n};\n\n_UnsafeRestQuery.prototype.denyProtectedFields = async function () {\n  if (this.auth.isMaster) {\n    return;\n  }\n  const schemaController = await this.config.database.loadSchema();\n  const protectedFields =\n    this.config.database.addProtectedFields(\n      schemaController,\n      this.className,\n      this.restWhere,\n      this.findOptions.acl,\n      this.auth,\n      this.findOptions\n    ) || [];\n  for (const key of protectedFields) {\n    if (this.restWhere[key]) {\n      throw createSanitizedError(\n        Parse.Error.OPERATION_FORBIDDEN,\n        `This user is not allowed to query ${key} on class ${this.className}`,\n        this.config\n      );\n    }\n  }\n};\n\n// Augments this.response with all pointers on an object\n_UnsafeRestQuery.prototype.handleIncludeAll = function () {\n  if (!this.includeAll) {\n    return;\n  }\n  return this.config.database\n    .loadSchema()\n    .then(schemaController => schemaController.getOneSchema(this.className))\n    .then(schema => {\n      const includeFields = [];\n      const keyFields = [];\n      for (const field in schema.fields) {\n        if (\n          (schema.fields[field].type && schema.fields[field].type === 'Pointer') ||\n          (schema.fields[field].type && schema.fields[field].type === 'Array')\n        ) {\n          includeFields.push([field]);\n          keyFields.push(field);\n        }\n      }\n      // Add fields to include, keys, remove dups\n      this.include = [...new Set([...this.include, ...includeFields])];\n      // if this.keys not set, then all keys are already included\n      if (this.keys) {\n        this.keys = [...new Set([...this.keys, ...keyFields])];\n      }\n    });\n};\n\n// Updates property `this.keys` to contain all keys but the ones unselected.\n_UnsafeRestQuery.prototype.handleExcludeKeys = function () {\n  if (!this.excludeKeys) {\n    return;\n  }\n  if (this.keys) {\n    this.keys = this.keys.filter(k => !this.excludeKeys.includes(k));\n    return;\n  }\n  return this.config.database\n    .loadSchema()\n    .then(schemaController => schemaController.getOneSchema(this.className))\n    .then(schema => {\n      const fields = Object.keys(schema.fields);\n      this.keys = fields.filter(k => !this.excludeKeys.includes(k));\n    });\n};\n\n// Augments this.response with data at the paths provided in this.include.\n_UnsafeRestQuery.prototype.handleInclude = async function () {\n  if (this.include.length == 0) {\n    return;\n  }\n\n  const indexedResults = this.response.results.reduce((indexed, result, i) => {\n    indexed[result.objectId] = i;\n    return indexed;\n  }, {});\n\n  // Build the execution tree\n  const executionTree = {}\n  this.include.forEach(path => {\n    let current = executionTree;\n    path.forEach((node) => {\n      if (!current[node]) {\n        current[node] = {\n          path,\n          children: {}\n        };\n      }\n      current = current[node].children\n    });\n  });\n\n  const recursiveExecutionTree = async (treeNode) => {\n    const { path, children } = treeNode;\n    const pathResponse = includePath(\n      this.config,\n      this.auth,\n      this.response,\n      path,\n      this.context,\n      this.restOptions,\n      this,\n    );\n    if (pathResponse.then) {\n      const newResponse = await pathResponse\n      newResponse.results.forEach(newObject => {\n        // We hydrate the root of each result with sub results\n        this.response.results[indexedResults[newObject.objectId]][path[0]] = newObject[path[0]];\n      })\n    }\n    return Promise.all(Object.values(children).map(recursiveExecutionTree));\n  }\n\n  await Promise.all(Object.values(executionTree).map(recursiveExecutionTree));\n  this.include = []\n};\n\n//Returns a promise of a processed set of results\n_UnsafeRestQuery.prototype.runAfterFindTrigger = function () {\n  if (!this.response) {\n    return;\n  }\n  if (!this.runAfterFind) {\n    return;\n  }\n  // Avoid doing any setup for triggers if there is no 'afterFind' trigger for this class.\n  const hasAfterFindHook = triggers.triggerExists(\n    this.className,\n    triggers.Types.afterFind,\n    this.config.applicationId\n  );\n  if (!hasAfterFindHook) {\n    return Promise.resolve();\n  }\n  // Skip Aggregate and Distinct Queries\n  if (this.findOptions.pipeline || this.findOptions.distinct) {\n    return Promise.resolve();\n  }\n\n  const json = Object.assign({}, this.restOptions);\n  json.where = this.restWhere;\n  const parseQuery = new Parse.Query(this.className);\n  parseQuery.withJSON(json);\n  // Run afterFind trigger and set the new results\n  return triggers\n    .maybeRunAfterFindTrigger(\n      triggers.Types.afterFind,\n      this.auth,\n      this.className,\n      this.response.results,\n      this.config,\n      parseQuery,\n      this.context,\n      this.isGet\n    )\n    .then(results => {\n      // Ensure we properly set the className back\n      if (this.redirectClassName) {\n        this.response.results = results.map(object => {\n          if (object instanceof Parse.Object) {\n            object = object.toJSON();\n          }\n          object.className = this.redirectClassName;\n          return object;\n        });\n      } else {\n        this.response.results = results;\n      }\n    });\n};\n\n_UnsafeRestQuery.prototype.handleAuthAdapters = async function () {\n  if (this.className !== '_User' || this.findOptions.explain) {\n    return;\n  }\n  await Promise.all(\n    this.response.results.map(result =>\n      this.config.authDataManager.runAfterFind(\n        { config: this.config, auth: this.auth },\n        result.authData\n      )\n    )\n  );\n};\n\n// Adds included values to the response.\n// Path is a list of field names.\n// Returns a promise for an augmented response.\nfunction includePath(config, auth, response, path, context, restOptions = {}) {\n  var pointers = findPointers(response.results, path);\n  if (pointers.length == 0) {\n    return response;\n  }\n  const pointersHash = {};\n  for (var pointer of pointers) {\n    if (!pointer) {\n      continue;\n    }\n    const className = pointer.className;\n    // only include the good pointers\n    if (className) {\n      pointersHash[className] = pointersHash[className] || new Set();\n      pointersHash[className].add(pointer.objectId);\n    }\n  }\n  const includeRestOptions = {};\n  if (restOptions.keys) {\n    const keys = new Set(restOptions.keys.split(','));\n    const keySet = Array.from(keys).reduce((set, key) => {\n      const keyPath = key.split('.');\n      let i = 0;\n      for (i; i < path.length; i++) {\n        if (path[i] != keyPath[i]) {\n          return set;\n        }\n      }\n      if (i < keyPath.length) {\n        set.add(keyPath[i]);\n      }\n      return set;\n    }, new Set());\n    if (keySet.size > 0) {\n      includeRestOptions.keys = Array.from(keySet).join(',');\n    }\n  }\n\n  if (restOptions.excludeKeys) {\n    const excludeKeys = new Set(restOptions.excludeKeys.split(','));\n    const excludeKeySet = Array.from(excludeKeys).reduce((set, key) => {\n      const keyPath = key.split('.');\n      let i = 0;\n      for (i; i < path.length; i++) {\n        if (path[i] != keyPath[i]) {\n          return set;\n        }\n      }\n      if (i == keyPath.length - 1) {\n        set.add(keyPath[i]);\n      }\n      return set;\n    }, new Set());\n    if (excludeKeySet.size > 0) {\n      includeRestOptions.excludeKeys = Array.from(excludeKeySet).join(',');\n    }\n  }\n\n  if (restOptions.includeReadPreference) {\n    includeRestOptions.readPreference = restOptions.includeReadPreference;\n    includeRestOptions.includeReadPreference = restOptions.includeReadPreference;\n  } else if (restOptions.readPreference) {\n    includeRestOptions.readPreference = restOptions.readPreference;\n  }\n  const queryPromises = Object.keys(pointersHash).map(async className => {\n    const objectIds = Array.from(pointersHash[className]);\n    let where;\n    if (objectIds.length === 1) {\n      where = { objectId: objectIds[0] };\n    } else {\n      where = { objectId: { $in: objectIds } };\n    }\n    const query = await RestQuery({\n      method: objectIds.length === 1 ? RestQuery.Method.get : RestQuery.Method.find,\n      config,\n      auth,\n      className,\n      restWhere: where,\n      restOptions: includeRestOptions,\n      context: context,\n    });\n    return query.execute({ op: 'get' }).then(results => {\n      results.className = className;\n      return Promise.resolve(results);\n    });\n  });\n\n  // Get the objects for all these object ids\n  return Promise.all(queryPromises).then(responses => {\n    var replace = responses.reduce((replace, includeResponse) => {\n      for (var obj of includeResponse.results) {\n        obj.__type = 'Object';\n        obj.className = includeResponse.className;\n\n        if (obj.className == '_User' && !auth.isMaster) {\n          delete obj.sessionToken;\n          delete obj.authData;\n        }\n        replace[obj.objectId] = obj;\n      }\n      return replace;\n    }, {});\n    var resp = {\n      results: replacePointers(response.results, path, replace),\n    };\n    if (response.count) {\n      resp.count = response.count;\n    }\n    return resp;\n  });\n}\n\n// Object may be a list of REST-format object to find pointers in, or\n// it may be a single object.\n// If the path yields things that aren't pointers, this throws an error.\n// Path is a list of fields to search into.\n// Returns a list of pointers in REST format.\nfunction findPointers(object, path) {\n  if (object instanceof Array) {\n    return object.map(x => findPointers(x, path)).flat();\n  }\n\n  if (typeof object !== 'object' || !object) {\n    return [];\n  }\n\n  if (path.length == 0) {\n    if (object === null || object.__type == 'Pointer') {\n      return [object];\n    }\n    return [];\n  }\n\n  var subobject = object[path[0]];\n  if (!subobject) {\n    return [];\n  }\n  return findPointers(subobject, path.slice(1));\n}\n\n// Object may be a list of REST-format objects to replace pointers\n// in, or it may be a single object.\n// Path is a list of fields to search into.\n// replace is a map from object id -> object.\n// Returns something analogous to object, but with the appropriate\n// pointers inflated.\nfunction replacePointers(object, path, replace) {\n  if (object instanceof Array) {\n    return object\n      .map(obj => replacePointers(obj, path, replace))\n      .filter(obj => typeof obj !== 'undefined');\n  }\n\n  if (typeof object !== 'object' || !object) {\n    return object;\n  }\n\n  if (path.length === 0) {\n    if (object && object.__type === 'Pointer') {\n      return replace[object.objectId];\n    }\n    return object;\n  }\n\n  var subobject = object[path[0]];\n  if (!subobject) {\n    return object;\n  }\n  var newsub = replacePointers(subobject, path.slice(1), replace);\n  var answer = {};\n  for (var key in object) {\n    if (key == path[0]) {\n      answer[key] = newsub;\n    } else {\n      answer[key] = object[key];\n    }\n  }\n  return answer;\n}\n\n// Finds a subobject that has the given key, if there is one.\n// Returns undefined otherwise.\nfunction findObjectWithKey(root, key) {\n  if (typeof root !== 'object') {\n    return;\n  }\n  if (root instanceof Array) {\n    for (var item of root) {\n      const answer = findObjectWithKey(item, key);\n      if (answer) {\n        return answer;\n      }\n    }\n  }\n  if (root && root[key]) {\n    return root;\n  }\n  for (var subkey in root) {\n    const answer = findObjectWithKey(root[subkey], key);\n    if (answer) {\n      return answer;\n    }\n  }\n}\n\nmodule.exports = RestQuery;\n// For tests\nmodule.exports._UnsafeRestQuery = _UnsafeRestQuery;\n"],"mappings":";;AAAA;AACA;;AAEA,IAAIA,gBAAgB,GAAGC,OAAO,CAAC,gCAAgC,CAAC;AAChE,IAAIC,KAAK,GAAGD,OAAO,CAAC,YAAY,CAAC,CAACC,KAAK;AACvC,MAAMC,QAAQ,GAAGF,OAAO,CAAC,YAAY,CAAC;AACtC,MAAM;EAAEG;AAAc,CAAC,GAAGH,OAAO,CAAC,6BAA6B,CAAC;AAChE,MAAMI,kBAAkB,GAAG,CAAC,UAAU,EAAE,WAAW,EAAE,WAAW,EAAE,KAAK,CAAC;AACxE,MAAM;EAAEC;AAAoB,CAAC,GAAGL,OAAO,CAAC,cAAc,CAAC;AACvD,MAAM;EAAEM;AAAqB,CAAC,GAAGN,OAAO,CAAC,SAAS,CAAC;;AAEnD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAeO,SAASA,CAAC;EACvBC,MAAM;EACNC,MAAM;EACNC,IAAI;EACJC,SAAS;EACTC,SAAS,GAAG,CAAC,CAAC;EACdC,WAAW,GAAG,CAAC,CAAC;EAChBC,SAAS;EACTC,YAAY,GAAG,IAAI;EACnBC,aAAa,GAAG,IAAI;EACpBC;AACF,CAAC,EAAE;EACD,IAAI,CAAC,CAACV,SAAS,CAACW,MAAM,CAACC,IAAI,EAAEZ,SAAS,CAACW,MAAM,CAACE,GAAG,CAAC,CAACC,QAAQ,CAACb,MAAM,CAAC,EAAE;IACnE,MAAM,IAAIP,KAAK,CAACqB,KAAK,CAACrB,KAAK,CAACqB,KAAK,CAACC,aAAa,EAAE,gBAAgB,CAAC;EACpE;EACA,MAAMC,KAAK,GAAGhB,MAAM,KAAKD,SAAS,CAACW,MAAM,CAACE,GAAG;EAC7Cf,mBAAmB,CAACG,MAAM,EAAEG,SAAS,EAAED,IAAI,EAAED,MAAM,CAAC;EACpD,MAAMgB,MAAM,GAAGT,aAAa,GACxB,MAAMd,QAAQ,CAACwB,oBAAoB,CACnCxB,QAAQ,CAACyB,KAAK,CAACC,UAAU,EACzBjB,SAAS,EACTC,SAAS,EACTC,WAAW,EACXJ,MAAM,EACNC,IAAI,EACJO,OAAO,EACPO,KACF,CAAC,GACCK,OAAO,CAACC,OAAO,CAAC;IAAElB,SAAS;IAAEC;EAAY,CAAC,CAAC;EAE/C,OAAO,IAAIkB,gBAAgB,CACzBtB,MAAM,EACNC,IAAI,EACJC,SAAS,EACTc,MAAM,CAACb,SAAS,IAAIA,SAAS,EAC7Ba,MAAM,CAACZ,WAAW,IAAIA,WAAW,EACjCC,SAAS,EACTC,YAAY,EACZE,OAAO,EACPO,KACF,CAAC;AACH;AAEAjB,SAAS,CAACW,MAAM,GAAGc,MAAM,CAACC,MAAM,CAAC;EAC/Bb,GAAG,EAAE,KAAK;EACVD,IAAI,EAAE;AACR,CAAC,CAAC;;AAEF;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAASY,gBAAgBA,CACvBtB,MAAM,EACNC,IAAI,EACJC,SAAS,EACTC,SAAS,GAAG,CAAC,CAAC,EACdC,WAAW,GAAG,CAAC,CAAC,EAChBC,SAAS,EACTC,YAAY,GAAG,IAAI,EACnBE,OAAO,EACPO,KAAK,EACL;EACA,IAAI,CAACf,MAAM,GAAGA,MAAM;EACpB,IAAI,CAACC,IAAI,GAAGA,IAAI;EAChB,IAAI,CAACC,SAAS,GAAGA,SAAS;EAC1B,IAAI,CAACC,SAAS,GAAGA,SAAS;EAC1B,IAAI,CAACC,WAAW,GAAGA,WAAW;EAC9B,IAAI,CAACC,SAAS,GAAGA,SAAS;EAC1B,IAAI,CAACC,YAAY,GAAGA,YAAY;EAChC,IAAI,CAACmB,QAAQ,GAAG,IAAI;EACpB,IAAI,CAACC,WAAW,GAAG,CAAC,CAAC;EACrB,IAAI,CAAClB,OAAO,GAAGA,OAAO,IAAI,CAAC,CAAC;EAC5B,IAAI,CAACO,KAAK,GAAGA,KAAK;EAClB,IAAI,CAAC,IAAI,CAACd,IAAI,CAAC0B,QAAQ,EAAE;IACvB,IAAI,IAAI,CAACzB,SAAS,IAAI,UAAU,EAAE;MAChC,IAAI,CAAC,IAAI,CAACD,IAAI,CAAC2B,IAAI,EAAE;QACnB,MAAM/B,oBAAoB,CAACL,KAAK,CAACqB,KAAK,CAACgB,qBAAqB,EAAE,uBAAuB,EAAE7B,MAAM,CAAC;MAChG;MACA,IAAI,CAACG,SAAS,GAAG;QACf2B,IAAI,EAAE,CACJ,IAAI,CAAC3B,SAAS,EACd;UACEyB,IAAI,EAAE;YACJG,MAAM,EAAE,SAAS;YACjB7B,SAAS,EAAE,OAAO;YAClB8B,QAAQ,EAAE,IAAI,CAAC/B,IAAI,CAAC2B,IAAI,CAACK;UAC3B;QACF,CAAC;MAEL,CAAC;IACH;EACF;EAEA,IAAI,CAACC,OAAO,GAAG,KAAK;EACpB,IAAI,CAACC,UAAU,GAAG,KAAK;;EAEvB;EACA;EACA;EACA;EACA;EACA;EACA,IAAI,CAACC,OAAO,GAAG,EAAE;EACjB,IAAIC,cAAc,GAAG,EAAE;;EAEvB;EACA;EACA,IAAId,MAAM,CAACe,SAAS,CAACC,cAAc,CAACC,IAAI,CAACpC,WAAW,EAAE,MAAM,CAAC,EAAE;IAC7DiC,cAAc,GAAGjC,WAAW,CAACqC,IAAI;EACnC;;EAEA;EACA;EACA,IAAIlB,MAAM,CAACe,SAAS,CAACC,cAAc,CAACC,IAAI,CAACpC,WAAW,EAAE,aAAa,CAAC,EAAE;IACpEiC,cAAc,IAAI,GAAG,GAAGjC,WAAW,CAACsC,WAAW;EACjD;EAEA,IAAIL,cAAc,CAACM,MAAM,GAAG,CAAC,EAAE;IAC7BN,cAAc,GAAGA,cAAc,CAC5BO,KAAK,CAAC,GAAG,CAAC,CACVC,MAAM,CAACC,GAAG,IAAI;MACb;MACA,OAAOA,GAAG,CAACF,KAAK,CAAC,GAAG,CAAC,CAACD,MAAM,GAAG,CAAC;IAClC,CAAC,CAAC,CACDI,GAAG,CAACD,GAAG,IAAI;MACV;MACA;MACA,OAAOA,GAAG,CAACE,KAAK,CAAC,CAAC,EAAEF,GAAG,CAACG,WAAW,CAAC,GAAG,CAAC,CAAC;IAC3C,CAAC,CAAC,CACDC,IAAI,CAAC,GAAG,CAAC;;IAEZ;IACA;IACA,IAAIb,cAAc,CAACM,MAAM,GAAG,CAAC,EAAE;MAC7B,IAAI,CAACvC,WAAW,CAACgC,OAAO,IAAIhC,WAAW,CAACgC,OAAO,CAACO,MAAM,IAAI,CAAC,EAAE;QAC3DvC,WAAW,CAACgC,OAAO,GAAGC,cAAc;MACtC,CAAC,MAAM;QACLjC,WAAW,CAACgC,OAAO,IAAI,GAAG,GAAGC,cAAc;MAC7C;IACF;EACF;EAEA,KAAK,IAAIc,MAAM,IAAI/C,WAAW,EAAE;IAC9B,QAAQ+C,MAAM;MACZ,KAAK,MAAM;QAAE;UACX,MAAMV,IAAI,GAAGrC,WAAW,CAACqC,IAAI,CAC1BG,KAAK,CAAC,GAAG,CAAC,CACVC,MAAM,CAACC,GAAG,IAAIA,GAAG,CAACH,MAAM,GAAG,CAAC,CAAC,CAC7BS,MAAM,CAACzD,kBAAkB,CAAC;UAC7B,IAAI,CAAC8C,IAAI,GAAGY,KAAK,CAACC,IAAI,CAAC,IAAIC,GAAG,CAACd,IAAI,CAAC,CAAC;UACrC;QACF;MACA,KAAK,aAAa;QAAE;UAClB,MAAMe,OAAO,GAAGpD,WAAW,CAACsC,WAAW,CACpCE,KAAK,CAAC,GAAG,CAAC,CACVC,MAAM,CAACY,CAAC,IAAI9D,kBAAkB,CAAC+D,OAAO,CAACD,CAAC,CAAC,GAAG,CAAC,CAAC;UACjD,IAAI,CAACf,WAAW,GAAGW,KAAK,CAACC,IAAI,CAAC,IAAIC,GAAG,CAACC,OAAO,CAAC,CAAC;UAC/C;QACF;MACA,KAAK,OAAO;QACV,IAAI,CAACtB,OAAO,GAAG,IAAI;QACnB;MACF,KAAK,YAAY;QACf,IAAI,CAACC,UAAU,GAAG,IAAI;QACtB;MACF,KAAK,SAAS;MACd,KAAK,MAAM;MACX,KAAK,UAAU;MACf,KAAK,UAAU;MACf,KAAK,MAAM;MACX,KAAK,OAAO;MACZ,KAAK,gBAAgB;MACrB,KAAK,SAAS;QACZ,IAAI,CAACT,WAAW,CAACyB,MAAM,CAAC,GAAG/C,WAAW,CAAC+C,MAAM,CAAC;QAC9C;MACF,KAAK,OAAO;QACV,IAAIQ,MAAM,GAAGvD,WAAW,CAACwD,KAAK,CAAChB,KAAK,CAAC,GAAG,CAAC;QACzC,IAAI,CAAClB,WAAW,CAACmC,IAAI,GAAGF,MAAM,CAACG,MAAM,CAAC,CAACC,OAAO,EAAEC,KAAK,KAAK;UACxDA,KAAK,GAAGA,KAAK,CAACC,IAAI,CAAC,CAAC;UACpB,IAAID,KAAK,KAAK,QAAQ,IAAIA,KAAK,KAAK,SAAS,EAAE;YAC7CD,OAAO,CAACG,KAAK,GAAG;cAAEC,KAAK,EAAE;YAAY,CAAC;UACxC,CAAC,MAAM,IAAIH,KAAK,CAAC,CAAC,CAAC,IAAI,GAAG,EAAE;YAC1BD,OAAO,CAACC,KAAK,CAAChB,KAAK,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC;UAC9B,CAAC,MAAM;YACLe,OAAO,CAACC,KAAK,CAAC,GAAG,CAAC;UACpB;UACA,OAAOD,OAAO;QAChB,CAAC,EAAE,CAAC,CAAC,CAAC;QACN;MACF,KAAK,SAAS;QAAE;UACd,MAAMK,KAAK,GAAGhE,WAAW,CAACgC,OAAO,CAACQ,KAAK,CAAC,GAAG,CAAC;UAC5C,IAAIwB,KAAK,CAACxD,QAAQ,CAAC,GAAG,CAAC,EAAE;YACvB,IAAI,CAACuB,UAAU,GAAG,IAAI;YACtB;UACF;UACA;UACA,MAAMkC,OAAO,GAAGD,KAAK,CAACN,MAAM,CAAC,CAACQ,IAAI,EAAEC,IAAI,KAAK;YAC3C;YACA;YACA;YACA,OAAOA,IAAI,CAAC3B,KAAK,CAAC,GAAG,CAAC,CAACkB,MAAM,CAAC,CAACQ,IAAI,EAAEC,IAAI,EAAEC,KAAK,EAAEC,KAAK,KAAK;cAC1DH,IAAI,CAACG,KAAK,CAACzB,KAAK,CAAC,CAAC,EAAEwB,KAAK,GAAG,CAAC,CAAC,CAACtB,IAAI,CAAC,GAAG,CAAC,CAAC,GAAG,IAAI;cAChD,OAAOoB,IAAI;YACb,CAAC,EAAEA,IAAI,CAAC;UACV,CAAC,EAAE,CAAC,CAAC,CAAC;UAEN,IAAI,CAAClC,OAAO,GAAGb,MAAM,CAACkB,IAAI,CAAC4B,OAAO,CAAC,CAChCtB,GAAG,CAAC2B,CAAC,IAAI;YACR,OAAOA,CAAC,CAAC9B,KAAK,CAAC,GAAG,CAAC;UACrB,CAAC,CAAC,CACDiB,IAAI,CAAC,CAACc,CAAC,EAAEC,CAAC,KAAK;YACd,OAAOD,CAAC,CAAChC,MAAM,GAAGiC,CAAC,CAACjC,MAAM,CAAC,CAAC;UAC9B,CAAC,CAAC;UACJ;QACF;MACA,KAAK,yBAAyB;QAC5B,IAAI,CAACkC,WAAW,GAAGzE,WAAW,CAAC0E,uBAAuB;QACtD,IAAI,CAACC,iBAAiB,GAAG,IAAI;QAC7B;MACF,KAAK,uBAAuB;MAC5B,KAAK,wBAAwB;QAC3B;MACF;QACE,MAAM,IAAIvF,KAAK,CAACqB,KAAK,CAACrB,KAAK,CAACqB,KAAK,CAACmE,YAAY,EAAE,cAAc,GAAG7B,MAAM,CAAC;IAC5E;EACF;AACF;;AAEA;AACA;AACA;AACA;AACA;AACA7B,gBAAgB,CAACgB,SAAS,CAAC2C,OAAO,GAAG,UAAUC,cAAc,EAAE;EAC7D,OAAO9D,OAAO,CAACC,OAAO,CAAC,CAAC,CACrB8D,IAAI,CAAC,MAAM;IACV,OAAO,IAAI,CAACC,cAAc,CAAC,CAAC;EAC9B,CAAC,CAAC,CACDD,IAAI,CAAC,MAAM;IACV,OAAO,IAAI,CAACE,mBAAmB,CAAC,CAAC;EACnC,CAAC,CAAC,CACDF,IAAI,CAAC,MAAM;IACV,OAAO,IAAI,CAACG,gBAAgB,CAAC,CAAC;EAChC,CAAC,CAAC,CACDH,IAAI,CAAC,MAAM;IACV,OAAO,IAAI,CAACI,iBAAiB,CAAC,CAAC;EACjC,CAAC,CAAC,CACDJ,IAAI,CAAC,MAAM;IACV,OAAO,IAAI,CAACK,OAAO,CAACN,cAAc,CAAC;EACrC,CAAC,CAAC,CACDC,IAAI,CAAC,MAAM;IACV,OAAO,IAAI,CAACM,QAAQ,CAAC,CAAC;EACxB,CAAC,CAAC,CACDN,IAAI,CAAC,MAAM;IACV,OAAO,IAAI,CAACO,aAAa,CAAC,CAAC;EAC7B,CAAC,CAAC,CACDP,IAAI,CAAC,MAAM;IACV,OAAO,IAAI,CAACQ,mBAAmB,CAAC,CAAC;EACnC,CAAC,CAAC,CACDR,IAAI,CAAC,MAAM;IACV,OAAO,IAAI,CAACS,kBAAkB,CAAC,CAAC;EAClC,CAAC,CAAC,CACDT,IAAI,CAAC,MAAM;IACV,OAAO,IAAI,CAAC1D,QAAQ;EACtB,CAAC,CAAC;AACN,CAAC;AAEDH,gBAAgB,CAACgB,SAAS,CAACuD,IAAI,GAAG,UAAUC,QAAQ,EAAE;EACpD,MAAM;IAAE9F,MAAM;IAAEC,IAAI;IAAEC,SAAS;IAAEC,SAAS;IAAEC,WAAW;IAAEC;EAAU,CAAC,GAAG,IAAI;EAC3E;EACAD,WAAW,CAAC2F,KAAK,GAAG3F,WAAW,CAAC2F,KAAK,IAAI,GAAG;EAC5C3F,WAAW,CAACwD,KAAK,GAAG,UAAU;EAC9B,IAAIoC,QAAQ,GAAG,KAAK;EAEpB,OAAOtG,aAAa,CAClB,MAAM;IACJ,OAAO,CAACsG,QAAQ;EAClB,CAAC,EACD,YAAY;IACV;IACA;IACA,MAAMC,KAAK,GAAG,IAAI3E,gBAAgB,CAChCtB,MAAM,EACNC,IAAI,EACJC,SAAS,EACTC,SAAS,EACTC,WAAW,EACXC,SAAS,EACT,IAAI,CAACC,YAAY,EACjB,IAAI,CAACE,OACP,CAAC;IACD,MAAM;MAAE0F;IAAQ,CAAC,GAAG,MAAMD,KAAK,CAAChB,OAAO,CAAC,CAAC;IACzCiB,OAAO,CAACC,OAAO,CAACL,QAAQ,CAAC;IACzBE,QAAQ,GAAGE,OAAO,CAACvD,MAAM,GAAGvC,WAAW,CAAC2F,KAAK;IAC7C,IAAI,CAACC,QAAQ,EAAE;MACb7F,SAAS,CAAC6B,QAAQ,GAAGT,MAAM,CAAC6E,MAAM,CAAC,CAAC,CAAC,EAAEjG,SAAS,CAAC6B,QAAQ,EAAE;QACzDqE,GAAG,EAAEH,OAAO,CAACA,OAAO,CAACvD,MAAM,GAAG,CAAC,CAAC,CAACX;MACnC,CAAC,CAAC;IACJ;EACF,CACF,CAAC;AACH,CAAC;AAEDV,gBAAgB,CAACgB,SAAS,CAAC8C,cAAc,GAAG,YAAY;EACtD,OAAOhE,OAAO,CAACC,OAAO,CAAC,CAAC,CACrB8D,IAAI,CAAC,MAAM;IACV,OAAO,IAAI,CAACmB,iBAAiB,CAAC,CAAC;EACjC,CAAC,CAAC,CACDnB,IAAI,CAAC,MAAM;IACV,OAAO,IAAI,CAACL,uBAAuB,CAAC,CAAC;EACvC,CAAC,CAAC,CACDK,IAAI,CAAC,MAAM;IACV,OAAO,IAAI,CAACoB,2BAA2B,CAAC,CAAC;EAC3C,CAAC,CAAC,CACDpB,IAAI,CAAC,MAAM;IACV,OAAO,IAAI,CAACqB,aAAa,CAAC,CAAC;EAC7B,CAAC,CAAC,CACDrB,IAAI,CAAC,MAAM;IACV,OAAO,IAAI,CAACsB,iBAAiB,CAAC,CAAC;EACjC,CAAC,CAAC,CACDtB,IAAI,CAAC,MAAM;IACV,OAAO,IAAI,CAACuB,cAAc,CAAC,CAAC;EAC9B,CAAC,CAAC,CACDvB,IAAI,CAAC,MAAM;IACV,OAAO,IAAI,CAACwB,iBAAiB,CAAC,CAAC;EACjC,CAAC,CAAC,CACDxB,IAAI,CAAC,MAAM;IACV,OAAO,IAAI,CAACyB,eAAe,CAAC,CAAC;EAC/B,CAAC,CAAC;AACN,CAAC;;AAED;AACAtF,gBAAgB,CAACgB,SAAS,CAACgE,iBAAiB,GAAG,YAAY;EACzD,IAAI,IAAI,CAACrG,IAAI,CAAC0B,QAAQ,EAAE;IACtB,OAAOP,OAAO,CAACC,OAAO,CAAC,CAAC;EAC1B;EAEA,IAAI,CAACK,WAAW,CAACmF,GAAG,GAAG,CAAC,GAAG,CAAC;EAE5B,IAAI,IAAI,CAAC5G,IAAI,CAAC2B,IAAI,EAAE;IAClB,OAAO,IAAI,CAAC3B,IAAI,CAAC6G,YAAY,CAAC,CAAC,CAAC3B,IAAI,CAAC4B,KAAK,IAAI;MAC5C,IAAI,CAACrF,WAAW,CAACmF,GAAG,GAAG,IAAI,CAACnF,WAAW,CAACmF,GAAG,CAACzD,MAAM,CAAC2D,KAAK,EAAE,CAAC,IAAI,CAAC9G,IAAI,CAAC2B,IAAI,CAACK,EAAE,CAAC,CAAC;MAC9E;IACF,CAAC,CAAC;EACJ,CAAC,MAAM;IACL,OAAOb,OAAO,CAACC,OAAO,CAAC,CAAC;EAC1B;AACF,CAAC;;AAED;AACA;AACAC,gBAAgB,CAACgB,SAAS,CAACwC,uBAAuB,GAAG,YAAY;EAC/D,IAAI,CAAC,IAAI,CAACD,WAAW,EAAE;IACrB,OAAOzD,OAAO,CAACC,OAAO,CAAC,CAAC;EAC1B;;EAEA;EACA,OAAO,IAAI,CAACrB,MAAM,CAACgH,QAAQ,CACxBlC,uBAAuB,CAAC,IAAI,CAAC5E,SAAS,EAAE,IAAI,CAAC2E,WAAW,CAAC,CACzDM,IAAI,CAAC8B,YAAY,IAAI;IACpB,IAAI,CAAC/G,SAAS,GAAG+G,YAAY;IAC7B,IAAI,CAAClC,iBAAiB,GAAGkC,YAAY;EACvC,CAAC,CAAC;AACN,CAAC;;AAED;AACA3F,gBAAgB,CAACgB,SAAS,CAACiE,2BAA2B,GAAG,YAAY;EACnE,IACE,IAAI,CAACvG,MAAM,CAACkH,wBAAwB,KAAK,KAAK,IAC9C,CAAC,IAAI,CAACjH,IAAI,CAAC0B,QAAQ,IACnBrC,gBAAgB,CAAC6H,aAAa,CAACzD,OAAO,CAAC,IAAI,CAACxD,SAAS,CAAC,KAAK,CAAC,CAAC,EAC7D;IACA,OAAO,IAAI,CAACF,MAAM,CAACgH,QAAQ,CACxBI,UAAU,CAAC,CAAC,CACZjC,IAAI,CAACkC,gBAAgB,IAAIA,gBAAgB,CAACC,QAAQ,CAAC,IAAI,CAACpH,SAAS,CAAC,CAAC,CACnEiF,IAAI,CAACmC,QAAQ,IAAI;MAChB,IAAIA,QAAQ,KAAK,IAAI,EAAE;QACrB,MAAMzH,oBAAoB,CACxBL,KAAK,CAACqB,KAAK,CAAC0G,mBAAmB,EAC/B,qCAAqC,GAAG,sBAAsB,GAAG,IAAI,CAACrH,SAAS,EAC/E,IAAI,CAACF,MACP,CAAC;MACH;IACF,CAAC,CAAC;EACN,CAAC,MAAM;IACL,OAAOoB,OAAO,CAACC,OAAO,CAAC,CAAC;EAC1B;AACF,CAAC;AAED,SAASmG,gBAAgBA,CAACC,aAAa,EAAEvH,SAAS,EAAEgG,OAAO,EAAE;EAC3D,IAAIwB,MAAM,GAAG,EAAE;EACf,KAAK,IAAI1G,MAAM,IAAIkF,OAAO,EAAE;IAC1BwB,MAAM,CAACC,IAAI,CAAC;MACV5F,MAAM,EAAE,SAAS;MACjB7B,SAAS,EAAEA,SAAS;MACpB8B,QAAQ,EAAEhB,MAAM,CAACgB;IACnB,CAAC,CAAC;EACJ;EACA,OAAOyF,aAAa,CAAC,UAAU,CAAC;EAChC,IAAIpE,KAAK,CAACuE,OAAO,CAACH,aAAa,CAAC,KAAK,CAAC,CAAC,EAAE;IACvCA,aAAa,CAAC,KAAK,CAAC,GAAGA,aAAa,CAAC,KAAK,CAAC,CAACrE,MAAM,CAACsE,MAAM,CAAC;EAC5D,CAAC,MAAM;IACLD,aAAa,CAAC,KAAK,CAAC,GAAGC,MAAM;EAC/B;AACF;;AAEA;AACA;AACA;AACA;AACApG,gBAAgB,CAACgB,SAAS,CAACoE,cAAc,GAAG,kBAAkB;EAC5D,IAAIe,aAAa,GAAGI,iBAAiB,CAAC,IAAI,CAAC1H,SAAS,EAAE,UAAU,CAAC;EACjE,IAAI,CAACsH,aAAa,EAAE;IAClB;EACF;;EAEA;EACA,IAAIK,YAAY,GAAGL,aAAa,CAAC,UAAU,CAAC;EAC5C,IAAI,CAACK,YAAY,CAACC,KAAK,IAAI,CAACD,YAAY,CAAC5H,SAAS,EAAE;IAClD,MAAM,IAAIV,KAAK,CAACqB,KAAK,CAACrB,KAAK,CAACqB,KAAK,CAACC,aAAa,EAAE,4BAA4B,CAAC;EAChF;EAEA,MAAMkH,iBAAiB,GAAG;IACxBlD,uBAAuB,EAAEgD,YAAY,CAAChD;EACxC,CAAC;EAED,IAAI,IAAI,CAAC1E,WAAW,CAAC6H,sBAAsB,EAAE;IAC3CD,iBAAiB,CAACE,cAAc,GAAG,IAAI,CAAC9H,WAAW,CAAC6H,sBAAsB;IAC1ED,iBAAiB,CAACC,sBAAsB,GAAG,IAAI,CAAC7H,WAAW,CAAC6H,sBAAsB;EACpF,CAAC,MAAM,IAAI,IAAI,CAAC7H,WAAW,CAAC8H,cAAc,EAAE;IAC1CF,iBAAiB,CAACE,cAAc,GAAG,IAAI,CAAC9H,WAAW,CAAC8H,cAAc;EACpE;EAEA,MAAMC,QAAQ,GAAG,MAAMrI,SAAS,CAAC;IAC/BC,MAAM,EAAED,SAAS,CAACW,MAAM,CAACC,IAAI;IAC7BV,MAAM,EAAE,IAAI,CAACA,MAAM;IACnBC,IAAI,EAAE,IAAI,CAACA,IAAI;IACfC,SAAS,EAAE4H,YAAY,CAAC5H,SAAS;IACjCC,SAAS,EAAE2H,YAAY,CAACC,KAAK;IAC7B3H,WAAW,EAAE4H,iBAAiB;IAC9BxH,OAAO,EAAE,IAAI,CAACA;EAChB,CAAC,CAAC;EACF,OAAO2H,QAAQ,CAAClD,OAAO,CAAC,CAAC,CAACE,IAAI,CAAC1D,QAAQ,IAAI;IACzC+F,gBAAgB,CAACC,aAAa,EAAEU,QAAQ,CAACjI,SAAS,EAAEuB,QAAQ,CAACyE,OAAO,CAAC;IACrE;IACA,OAAO,IAAI,CAACQ,cAAc,CAAC,CAAC;EAC9B,CAAC,CAAC;AACJ,CAAC;AAED,SAAS0B,mBAAmBA,CAACC,gBAAgB,EAAEnI,SAAS,EAAEgG,OAAO,EAAE;EACjE,IAAIwB,MAAM,GAAG,EAAE;EACf,KAAK,IAAI1G,MAAM,IAAIkF,OAAO,EAAE;IAC1BwB,MAAM,CAACC,IAAI,CAAC;MACV5F,MAAM,EAAE,SAAS;MACjB7B,SAAS,EAAEA,SAAS;MACpB8B,QAAQ,EAAEhB,MAAM,CAACgB;IACnB,CAAC,CAAC;EACJ;EACA,OAAOqG,gBAAgB,CAAC,aAAa,CAAC;EACtC,IAAIhF,KAAK,CAACuE,OAAO,CAACS,gBAAgB,CAAC,MAAM,CAAC,CAAC,EAAE;IAC3CA,gBAAgB,CAAC,MAAM,CAAC,GAAGA,gBAAgB,CAAC,MAAM,CAAC,CAACjF,MAAM,CAACsE,MAAM,CAAC;EACpE,CAAC,MAAM;IACLW,gBAAgB,CAAC,MAAM,CAAC,GAAGX,MAAM;EACnC;AACF;;AAEA;AACA;AACA;AACA;AACApG,gBAAgB,CAACgB,SAAS,CAACqE,iBAAiB,GAAG,kBAAkB;EAC/D,IAAI0B,gBAAgB,GAAGR,iBAAiB,CAAC,IAAI,CAAC1H,SAAS,EAAE,aAAa,CAAC;EACvE,IAAI,CAACkI,gBAAgB,EAAE;IACrB;EACF;;EAEA;EACA,IAAIC,eAAe,GAAGD,gBAAgB,CAAC,aAAa,CAAC;EACrD,IAAI,CAACC,eAAe,CAACP,KAAK,IAAI,CAACO,eAAe,CAACpI,SAAS,EAAE;IACxD,MAAM,IAAIV,KAAK,CAACqB,KAAK,CAACrB,KAAK,CAACqB,KAAK,CAACC,aAAa,EAAE,+BAA+B,CAAC;EACnF;EAEA,MAAMkH,iBAAiB,GAAG;IACxBlD,uBAAuB,EAAEwD,eAAe,CAACxD;EAC3C,CAAC;EAED,IAAI,IAAI,CAAC1E,WAAW,CAAC6H,sBAAsB,EAAE;IAC3CD,iBAAiB,CAACE,cAAc,GAAG,IAAI,CAAC9H,WAAW,CAAC6H,sBAAsB;IAC1ED,iBAAiB,CAACC,sBAAsB,GAAG,IAAI,CAAC7H,WAAW,CAAC6H,sBAAsB;EACpF,CAAC,MAAM,IAAI,IAAI,CAAC7H,WAAW,CAAC8H,cAAc,EAAE;IAC1CF,iBAAiB,CAACE,cAAc,GAAG,IAAI,CAAC9H,WAAW,CAAC8H,cAAc;EACpE;EAEA,MAAMC,QAAQ,GAAG,MAAMrI,SAAS,CAAC;IAC/BC,MAAM,EAAED,SAAS,CAACW,MAAM,CAACC,IAAI;IAC7BV,MAAM,EAAE,IAAI,CAACA,MAAM;IACnBC,IAAI,EAAE,IAAI,CAACA,IAAI;IACfC,SAAS,EAAEoI,eAAe,CAACpI,SAAS;IACpCC,SAAS,EAAEmI,eAAe,CAACP,KAAK;IAChC3H,WAAW,EAAE4H,iBAAiB;IAC9BxH,OAAO,EAAE,IAAI,CAACA;EAChB,CAAC,CAAC;EAEF,OAAO2H,QAAQ,CAAClD,OAAO,CAAC,CAAC,CAACE,IAAI,CAAC1D,QAAQ,IAAI;IACzC2G,mBAAmB,CAACC,gBAAgB,EAAEF,QAAQ,CAACjI,SAAS,EAAEuB,QAAQ,CAACyE,OAAO,CAAC;IAC3E;IACA,OAAO,IAAI,CAACS,iBAAiB,CAAC,CAAC;EACjC,CAAC,CAAC;AACJ,CAAC;;AAED;AACA,MAAM4B,uBAAuB,GAAGA,CAACC,IAAI,EAAE1F,GAAG,EAAE2F,GAAG,EAAEC,GAAG,KAAK;EACvD,IAAI5F,GAAG,IAAI0F,IAAI,EAAE;IACf,OAAOA,IAAI,CAAC1F,GAAG,CAAC;EAClB;EACA4F,GAAG,CAACC,MAAM,CAAC,CAAC,CAAC,CAAC,CAAC;AACjB,CAAC;AAED,MAAMC,eAAe,GAAGA,CAACC,YAAY,EAAE/F,GAAG,EAAEgG,OAAO,KAAK;EACtD,IAAIpB,MAAM,GAAG,EAAE;EACf,KAAK,IAAI1G,MAAM,IAAI8H,OAAO,EAAE;IAC1BpB,MAAM,CAACC,IAAI,CAAC7E,GAAG,CAACF,KAAK,CAAC,GAAG,CAAC,CAACkB,MAAM,CAACyE,uBAAuB,EAAEvH,MAAM,CAAC,CAAC;EACrE;EACA,OAAO6H,YAAY,CAAC,SAAS,CAAC;EAC9B,IAAIxF,KAAK,CAACuE,OAAO,CAACiB,YAAY,CAAC,KAAK,CAAC,CAAC,EAAE;IACtCA,YAAY,CAAC,KAAK,CAAC,GAAGA,YAAY,CAAC,KAAK,CAAC,CAACzF,MAAM,CAACsE,MAAM,CAAC;EAC1D,CAAC,MAAM;IACLmB,YAAY,CAAC,KAAK,CAAC,GAAGnB,MAAM;EAC9B;AACF,CAAC;;AAED;AACA;AACA;AACA;AACA;AACApG,gBAAgB,CAACgB,SAAS,CAACkE,aAAa,GAAG,kBAAkB;EAC3D,IAAIqC,YAAY,GAAGhB,iBAAiB,CAAC,IAAI,CAAC1H,SAAS,EAAE,SAAS,CAAC;EAC/D,IAAI,CAAC0I,YAAY,EAAE;IACjB;EACF;;EAEA;EACA,IAAIE,WAAW,GAAGF,YAAY,CAAC,SAAS,CAAC;EACzC;EACA,IACE,CAACE,WAAW,CAAC9C,KAAK,IAClB,CAAC8C,WAAW,CAACjG,GAAG,IAChB,OAAOiG,WAAW,CAAC9C,KAAK,KAAK,QAAQ,IACrC,CAAC8C,WAAW,CAAC9C,KAAK,CAAC/F,SAAS,IAC5BqB,MAAM,CAACkB,IAAI,CAACsG,WAAW,CAAC,CAACpG,MAAM,KAAK,CAAC,EACrC;IACA,MAAM,IAAInD,KAAK,CAACqB,KAAK,CAACrB,KAAK,CAACqB,KAAK,CAACC,aAAa,EAAE,2BAA2B,CAAC;EAC/E;EAEA,MAAMkH,iBAAiB,GAAG;IACxBlD,uBAAuB,EAAEiE,WAAW,CAAC9C,KAAK,CAACnB;EAC7C,CAAC;EAED,IAAI,IAAI,CAAC1E,WAAW,CAAC6H,sBAAsB,EAAE;IAC3CD,iBAAiB,CAACE,cAAc,GAAG,IAAI,CAAC9H,WAAW,CAAC6H,sBAAsB;IAC1ED,iBAAiB,CAACC,sBAAsB,GAAG,IAAI,CAAC7H,WAAW,CAAC6H,sBAAsB;EACpF,CAAC,MAAM,IAAI,IAAI,CAAC7H,WAAW,CAAC8H,cAAc,EAAE;IAC1CF,iBAAiB,CAACE,cAAc,GAAG,IAAI,CAAC9H,WAAW,CAAC8H,cAAc;EACpE;EAEA,MAAMC,QAAQ,GAAG,MAAMrI,SAAS,CAAC;IAC/BC,MAAM,EAAED,SAAS,CAACW,MAAM,CAACC,IAAI;IAC7BV,MAAM,EAAE,IAAI,CAACA,MAAM;IACnBC,IAAI,EAAE,IAAI,CAACA,IAAI;IACfC,SAAS,EAAE6I,WAAW,CAAC9C,KAAK,CAAC/F,SAAS;IACtCC,SAAS,EAAE4I,WAAW,CAAC9C,KAAK,CAAC8B,KAAK;IAClC3H,WAAW,EAAE4H,iBAAiB;IAC9BxH,OAAO,EAAE,IAAI,CAACA;EAChB,CAAC,CAAC;EAEF,OAAO2H,QAAQ,CAAClD,OAAO,CAAC,CAAC,CAACE,IAAI,CAAC1D,QAAQ,IAAI;IACzCmH,eAAe,CAACC,YAAY,EAAEE,WAAW,CAACjG,GAAG,EAAErB,QAAQ,CAACyE,OAAO,CAAC;IAChE;IACA,OAAO,IAAI,CAACM,aAAa,CAAC,CAAC;EAC7B,CAAC,CAAC;AACJ,CAAC;AAED,MAAMwC,mBAAmB,GAAGA,CAACC,gBAAgB,EAAEnG,GAAG,EAAEgG,OAAO,KAAK;EAC9D,IAAIpB,MAAM,GAAG,EAAE;EACf,KAAK,IAAI1G,MAAM,IAAI8H,OAAO,EAAE;IAC1BpB,MAAM,CAACC,IAAI,CAAC7E,GAAG,CAACF,KAAK,CAAC,GAAG,CAAC,CAACkB,MAAM,CAACyE,uBAAuB,EAAEvH,MAAM,CAAC,CAAC;EACrE;EACA,OAAOiI,gBAAgB,CAAC,aAAa,CAAC;EACtC,IAAI5F,KAAK,CAACuE,OAAO,CAACqB,gBAAgB,CAAC,MAAM,CAAC,CAAC,EAAE;IAC3CA,gBAAgB,CAAC,MAAM,CAAC,GAAGA,gBAAgB,CAAC,MAAM,CAAC,CAAC7F,MAAM,CAACsE,MAAM,CAAC;EACpE,CAAC,MAAM;IACLuB,gBAAgB,CAAC,MAAM,CAAC,GAAGvB,MAAM;EACnC;AACF,CAAC;;AAED;AACA;AACA;AACA;AACA;AACApG,gBAAgB,CAACgB,SAAS,CAACmE,iBAAiB,GAAG,kBAAkB;EAC/D,IAAIwC,gBAAgB,GAAGpB,iBAAiB,CAAC,IAAI,CAAC1H,SAAS,EAAE,aAAa,CAAC;EACvE,IAAI,CAAC8I,gBAAgB,EAAE;IACrB;EACF;;EAEA;EACA,IAAIC,eAAe,GAAGD,gBAAgB,CAAC,aAAa,CAAC;EACrD,IACE,CAACC,eAAe,CAACjD,KAAK,IACtB,CAACiD,eAAe,CAACpG,GAAG,IACpB,OAAOoG,eAAe,CAACjD,KAAK,KAAK,QAAQ,IACzC,CAACiD,eAAe,CAACjD,KAAK,CAAC/F,SAAS,IAChCqB,MAAM,CAACkB,IAAI,CAACyG,eAAe,CAAC,CAACvG,MAAM,KAAK,CAAC,EACzC;IACA,MAAM,IAAInD,KAAK,CAACqB,KAAK,CAACrB,KAAK,CAACqB,KAAK,CAACC,aAAa,EAAE,+BAA+B,CAAC;EACnF;EACA,MAAMkH,iBAAiB,GAAG;IACxBlD,uBAAuB,EAAEoE,eAAe,CAACjD,KAAK,CAACnB;EACjD,CAAC;EAED,IAAI,IAAI,CAAC1E,WAAW,CAAC6H,sBAAsB,EAAE;IAC3CD,iBAAiB,CAACE,cAAc,GAAG,IAAI,CAAC9H,WAAW,CAAC6H,sBAAsB;IAC1ED,iBAAiB,CAACC,sBAAsB,GAAG,IAAI,CAAC7H,WAAW,CAAC6H,sBAAsB;EACpF,CAAC,MAAM,IAAI,IAAI,CAAC7H,WAAW,CAAC8H,cAAc,EAAE;IAC1CF,iBAAiB,CAACE,cAAc,GAAG,IAAI,CAAC9H,WAAW,CAAC8H,cAAc;EACpE;EAEA,MAAMC,QAAQ,GAAG,MAAMrI,SAAS,CAAC;IAC/BC,MAAM,EAAED,SAAS,CAACW,MAAM,CAACC,IAAI;IAC7BV,MAAM,EAAE,IAAI,CAACA,MAAM;IACnBC,IAAI,EAAE,IAAI,CAACA,IAAI;IACfC,SAAS,EAAEgJ,eAAe,CAACjD,KAAK,CAAC/F,SAAS;IAC1CC,SAAS,EAAE+I,eAAe,CAACjD,KAAK,CAAC8B,KAAK;IACtC3H,WAAW,EAAE4H,iBAAiB;IAC9BxH,OAAO,EAAE,IAAI,CAACA;EAChB,CAAC,CAAC;EAEF,OAAO2H,QAAQ,CAAClD,OAAO,CAAC,CAAC,CAACE,IAAI,CAAC1D,QAAQ,IAAI;IACzCuH,mBAAmB,CAACC,gBAAgB,EAAEC,eAAe,CAACpG,GAAG,EAAErB,QAAQ,CAACyE,OAAO,CAAC;IAC5E;IACA,OAAO,IAAI,CAACO,iBAAiB,CAAC,CAAC;EACjC,CAAC,CAAC;AACJ,CAAC;AAEDnF,gBAAgB,CAACgB,SAAS,CAAC6G,mBAAmB,GAAG,UAAUnI,MAAM,EAAE;EACjE,OAAOA,MAAM,CAACoI,QAAQ;EACtB,IAAIpI,MAAM,CAACqI,QAAQ,EAAE;IACnB9H,MAAM,CAACkB,IAAI,CAACzB,MAAM,CAACqI,QAAQ,CAAC,CAAClD,OAAO,CAACmD,QAAQ,IAAI;MAC/C,IAAItI,MAAM,CAACqI,QAAQ,CAACC,QAAQ,CAAC,KAAK,IAAI,EAAE;QACtC,OAAOtI,MAAM,CAACqI,QAAQ,CAACC,QAAQ,CAAC;MAClC;IACF,CAAC,CAAC;IAEF,IAAI/H,MAAM,CAACkB,IAAI,CAACzB,MAAM,CAACqI,QAAQ,CAAC,CAAC1G,MAAM,IAAI,CAAC,EAAE;MAC5C,OAAO3B,MAAM,CAACqI,QAAQ;IACxB;EACF;AACF,CAAC;AAED,MAAME,yBAAyB,GAAGC,UAAU,IAAI;EAC9C,IAAI,OAAOA,UAAU,KAAK,QAAQ,EAAE;IAClC,OAAOA,UAAU;EACnB;EACA,MAAMC,aAAa,GAAG,CAAC,CAAC;EACxB,IAAIC,mBAAmB,GAAG,KAAK;EAC/B,IAAIC,qBAAqB,GAAG,KAAK;EACjC,KAAK,MAAM7G,GAAG,IAAI0G,UAAU,EAAE;IAC5B,IAAI1G,GAAG,CAACY,OAAO,CAAC,GAAG,CAAC,KAAK,CAAC,EAAE;MAC1BgG,mBAAmB,GAAG,IAAI;MAC1BD,aAAa,CAAC3G,GAAG,CAAC,GAAG0G,UAAU,CAAC1G,GAAG,CAAC;IACtC,CAAC,MAAM;MACL6G,qBAAqB,GAAG,IAAI;IAC9B;EACF;EACA,IAAID,mBAAmB,IAAIC,qBAAqB,EAAE;IAChDH,UAAU,CAAC,KAAK,CAAC,GAAGC,aAAa;IACjClI,MAAM,CAACkB,IAAI,CAACgH,aAAa,CAAC,CAACtD,OAAO,CAACrD,GAAG,IAAI;MACxC,OAAO0G,UAAU,CAAC1G,GAAG,CAAC;IACxB,CAAC,CAAC;EACJ;EACA,OAAO0G,UAAU;AACnB,CAAC;AAEDlI,gBAAgB,CAACgB,SAAS,CAACsE,eAAe,GAAG,YAAY;EACvD,IAAI,OAAO,IAAI,CAACzG,SAAS,KAAK,QAAQ,EAAE;IACtC;EACF;EACA,KAAK,MAAM2C,GAAG,IAAI,IAAI,CAAC3C,SAAS,EAAE;IAChC,IAAI,CAACA,SAAS,CAAC2C,GAAG,CAAC,GAAGyG,yBAAyB,CAAC,IAAI,CAACpJ,SAAS,CAAC2C,GAAG,CAAC,CAAC;EACtE;AACF,CAAC;;AAED;AACA;AACAxB,gBAAgB,CAACgB,SAAS,CAACkD,OAAO,GAAG,gBAAgBoE,OAAO,GAAG,CAAC,CAAC,EAAE;EACjE,IAAI,IAAI,CAAClI,WAAW,CAACqE,KAAK,KAAK,CAAC,EAAE;IAChC,IAAI,CAACtE,QAAQ,GAAG;MAAEyE,OAAO,EAAE;IAAG,CAAC;IAC/B,OAAO9E,OAAO,CAACC,OAAO,CAAC,CAAC;EAC1B;EACA,MAAMK,WAAW,GAAGH,MAAM,CAAC6E,MAAM,CAAC,CAAC,CAAC,EAAE,IAAI,CAAC1E,WAAW,CAAC;EACvD,IAAI,IAAI,CAACe,IAAI,EAAE;IACbf,WAAW,CAACe,IAAI,GAAG,IAAI,CAACA,IAAI,CAACM,GAAG,CAACD,GAAG,IAAI;MACtC,OAAOA,GAAG,CAACF,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC;IAC1B,CAAC,CAAC;EACJ;EACA,IAAIgH,OAAO,CAACC,EAAE,EAAE;IACdnI,WAAW,CAACmI,EAAE,GAAGD,OAAO,CAACC,EAAE;EAC7B;EACA,MAAM3D,OAAO,GAAG,MAAM,IAAI,CAAClG,MAAM,CAACgH,QAAQ,CAACtG,IAAI,CAAC,IAAI,CAACR,SAAS,EAAE,IAAI,CAACC,SAAS,EAAEuB,WAAW,EAAE,IAAI,CAACzB,IAAI,CAAC;EACvG,IAAI,IAAI,CAACC,SAAS,KAAK,OAAO,IAAI,CAACwB,WAAW,CAACoI,OAAO,EAAE;IACtD,KAAK,IAAI9I,MAAM,IAAIkF,OAAO,EAAE;MAC1B,IAAI,CAACiD,mBAAmB,CAACnI,MAAM,CAAC;IAClC;EACF;EAEA,MAAM,IAAI,CAAChB,MAAM,CAAC+J,eAAe,CAACC,mBAAmB,CAAC,IAAI,CAAChK,MAAM,EAAEkG,OAAO,CAAC;EAE3E,IAAI,IAAI,CAACnB,iBAAiB,EAAE;IAC1B,KAAK,IAAIkF,CAAC,IAAI/D,OAAO,EAAE;MACrB+D,CAAC,CAAC/J,SAAS,GAAG,IAAI,CAAC6E,iBAAiB;IACtC;EACF;EACA,IAAI,CAACtD,QAAQ,GAAG;IAAEyE,OAAO,EAAEA;EAAQ,CAAC;AACtC,CAAC;;AAED;AACA;AACA5E,gBAAgB,CAACgB,SAAS,CAACmD,QAAQ,GAAG,YAAY;EAChD,IAAI,CAAC,IAAI,CAACvD,OAAO,EAAE;IACjB;EACF;EACA,IAAI,CAACR,WAAW,CAACwI,KAAK,GAAG,IAAI;EAC7B,OAAO,IAAI,CAACxI,WAAW,CAACyI,IAAI;EAC5B,OAAO,IAAI,CAACzI,WAAW,CAACqE,KAAK;EAC7B,OAAO,IAAI,CAAC/F,MAAM,CAACgH,QAAQ,CAACtG,IAAI,CAAC,IAAI,CAACR,SAAS,EAAE,IAAI,CAACC,SAAS,EAAE,IAAI,CAACuB,WAAW,CAAC,CAACyD,IAAI,CAACiF,CAAC,IAAI;IAC3F,IAAI,CAAC3I,QAAQ,CAACyI,KAAK,GAAGE,CAAC;EACzB,CAAC,CAAC;AACJ,CAAC;AAED9I,gBAAgB,CAACgB,SAAS,CAAC+C,mBAAmB,GAAG,kBAAkB;EACjE,IAAI,IAAI,CAACpF,IAAI,CAAC0B,QAAQ,EAAE;IACtB;EACF;EACA,MAAM0F,gBAAgB,GAAG,MAAM,IAAI,CAACrH,MAAM,CAACgH,QAAQ,CAACI,UAAU,CAAC,CAAC;EAChE,MAAMiD,eAAe,GACnB,IAAI,CAACrK,MAAM,CAACgH,QAAQ,CAACsD,kBAAkB,CACrCjD,gBAAgB,EAChB,IAAI,CAACnH,SAAS,EACd,IAAI,CAACC,SAAS,EACd,IAAI,CAACuB,WAAW,CAACmF,GAAG,EACpB,IAAI,CAAC5G,IAAI,EACT,IAAI,CAACyB,WACP,CAAC,IAAI,EAAE;EACT,KAAK,MAAMoB,GAAG,IAAIuH,eAAe,EAAE;IACjC,IAAI,IAAI,CAAClK,SAAS,CAAC2C,GAAG,CAAC,EAAE;MACvB,MAAMjD,oBAAoB,CACxBL,KAAK,CAACqB,KAAK,CAAC0G,mBAAmB,EAC/B,qCAAqCzE,GAAG,aAAa,IAAI,CAAC5C,SAAS,EAAE,EACrE,IAAI,CAACF,MACP,CAAC;IACH;EACF;AACF,CAAC;;AAED;AACAsB,gBAAgB,CAACgB,SAAS,CAACgD,gBAAgB,GAAG,YAAY;EACxD,IAAI,CAAC,IAAI,CAACnD,UAAU,EAAE;IACpB;EACF;EACA,OAAO,IAAI,CAACnC,MAAM,CAACgH,QAAQ,CACxBI,UAAU,CAAC,CAAC,CACZjC,IAAI,CAACkC,gBAAgB,IAAIA,gBAAgB,CAACkD,YAAY,CAAC,IAAI,CAACrK,SAAS,CAAC,CAAC,CACvEiF,IAAI,CAACqF,MAAM,IAAI;IACd,MAAMC,aAAa,GAAG,EAAE;IACxB,MAAMC,SAAS,GAAG,EAAE;IACpB,KAAK,MAAM1G,KAAK,IAAIwG,MAAM,CAAC7G,MAAM,EAAE;MACjC,IACG6G,MAAM,CAAC7G,MAAM,CAACK,KAAK,CAAC,CAAC2G,IAAI,IAAIH,MAAM,CAAC7G,MAAM,CAACK,KAAK,CAAC,CAAC2G,IAAI,KAAK,SAAS,IACpEH,MAAM,CAAC7G,MAAM,CAACK,KAAK,CAAC,CAAC2G,IAAI,IAAIH,MAAM,CAAC7G,MAAM,CAACK,KAAK,CAAC,CAAC2G,IAAI,KAAK,OAAQ,EACpE;QACAF,aAAa,CAAC9C,IAAI,CAAC,CAAC3D,KAAK,CAAC,CAAC;QAC3B0G,SAAS,CAAC/C,IAAI,CAAC3D,KAAK,CAAC;MACvB;IACF;IACA;IACA,IAAI,CAAC5B,OAAO,GAAG,CAAC,GAAG,IAAImB,GAAG,CAAC,CAAC,GAAG,IAAI,CAACnB,OAAO,EAAE,GAAGqI,aAAa,CAAC,CAAC,CAAC;IAChE;IACA,IAAI,IAAI,CAAChI,IAAI,EAAE;MACb,IAAI,CAACA,IAAI,GAAG,CAAC,GAAG,IAAIc,GAAG,CAAC,CAAC,GAAG,IAAI,CAACd,IAAI,EAAE,GAAGiI,SAAS,CAAC,CAAC,CAAC;IACxD;EACF,CAAC,CAAC;AACN,CAAC;;AAED;AACApJ,gBAAgB,CAACgB,SAAS,CAACiD,iBAAiB,GAAG,YAAY;EACzD,IAAI,CAAC,IAAI,CAAC7C,WAAW,EAAE;IACrB;EACF;EACA,IAAI,IAAI,CAACD,IAAI,EAAE;IACb,IAAI,CAACA,IAAI,GAAG,IAAI,CAACA,IAAI,CAACI,MAAM,CAACY,CAAC,IAAI,CAAC,IAAI,CAACf,WAAW,CAAC9B,QAAQ,CAAC6C,CAAC,CAAC,CAAC;IAChE;EACF;EACA,OAAO,IAAI,CAACzD,MAAM,CAACgH,QAAQ,CACxBI,UAAU,CAAC,CAAC,CACZjC,IAAI,CAACkC,gBAAgB,IAAIA,gBAAgB,CAACkD,YAAY,CAAC,IAAI,CAACrK,SAAS,CAAC,CAAC,CACvEiF,IAAI,CAACqF,MAAM,IAAI;IACd,MAAM7G,MAAM,GAAGpC,MAAM,CAACkB,IAAI,CAAC+H,MAAM,CAAC7G,MAAM,CAAC;IACzC,IAAI,CAAClB,IAAI,GAAGkB,MAAM,CAACd,MAAM,CAACY,CAAC,IAAI,CAAC,IAAI,CAACf,WAAW,CAAC9B,QAAQ,CAAC6C,CAAC,CAAC,CAAC;EAC/D,CAAC,CAAC;AACN,CAAC;;AAED;AACAnC,gBAAgB,CAACgB,SAAS,CAACoD,aAAa,GAAG,kBAAkB;EAC3D,IAAI,IAAI,CAACtD,OAAO,CAACO,MAAM,IAAI,CAAC,EAAE;IAC5B;EACF;EAEA,MAAMiI,cAAc,GAAG,IAAI,CAACnJ,QAAQ,CAACyE,OAAO,CAACpC,MAAM,CAAC,CAAC+G,OAAO,EAAE7J,MAAM,EAAE8J,CAAC,KAAK;IAC1ED,OAAO,CAAC7J,MAAM,CAACgB,QAAQ,CAAC,GAAG8I,CAAC;IAC5B,OAAOD,OAAO;EAChB,CAAC,EAAE,CAAC,CAAC,CAAC;;EAEN;EACA,MAAME,aAAa,GAAG,CAAC,CAAC;EACxB,IAAI,CAAC3I,OAAO,CAAC+D,OAAO,CAAC5B,IAAI,IAAI;IAC3B,IAAIyG,OAAO,GAAGD,aAAa;IAC3BxG,IAAI,CAAC4B,OAAO,CAAE8E,IAAI,IAAK;MACrB,IAAI,CAACD,OAAO,CAACC,IAAI,CAAC,EAAE;QAClBD,OAAO,CAACC,IAAI,CAAC,GAAG;UACd1G,IAAI;UACJ2G,QAAQ,EAAE,CAAC;QACb,CAAC;MACH;MACAF,OAAO,GAAGA,OAAO,CAACC,IAAI,CAAC,CAACC,QAAQ;IAClC,CAAC,CAAC;EACJ,CAAC,CAAC;EAEF,MAAMC,sBAAsB,GAAG,MAAOC,QAAQ,IAAK;IACjD,MAAM;MAAE7G,IAAI;MAAE2G;IAAS,CAAC,GAAGE,QAAQ;IACnC,MAAMC,YAAY,GAAGC,WAAW,CAC9B,IAAI,CAACtL,MAAM,EACX,IAAI,CAACC,IAAI,EACT,IAAI,CAACwB,QAAQ,EACb8C,IAAI,EACJ,IAAI,CAAC/D,OAAO,EACZ,IAAI,CAACJ,WAAW,EAChB,IACF,CAAC;IACD,IAAIiL,YAAY,CAAClG,IAAI,EAAE;MACrB,MAAMoG,WAAW,GAAG,MAAMF,YAAY;MACtCE,WAAW,CAACrF,OAAO,CAACC,OAAO,CAACqF,SAAS,IAAI;QACvC;QACA,IAAI,CAAC/J,QAAQ,CAACyE,OAAO,CAAC0E,cAAc,CAACY,SAAS,CAACxJ,QAAQ,CAAC,CAAC,CAACuC,IAAI,CAAC,CAAC,CAAC,CAAC,GAAGiH,SAAS,CAACjH,IAAI,CAAC,CAAC,CAAC,CAAC;MACzF,CAAC,CAAC;IACJ;IACA,OAAOnD,OAAO,CAACqK,GAAG,CAAClK,MAAM,CAACmG,MAAM,CAACwD,QAAQ,CAAC,CAACnI,GAAG,CAACoI,sBAAsB,CAAC,CAAC;EACzE,CAAC;EAED,MAAM/J,OAAO,CAACqK,GAAG,CAAClK,MAAM,CAACmG,MAAM,CAACqD,aAAa,CAAC,CAAChI,GAAG,CAACoI,sBAAsB,CAAC,CAAC;EAC3E,IAAI,CAAC/I,OAAO,GAAG,EAAE;AACnB,CAAC;;AAED;AACAd,gBAAgB,CAACgB,SAAS,CAACqD,mBAAmB,GAAG,YAAY;EAC3D,IAAI,CAAC,IAAI,CAAClE,QAAQ,EAAE;IAClB;EACF;EACA,IAAI,CAAC,IAAI,CAACnB,YAAY,EAAE;IACtB;EACF;EACA;EACA,MAAMoL,gBAAgB,GAAGjM,QAAQ,CAACkM,aAAa,CAC7C,IAAI,CAACzL,SAAS,EACdT,QAAQ,CAACyB,KAAK,CAAC0K,SAAS,EACxB,IAAI,CAAC5L,MAAM,CAAC6L,aACd,CAAC;EACD,IAAI,CAACH,gBAAgB,EAAE;IACrB,OAAOtK,OAAO,CAACC,OAAO,CAAC,CAAC;EAC1B;EACA;EACA,IAAI,IAAI,CAACK,WAAW,CAACoK,QAAQ,IAAI,IAAI,CAACpK,WAAW,CAACqK,QAAQ,EAAE;IAC1D,OAAO3K,OAAO,CAACC,OAAO,CAAC,CAAC;EAC1B;EAEA,MAAMmH,IAAI,GAAGjH,MAAM,CAAC6E,MAAM,CAAC,CAAC,CAAC,EAAE,IAAI,CAAChG,WAAW,CAAC;EAChDoI,IAAI,CAACT,KAAK,GAAG,IAAI,CAAC5H,SAAS;EAC3B,MAAM6L,UAAU,GAAG,IAAIxM,KAAK,CAACyM,KAAK,CAAC,IAAI,CAAC/L,SAAS,CAAC;EAClD8L,UAAU,CAACE,QAAQ,CAAC1D,IAAI,CAAC;EACzB;EACA,OAAO/I,QAAQ,CACZ0M,wBAAwB,CACvB1M,QAAQ,CAACyB,KAAK,CAAC0K,SAAS,EACxB,IAAI,CAAC3L,IAAI,EACT,IAAI,CAACC,SAAS,EACd,IAAI,CAACuB,QAAQ,CAACyE,OAAO,EACrB,IAAI,CAAClG,MAAM,EACXgM,UAAU,EACV,IAAI,CAACxL,OAAO,EACZ,IAAI,CAACO,KACP,CAAC,CACAoE,IAAI,CAACe,OAAO,IAAI;IACf;IACA,IAAI,IAAI,CAACnB,iBAAiB,EAAE;MAC1B,IAAI,CAACtD,QAAQ,CAACyE,OAAO,GAAGA,OAAO,CAACnD,GAAG,CAACqJ,MAAM,IAAI;QAC5C,IAAIA,MAAM,YAAY5M,KAAK,CAAC+B,MAAM,EAAE;UAClC6K,MAAM,GAAGA,MAAM,CAACC,MAAM,CAAC,CAAC;QAC1B;QACAD,MAAM,CAAClM,SAAS,GAAG,IAAI,CAAC6E,iBAAiB;QACzC,OAAOqH,MAAM;MACf,CAAC,CAAC;IACJ,CAAC,MAAM;MACL,IAAI,CAAC3K,QAAQ,CAACyE,OAAO,GAAGA,OAAO;IACjC;EACF,CAAC,CAAC;AACN,CAAC;AAED5E,gBAAgB,CAACgB,SAAS,CAACsD,kBAAkB,GAAG,kBAAkB;EAChE,IAAI,IAAI,CAAC1F,SAAS,KAAK,OAAO,IAAI,IAAI,CAACwB,WAAW,CAACoI,OAAO,EAAE;IAC1D;EACF;EACA,MAAM1I,OAAO,CAACqK,GAAG,CACf,IAAI,CAAChK,QAAQ,CAACyE,OAAO,CAACnD,GAAG,CAAC/B,MAAM,IAC9B,IAAI,CAAChB,MAAM,CAACsM,eAAe,CAAChM,YAAY,CACtC;IAAEN,MAAM,EAAE,IAAI,CAACA,MAAM;IAAEC,IAAI,EAAE,IAAI,CAACA;EAAK,CAAC,EACxCe,MAAM,CAACqI,QACT,CACF,CACF,CAAC;AACH,CAAC;;AAED;AACA;AACA;AACA,SAASiC,WAAWA,CAACtL,MAAM,EAAEC,IAAI,EAAEwB,QAAQ,EAAE8C,IAAI,EAAE/D,OAAO,EAAEJ,WAAW,GAAG,CAAC,CAAC,EAAE;EAC5E,IAAImM,QAAQ,GAAGC,YAAY,CAAC/K,QAAQ,CAACyE,OAAO,EAAE3B,IAAI,CAAC;EACnD,IAAIgI,QAAQ,CAAC5J,MAAM,IAAI,CAAC,EAAE;IACxB,OAAOlB,QAAQ;EACjB;EACA,MAAMgL,YAAY,GAAG,CAAC,CAAC;EACvB,KAAK,IAAIC,OAAO,IAAIH,QAAQ,EAAE;IAC5B,IAAI,CAACG,OAAO,EAAE;MACZ;IACF;IACA,MAAMxM,SAAS,GAAGwM,OAAO,CAACxM,SAAS;IACnC;IACA,IAAIA,SAAS,EAAE;MACbuM,YAAY,CAACvM,SAAS,CAAC,GAAGuM,YAAY,CAACvM,SAAS,CAAC,IAAI,IAAIqD,GAAG,CAAC,CAAC;MAC9DkJ,YAAY,CAACvM,SAAS,CAAC,CAACyM,GAAG,CAACD,OAAO,CAAC1K,QAAQ,CAAC;IAC/C;EACF;EACA,MAAM4K,kBAAkB,GAAG,CAAC,CAAC;EAC7B,IAAIxM,WAAW,CAACqC,IAAI,EAAE;IACpB,MAAMA,IAAI,GAAG,IAAIc,GAAG,CAACnD,WAAW,CAACqC,IAAI,CAACG,KAAK,CAAC,GAAG,CAAC,CAAC;IACjD,MAAMiK,MAAM,GAAGxJ,KAAK,CAACC,IAAI,CAACb,IAAI,CAAC,CAACqB,MAAM,CAAC,CAACgJ,GAAG,EAAEhK,GAAG,KAAK;MACnD,MAAMiK,OAAO,GAAGjK,GAAG,CAACF,KAAK,CAAC,GAAG,CAAC;MAC9B,IAAIkI,CAAC,GAAG,CAAC;MACT,KAAKA,CAAC,EAAEA,CAAC,GAAGvG,IAAI,CAAC5B,MAAM,EAAEmI,CAAC,EAAE,EAAE;QAC5B,IAAIvG,IAAI,CAACuG,CAAC,CAAC,IAAIiC,OAAO,CAACjC,CAAC,CAAC,EAAE;UACzB,OAAOgC,GAAG;QACZ;MACF;MACA,IAAIhC,CAAC,GAAGiC,OAAO,CAACpK,MAAM,EAAE;QACtBmK,GAAG,CAACH,GAAG,CAACI,OAAO,CAACjC,CAAC,CAAC,CAAC;MACrB;MACA,OAAOgC,GAAG;IACZ,CAAC,EAAE,IAAIvJ,GAAG,CAAC,CAAC,CAAC;IACb,IAAIsJ,MAAM,CAACG,IAAI,GAAG,CAAC,EAAE;MACnBJ,kBAAkB,CAACnK,IAAI,GAAGY,KAAK,CAACC,IAAI,CAACuJ,MAAM,CAAC,CAAC3J,IAAI,CAAC,GAAG,CAAC;IACxD;EACF;EAEA,IAAI9C,WAAW,CAACsC,WAAW,EAAE;IAC3B,MAAMA,WAAW,GAAG,IAAIa,GAAG,CAACnD,WAAW,CAACsC,WAAW,CAACE,KAAK,CAAC,GAAG,CAAC,CAAC;IAC/D,MAAMqK,aAAa,GAAG5J,KAAK,CAACC,IAAI,CAACZ,WAAW,CAAC,CAACoB,MAAM,CAAC,CAACgJ,GAAG,EAAEhK,GAAG,KAAK;MACjE,MAAMiK,OAAO,GAAGjK,GAAG,CAACF,KAAK,CAAC,GAAG,CAAC;MAC9B,IAAIkI,CAAC,GAAG,CAAC;MACT,KAAKA,CAAC,EAAEA,CAAC,GAAGvG,IAAI,CAAC5B,MAAM,EAAEmI,CAAC,EAAE,EAAE;QAC5B,IAAIvG,IAAI,CAACuG,CAAC,CAAC,IAAIiC,OAAO,CAACjC,CAAC,CAAC,EAAE;UACzB,OAAOgC,GAAG;QACZ;MACF;MACA,IAAIhC,CAAC,IAAIiC,OAAO,CAACpK,MAAM,GAAG,CAAC,EAAE;QAC3BmK,GAAG,CAACH,GAAG,CAACI,OAAO,CAACjC,CAAC,CAAC,CAAC;MACrB;MACA,OAAOgC,GAAG;IACZ,CAAC,EAAE,IAAIvJ,GAAG,CAAC,CAAC,CAAC;IACb,IAAI0J,aAAa,CAACD,IAAI,GAAG,CAAC,EAAE;MAC1BJ,kBAAkB,CAAClK,WAAW,GAAGW,KAAK,CAACC,IAAI,CAAC2J,aAAa,CAAC,CAAC/J,IAAI,CAAC,GAAG,CAAC;IACtE;EACF;EAEA,IAAI9C,WAAW,CAAC8M,qBAAqB,EAAE;IACrCN,kBAAkB,CAAC1E,cAAc,GAAG9H,WAAW,CAAC8M,qBAAqB;IACrEN,kBAAkB,CAACM,qBAAqB,GAAG9M,WAAW,CAAC8M,qBAAqB;EAC9E,CAAC,MAAM,IAAI9M,WAAW,CAAC8H,cAAc,EAAE;IACrC0E,kBAAkB,CAAC1E,cAAc,GAAG9H,WAAW,CAAC8H,cAAc;EAChE;EACA,MAAMiF,aAAa,GAAG5L,MAAM,CAACkB,IAAI,CAACgK,YAAY,CAAC,CAAC1J,GAAG,CAAC,MAAM7C,SAAS,IAAI;IACrE,MAAMkN,SAAS,GAAG/J,KAAK,CAACC,IAAI,CAACmJ,YAAY,CAACvM,SAAS,CAAC,CAAC;IACrD,IAAI6H,KAAK;IACT,IAAIqF,SAAS,CAACzK,MAAM,KAAK,CAAC,EAAE;MAC1BoF,KAAK,GAAG;QAAE/F,QAAQ,EAAEoL,SAAS,CAAC,CAAC;MAAE,CAAC;IACpC,CAAC,MAAM;MACLrF,KAAK,GAAG;QAAE/F,QAAQ,EAAE;UAAEqL,GAAG,EAAED;QAAU;MAAE,CAAC;IAC1C;IACA,MAAMnH,KAAK,GAAG,MAAMnG,SAAS,CAAC;MAC5BC,MAAM,EAAEqN,SAAS,CAACzK,MAAM,KAAK,CAAC,GAAG7C,SAAS,CAACW,MAAM,CAACE,GAAG,GAAGb,SAAS,CAACW,MAAM,CAACC,IAAI;MAC7EV,MAAM;MACNC,IAAI;MACJC,SAAS;MACTC,SAAS,EAAE4H,KAAK;MAChB3H,WAAW,EAAEwM,kBAAkB;MAC/BpM,OAAO,EAAEA;IACX,CAAC,CAAC;IACF,OAAOyF,KAAK,CAAChB,OAAO,CAAC;MAAE4E,EAAE,EAAE;IAAM,CAAC,CAAC,CAAC1E,IAAI,CAACe,OAAO,IAAI;MAClDA,OAAO,CAAChG,SAAS,GAAGA,SAAS;MAC7B,OAAOkB,OAAO,CAACC,OAAO,CAAC6E,OAAO,CAAC;IACjC,CAAC,CAAC;EACJ,CAAC,CAAC;;EAEF;EACA,OAAO9E,OAAO,CAACqK,GAAG,CAAC0B,aAAa,CAAC,CAAChI,IAAI,CAACmI,SAAS,IAAI;IAClD,IAAIC,OAAO,GAAGD,SAAS,CAACxJ,MAAM,CAAC,CAACyJ,OAAO,EAAEC,eAAe,KAAK;MAC3D,KAAK,IAAIC,GAAG,IAAID,eAAe,CAACtH,OAAO,EAAE;QACvCuH,GAAG,CAAC1L,MAAM,GAAG,QAAQ;QACrB0L,GAAG,CAACvN,SAAS,GAAGsN,eAAe,CAACtN,SAAS;QAEzC,IAAIuN,GAAG,CAACvN,SAAS,IAAI,OAAO,IAAI,CAACD,IAAI,CAAC0B,QAAQ,EAAE;UAC9C,OAAO8L,GAAG,CAACC,YAAY;UACvB,OAAOD,GAAG,CAACpE,QAAQ;QACrB;QACAkE,OAAO,CAACE,GAAG,CAACzL,QAAQ,CAAC,GAAGyL,GAAG;MAC7B;MACA,OAAOF,OAAO;IAChB,CAAC,EAAE,CAAC,CAAC,CAAC;IACN,IAAII,IAAI,GAAG;MACTzH,OAAO,EAAE0H,eAAe,CAACnM,QAAQ,CAACyE,OAAO,EAAE3B,IAAI,EAAEgJ,OAAO;IAC1D,CAAC;IACD,IAAI9L,QAAQ,CAACyI,KAAK,EAAE;MAClByD,IAAI,CAACzD,KAAK,GAAGzI,QAAQ,CAACyI,KAAK;IAC7B;IACA,OAAOyD,IAAI;EACb,CAAC,CAAC;AACJ;;AAEA;AACA;AACA;AACA;AACA;AACA,SAASnB,YAAYA,CAACJ,MAAM,EAAE7H,IAAI,EAAE;EAClC,IAAI6H,MAAM,YAAY/I,KAAK,EAAE;IAC3B,OAAO+I,MAAM,CAACrJ,GAAG,CAAC8K,CAAC,IAAIrB,YAAY,CAACqB,CAAC,EAAEtJ,IAAI,CAAC,CAAC,CAACuJ,IAAI,CAAC,CAAC;EACtD;EAEA,IAAI,OAAO1B,MAAM,KAAK,QAAQ,IAAI,CAACA,MAAM,EAAE;IACzC,OAAO,EAAE;EACX;EAEA,IAAI7H,IAAI,CAAC5B,MAAM,IAAI,CAAC,EAAE;IACpB,IAAIyJ,MAAM,KAAK,IAAI,IAAIA,MAAM,CAACrK,MAAM,IAAI,SAAS,EAAE;MACjD,OAAO,CAACqK,MAAM,CAAC;IACjB;IACA,OAAO,EAAE;EACX;EAEA,IAAI2B,SAAS,GAAG3B,MAAM,CAAC7H,IAAI,CAAC,CAAC,CAAC,CAAC;EAC/B,IAAI,CAACwJ,SAAS,EAAE;IACd,OAAO,EAAE;EACX;EACA,OAAOvB,YAAY,CAACuB,SAAS,EAAExJ,IAAI,CAACvB,KAAK,CAAC,CAAC,CAAC,CAAC;AAC/C;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS4K,eAAeA,CAACxB,MAAM,EAAE7H,IAAI,EAAEgJ,OAAO,EAAE;EAC9C,IAAInB,MAAM,YAAY/I,KAAK,EAAE;IAC3B,OAAO+I,MAAM,CACVrJ,GAAG,CAAC0K,GAAG,IAAIG,eAAe,CAACH,GAAG,EAAElJ,IAAI,EAAEgJ,OAAO,CAAC,CAAC,CAC/C1K,MAAM,CAAC4K,GAAG,IAAI,OAAOA,GAAG,KAAK,WAAW,CAAC;EAC9C;EAEA,IAAI,OAAOrB,MAAM,KAAK,QAAQ,IAAI,CAACA,MAAM,EAAE;IACzC,OAAOA,MAAM;EACf;EAEA,IAAI7H,IAAI,CAAC5B,MAAM,KAAK,CAAC,EAAE;IACrB,IAAIyJ,MAAM,IAAIA,MAAM,CAACrK,MAAM,KAAK,SAAS,EAAE;MACzC,OAAOwL,OAAO,CAACnB,MAAM,CAACpK,QAAQ,CAAC;IACjC;IACA,OAAOoK,MAAM;EACf;EAEA,IAAI2B,SAAS,GAAG3B,MAAM,CAAC7H,IAAI,CAAC,CAAC,CAAC,CAAC;EAC/B,IAAI,CAACwJ,SAAS,EAAE;IACd,OAAO3B,MAAM;EACf;EACA,IAAI4B,MAAM,GAAGJ,eAAe,CAACG,SAAS,EAAExJ,IAAI,CAACvB,KAAK,CAAC,CAAC,CAAC,EAAEuK,OAAO,CAAC;EAC/D,IAAIU,MAAM,GAAG,CAAC,CAAC;EACf,KAAK,IAAInL,GAAG,IAAIsJ,MAAM,EAAE;IACtB,IAAItJ,GAAG,IAAIyB,IAAI,CAAC,CAAC,CAAC,EAAE;MAClB0J,MAAM,CAACnL,GAAG,CAAC,GAAGkL,MAAM;IACtB,CAAC,MAAM;MACLC,MAAM,CAACnL,GAAG,CAAC,GAAGsJ,MAAM,CAACtJ,GAAG,CAAC;IAC3B;EACF;EACA,OAAOmL,MAAM;AACf;;AAEA;AACA;AACA,SAASpG,iBAAiBA,CAACqG,IAAI,EAAEpL,GAAG,EAAE;EACpC,IAAI,OAAOoL,IAAI,KAAK,QAAQ,EAAE;IAC5B;EACF;EACA,IAAIA,IAAI,YAAY7K,KAAK,EAAE;IACzB,KAAK,IAAI8K,IAAI,IAAID,IAAI,EAAE;MACrB,MAAMD,MAAM,GAAGpG,iBAAiB,CAACsG,IAAI,EAAErL,GAAG,CAAC;MAC3C,IAAImL,MAAM,EAAE;QACV,OAAOA,MAAM;MACf;IACF;EACF;EACA,IAAIC,IAAI,IAAIA,IAAI,CAACpL,GAAG,CAAC,EAAE;IACrB,OAAOoL,IAAI;EACb;EACA,KAAK,IAAIE,MAAM,IAAIF,IAAI,EAAE;IACvB,MAAMD,MAAM,GAAGpG,iBAAiB,CAACqG,IAAI,CAACE,MAAM,CAAC,EAAEtL,GAAG,CAAC;IACnD,IAAImL,MAAM,EAAE;MACV,OAAOA,MAAM;IACf;EACF;AACF;AAEAI,MAAM,CAACC,OAAO,GAAGxO,SAAS;AAC1B;AACAuO,MAAM,CAACC,OAAO,CAAChN,gBAAgB,GAAGA,gBAAgB","ignoreList":[]}