RestQuery.js 124 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001
  1. "use strict";
  2. // An object that encapsulates everything we need to run a 'find'
  3. // operation, encoded in the REST API format.
  4. var SchemaController = require('./Controllers/SchemaController');
  5. var Parse = require('parse/node').Parse;
  6. const triggers = require('./triggers');
  7. const {
  8. continueWhile
  9. } = require('parse/lib/node/promiseUtils');
  10. const AlwaysSelectedKeys = ['objectId', 'createdAt', 'updatedAt', 'ACL'];
  11. const {
  12. enforceRoleSecurity
  13. } = require('./SharedRest');
  14. // restOptions can include:
  15. // skip
  16. // limit
  17. // order
  18. // count
  19. // include
  20. // keys
  21. // excludeKeys
  22. // redirectClassNameForKey
  23. // readPreference
  24. // includeReadPreference
  25. // subqueryReadPreference
  26. /**
  27. * Use to perform a query on a class. It will run security checks and triggers.
  28. * @param options
  29. * @param options.method {RestQuery.Method} The type of query to perform
  30. * @param options.config {ParseServerConfiguration} The server configuration
  31. * @param options.auth {Auth} The auth object for the request
  32. * @param options.className {string} The name of the class to query
  33. * @param options.restWhere {object} The where object for the query
  34. * @param options.restOptions {object} The options object for the query
  35. * @param options.clientSDK {string} The client SDK that is performing the query
  36. * @param options.runAfterFind {boolean} Whether to run the afterFind trigger
  37. * @param options.runBeforeFind {boolean} Whether to run the beforeFind trigger
  38. * @param options.context {object} The context object for the query
  39. * @returns {Promise<_UnsafeRestQuery>} A promise that is resolved with the _UnsafeRestQuery object
  40. */
  41. async function RestQuery({
  42. method,
  43. config,
  44. auth,
  45. className,
  46. restWhere = {},
  47. restOptions = {},
  48. clientSDK,
  49. runAfterFind = true,
  50. runBeforeFind = true,
  51. context
  52. }) {
  53. if (![RestQuery.Method.find, RestQuery.Method.get].includes(method)) {
  54. throw new Parse.Error(Parse.Error.INVALID_QUERY, 'bad query type');
  55. }
  56. enforceRoleSecurity(method, className, auth);
  57. const result = runBeforeFind ? await triggers.maybeRunQueryTrigger(triggers.Types.beforeFind, className, restWhere, restOptions, config, auth, context, method === RestQuery.Method.get) : Promise.resolve({
  58. restWhere,
  59. restOptions
  60. });
  61. return new _UnsafeRestQuery(config, auth, className, result.restWhere || restWhere, result.restOptions || restOptions, clientSDK, runAfterFind, context);
  62. }
  63. RestQuery.Method = Object.freeze({
  64. get: 'get',
  65. find: 'find'
  66. });
  67. /**
  68. * _UnsafeRestQuery is meant for specific internal usage only. When you need to skip security checks or some triggers.
  69. * Don't use it if you don't know what you are doing.
  70. * @param config
  71. * @param auth
  72. * @param className
  73. * @param restWhere
  74. * @param restOptions
  75. * @param clientSDK
  76. * @param runAfterFind
  77. * @param context
  78. */
  79. function _UnsafeRestQuery(config, auth, className, restWhere = {}, restOptions = {}, clientSDK, runAfterFind = true, context) {
  80. this.config = config;
  81. this.auth = auth;
  82. this.className = className;
  83. this.restWhere = restWhere;
  84. this.restOptions = restOptions;
  85. this.clientSDK = clientSDK;
  86. this.runAfterFind = runAfterFind;
  87. this.response = null;
  88. this.findOptions = {};
  89. this.context = context || {};
  90. if (!this.auth.isMaster) {
  91. if (this.className == '_Session') {
  92. if (!this.auth.user) {
  93. throw new Parse.Error(Parse.Error.INVALID_SESSION_TOKEN, 'Invalid session token');
  94. }
  95. this.restWhere = {
  96. $and: [this.restWhere, {
  97. user: {
  98. __type: 'Pointer',
  99. className: '_User',
  100. objectId: this.auth.user.id
  101. }
  102. }]
  103. };
  104. }
  105. }
  106. this.doCount = false;
  107. this.includeAll = false;
  108. // The format for this.include is not the same as the format for the
  109. // include option - it's the paths we should include, in order,
  110. // stored as arrays, taking into account that we need to include foo
  111. // before including foo.bar. Also it should dedupe.
  112. // For example, passing an arg of include=foo.bar,foo.baz could lead to
  113. // this.include = [['foo'], ['foo', 'baz'], ['foo', 'bar']]
  114. this.include = [];
  115. let keysForInclude = '';
  116. // If we have keys, we probably want to force some includes (n-1 level)
  117. // See issue: https://github.com/parse-community/parse-server/issues/3185
  118. if (Object.prototype.hasOwnProperty.call(restOptions, 'keys')) {
  119. keysForInclude = restOptions.keys;
  120. }
  121. // If we have keys, we probably want to force some includes (n-1 level)
  122. // in order to exclude specific keys.
  123. if (Object.prototype.hasOwnProperty.call(restOptions, 'excludeKeys')) {
  124. keysForInclude += ',' + restOptions.excludeKeys;
  125. }
  126. if (keysForInclude.length > 0) {
  127. keysForInclude = keysForInclude.split(',').filter(key => {
  128. // At least 2 components
  129. return key.split('.').length > 1;
  130. }).map(key => {
  131. // Slice the last component (a.b.c -> a.b)
  132. // Otherwise we'll include one level too much.
  133. return key.slice(0, key.lastIndexOf('.'));
  134. }).join(',');
  135. // Concat the possibly present include string with the one from the keys
  136. // Dedup / sorting is handle in 'include' case.
  137. if (keysForInclude.length > 0) {
  138. if (!restOptions.include || restOptions.include.length == 0) {
  139. restOptions.include = keysForInclude;
  140. } else {
  141. restOptions.include += ',' + keysForInclude;
  142. }
  143. }
  144. }
  145. for (var option in restOptions) {
  146. switch (option) {
  147. case 'keys':
  148. {
  149. const keys = restOptions.keys.split(',').filter(key => key.length > 0).concat(AlwaysSelectedKeys);
  150. this.keys = Array.from(new Set(keys));
  151. break;
  152. }
  153. case 'excludeKeys':
  154. {
  155. const exclude = restOptions.excludeKeys.split(',').filter(k => AlwaysSelectedKeys.indexOf(k) < 0);
  156. this.excludeKeys = Array.from(new Set(exclude));
  157. break;
  158. }
  159. case 'count':
  160. this.doCount = true;
  161. break;
  162. case 'includeAll':
  163. this.includeAll = true;
  164. break;
  165. case 'explain':
  166. case 'hint':
  167. case 'distinct':
  168. case 'pipeline':
  169. case 'skip':
  170. case 'limit':
  171. case 'readPreference':
  172. case 'comment':
  173. this.findOptions[option] = restOptions[option];
  174. break;
  175. case 'order':
  176. var fields = restOptions.order.split(',');
  177. this.findOptions.sort = fields.reduce((sortMap, field) => {
  178. field = field.trim();
  179. if (field === '$score' || field === '-$score') {
  180. sortMap.score = {
  181. $meta: 'textScore'
  182. };
  183. } else if (field[0] == '-') {
  184. sortMap[field.slice(1)] = -1;
  185. } else {
  186. sortMap[field] = 1;
  187. }
  188. return sortMap;
  189. }, {});
  190. break;
  191. case 'include':
  192. {
  193. const paths = restOptions.include.split(',');
  194. if (paths.includes('*')) {
  195. this.includeAll = true;
  196. break;
  197. }
  198. // Load the existing includes (from keys)
  199. const pathSet = paths.reduce((memo, path) => {
  200. // Split each paths on . (a.b.c -> [a,b,c])
  201. // reduce to create all paths
  202. // ([a,b,c] -> {a: true, 'a.b': true, 'a.b.c': true})
  203. return path.split('.').reduce((memo, path, index, parts) => {
  204. memo[parts.slice(0, index + 1).join('.')] = true;
  205. return memo;
  206. }, memo);
  207. }, {});
  208. this.include = Object.keys(pathSet).map(s => {
  209. return s.split('.');
  210. }).sort((a, b) => {
  211. return a.length - b.length; // Sort by number of components
  212. });
  213. break;
  214. }
  215. case 'redirectClassNameForKey':
  216. this.redirectKey = restOptions.redirectClassNameForKey;
  217. this.redirectClassName = null;
  218. break;
  219. case 'includeReadPreference':
  220. case 'subqueryReadPreference':
  221. break;
  222. default:
  223. throw new Parse.Error(Parse.Error.INVALID_JSON, 'bad option: ' + option);
  224. }
  225. }
  226. }
  227. // A convenient method to perform all the steps of processing a query
  228. // in order.
  229. // Returns a promise for the response - an object with optional keys
  230. // 'results' and 'count'.
  231. // TODO: consolidate the replaceX functions
  232. _UnsafeRestQuery.prototype.execute = function (executeOptions) {
  233. return Promise.resolve().then(() => {
  234. return this.buildRestWhere();
  235. }).then(() => {
  236. return this.denyProtectedFields();
  237. }).then(() => {
  238. return this.handleIncludeAll();
  239. }).then(() => {
  240. return this.handleExcludeKeys();
  241. }).then(() => {
  242. return this.runFind(executeOptions);
  243. }).then(() => {
  244. return this.runCount();
  245. }).then(() => {
  246. return this.handleInclude();
  247. }).then(() => {
  248. return this.runAfterFindTrigger();
  249. }).then(() => {
  250. return this.handleAuthAdapters();
  251. }).then(() => {
  252. return this.response;
  253. });
  254. };
  255. _UnsafeRestQuery.prototype.each = function (callback) {
  256. const {
  257. config,
  258. auth,
  259. className,
  260. restWhere,
  261. restOptions,
  262. clientSDK
  263. } = this;
  264. // if the limit is set, use it
  265. restOptions.limit = restOptions.limit || 100;
  266. restOptions.order = 'objectId';
  267. let finished = false;
  268. return continueWhile(() => {
  269. return !finished;
  270. }, async () => {
  271. // Safe here to use _UnsafeRestQuery because the security was already
  272. // checked during "await RestQuery()"
  273. const query = new _UnsafeRestQuery(config, auth, className, restWhere, restOptions, clientSDK, this.runAfterFind, this.context);
  274. const {
  275. results
  276. } = await query.execute();
  277. results.forEach(callback);
  278. finished = results.length < restOptions.limit;
  279. if (!finished) {
  280. restWhere.objectId = Object.assign({}, restWhere.objectId, {
  281. $gt: results[results.length - 1].objectId
  282. });
  283. }
  284. });
  285. };
  286. _UnsafeRestQuery.prototype.buildRestWhere = function () {
  287. return Promise.resolve().then(() => {
  288. return this.getUserAndRoleACL();
  289. }).then(() => {
  290. return this.redirectClassNameForKey();
  291. }).then(() => {
  292. return this.validateClientClassCreation();
  293. }).then(() => {
  294. return this.replaceSelect();
  295. }).then(() => {
  296. return this.replaceDontSelect();
  297. }).then(() => {
  298. return this.replaceInQuery();
  299. }).then(() => {
  300. return this.replaceNotInQuery();
  301. }).then(() => {
  302. return this.replaceEquality();
  303. });
  304. };
  305. // Uses the Auth object to get the list of roles, adds the user id
  306. _UnsafeRestQuery.prototype.getUserAndRoleACL = function () {
  307. if (this.auth.isMaster) {
  308. return Promise.resolve();
  309. }
  310. this.findOptions.acl = ['*'];
  311. if (this.auth.user) {
  312. return this.auth.getUserRoles().then(roles => {
  313. this.findOptions.acl = this.findOptions.acl.concat(roles, [this.auth.user.id]);
  314. return;
  315. });
  316. } else {
  317. return Promise.resolve();
  318. }
  319. };
  320. // Changes the className if redirectClassNameForKey is set.
  321. // Returns a promise.
  322. _UnsafeRestQuery.prototype.redirectClassNameForKey = function () {
  323. if (!this.redirectKey) {
  324. return Promise.resolve();
  325. }
  326. // We need to change the class name based on the schema
  327. return this.config.database.redirectClassNameForKey(this.className, this.redirectKey).then(newClassName => {
  328. this.className = newClassName;
  329. this.redirectClassName = newClassName;
  330. });
  331. };
  332. // Validates this operation against the allowClientClassCreation config.
  333. _UnsafeRestQuery.prototype.validateClientClassCreation = function () {
  334. if (this.config.allowClientClassCreation === false && !this.auth.isMaster && SchemaController.systemClasses.indexOf(this.className) === -1) {
  335. return this.config.database.loadSchema().then(schemaController => schemaController.hasClass(this.className)).then(hasClass => {
  336. if (hasClass !== true) {
  337. throw new Parse.Error(Parse.Error.OPERATION_FORBIDDEN, 'This user is not allowed to access ' + 'non-existent class: ' + this.className);
  338. }
  339. });
  340. } else {
  341. return Promise.resolve();
  342. }
  343. };
  344. function transformInQuery(inQueryObject, className, results) {
  345. var values = [];
  346. for (var result of results) {
  347. values.push({
  348. __type: 'Pointer',
  349. className: className,
  350. objectId: result.objectId
  351. });
  352. }
  353. delete inQueryObject['$inQuery'];
  354. if (Array.isArray(inQueryObject['$in'])) {
  355. inQueryObject['$in'] = inQueryObject['$in'].concat(values);
  356. } else {
  357. inQueryObject['$in'] = values;
  358. }
  359. }
  360. // Replaces a $inQuery clause by running the subquery, if there is an
  361. // $inQuery clause.
  362. // The $inQuery clause turns into an $in with values that are just
  363. // pointers to the objects returned in the subquery.
  364. _UnsafeRestQuery.prototype.replaceInQuery = async function () {
  365. var inQueryObject = findObjectWithKey(this.restWhere, '$inQuery');
  366. if (!inQueryObject) {
  367. return;
  368. }
  369. // The inQuery value must have precisely two keys - where and className
  370. var inQueryValue = inQueryObject['$inQuery'];
  371. if (!inQueryValue.where || !inQueryValue.className) {
  372. throw new Parse.Error(Parse.Error.INVALID_QUERY, 'improper usage of $inQuery');
  373. }
  374. const additionalOptions = {
  375. redirectClassNameForKey: inQueryValue.redirectClassNameForKey
  376. };
  377. if (this.restOptions.subqueryReadPreference) {
  378. additionalOptions.readPreference = this.restOptions.subqueryReadPreference;
  379. additionalOptions.subqueryReadPreference = this.restOptions.subqueryReadPreference;
  380. } else if (this.restOptions.readPreference) {
  381. additionalOptions.readPreference = this.restOptions.readPreference;
  382. }
  383. const subquery = await RestQuery({
  384. method: RestQuery.Method.find,
  385. config: this.config,
  386. auth: this.auth,
  387. className: inQueryValue.className,
  388. restWhere: inQueryValue.where,
  389. restOptions: additionalOptions,
  390. context: this.context
  391. });
  392. return subquery.execute().then(response => {
  393. transformInQuery(inQueryObject, subquery.className, response.results);
  394. // Recurse to repeat
  395. return this.replaceInQuery();
  396. });
  397. };
  398. function transformNotInQuery(notInQueryObject, className, results) {
  399. var values = [];
  400. for (var result of results) {
  401. values.push({
  402. __type: 'Pointer',
  403. className: className,
  404. objectId: result.objectId
  405. });
  406. }
  407. delete notInQueryObject['$notInQuery'];
  408. if (Array.isArray(notInQueryObject['$nin'])) {
  409. notInQueryObject['$nin'] = notInQueryObject['$nin'].concat(values);
  410. } else {
  411. notInQueryObject['$nin'] = values;
  412. }
  413. }
  414. // Replaces a $notInQuery clause by running the subquery, if there is an
  415. // $notInQuery clause.
  416. // The $notInQuery clause turns into a $nin with values that are just
  417. // pointers to the objects returned in the subquery.
  418. _UnsafeRestQuery.prototype.replaceNotInQuery = async function () {
  419. var notInQueryObject = findObjectWithKey(this.restWhere, '$notInQuery');
  420. if (!notInQueryObject) {
  421. return;
  422. }
  423. // The notInQuery value must have precisely two keys - where and className
  424. var notInQueryValue = notInQueryObject['$notInQuery'];
  425. if (!notInQueryValue.where || !notInQueryValue.className) {
  426. throw new Parse.Error(Parse.Error.INVALID_QUERY, 'improper usage of $notInQuery');
  427. }
  428. const additionalOptions = {
  429. redirectClassNameForKey: notInQueryValue.redirectClassNameForKey
  430. };
  431. if (this.restOptions.subqueryReadPreference) {
  432. additionalOptions.readPreference = this.restOptions.subqueryReadPreference;
  433. additionalOptions.subqueryReadPreference = this.restOptions.subqueryReadPreference;
  434. } else if (this.restOptions.readPreference) {
  435. additionalOptions.readPreference = this.restOptions.readPreference;
  436. }
  437. const subquery = await RestQuery({
  438. method: RestQuery.Method.find,
  439. config: this.config,
  440. auth: this.auth,
  441. className: notInQueryValue.className,
  442. restWhere: notInQueryValue.where,
  443. restOptions: additionalOptions,
  444. context: this.context
  445. });
  446. return subquery.execute().then(response => {
  447. transformNotInQuery(notInQueryObject, subquery.className, response.results);
  448. // Recurse to repeat
  449. return this.replaceNotInQuery();
  450. });
  451. };
  452. // Used to get the deepest object from json using dot notation.
  453. const getDeepestObjectFromKey = (json, key, idx, src) => {
  454. if (key in json) {
  455. return json[key];
  456. }
  457. src.splice(1); // Exit Early
  458. };
  459. const transformSelect = (selectObject, key, objects) => {
  460. var values = [];
  461. for (var result of objects) {
  462. values.push(key.split('.').reduce(getDeepestObjectFromKey, result));
  463. }
  464. delete selectObject['$select'];
  465. if (Array.isArray(selectObject['$in'])) {
  466. selectObject['$in'] = selectObject['$in'].concat(values);
  467. } else {
  468. selectObject['$in'] = values;
  469. }
  470. };
  471. // Replaces a $select clause by running the subquery, if there is a
  472. // $select clause.
  473. // The $select clause turns into an $in with values selected out of
  474. // the subquery.
  475. // Returns a possible-promise.
  476. _UnsafeRestQuery.prototype.replaceSelect = async function () {
  477. var selectObject = findObjectWithKey(this.restWhere, '$select');
  478. if (!selectObject) {
  479. return;
  480. }
  481. // The select value must have precisely two keys - query and key
  482. var selectValue = selectObject['$select'];
  483. // iOS SDK don't send where if not set, let it pass
  484. if (!selectValue.query || !selectValue.key || typeof selectValue.query !== 'object' || !selectValue.query.className || Object.keys(selectValue).length !== 2) {
  485. throw new Parse.Error(Parse.Error.INVALID_QUERY, 'improper usage of $select');
  486. }
  487. const additionalOptions = {
  488. redirectClassNameForKey: selectValue.query.redirectClassNameForKey
  489. };
  490. if (this.restOptions.subqueryReadPreference) {
  491. additionalOptions.readPreference = this.restOptions.subqueryReadPreference;
  492. additionalOptions.subqueryReadPreference = this.restOptions.subqueryReadPreference;
  493. } else if (this.restOptions.readPreference) {
  494. additionalOptions.readPreference = this.restOptions.readPreference;
  495. }
  496. const subquery = await RestQuery({
  497. method: RestQuery.Method.find,
  498. config: this.config,
  499. auth: this.auth,
  500. className: selectValue.query.className,
  501. restWhere: selectValue.query.where,
  502. restOptions: additionalOptions,
  503. context: this.context
  504. });
  505. return subquery.execute().then(response => {
  506. transformSelect(selectObject, selectValue.key, response.results);
  507. // Keep replacing $select clauses
  508. return this.replaceSelect();
  509. });
  510. };
  511. const transformDontSelect = (dontSelectObject, key, objects) => {
  512. var values = [];
  513. for (var result of objects) {
  514. values.push(key.split('.').reduce(getDeepestObjectFromKey, result));
  515. }
  516. delete dontSelectObject['$dontSelect'];
  517. if (Array.isArray(dontSelectObject['$nin'])) {
  518. dontSelectObject['$nin'] = dontSelectObject['$nin'].concat(values);
  519. } else {
  520. dontSelectObject['$nin'] = values;
  521. }
  522. };
  523. // Replaces a $dontSelect clause by running the subquery, if there is a
  524. // $dontSelect clause.
  525. // The $dontSelect clause turns into an $nin with values selected out of
  526. // the subquery.
  527. // Returns a possible-promise.
  528. _UnsafeRestQuery.prototype.replaceDontSelect = async function () {
  529. var dontSelectObject = findObjectWithKey(this.restWhere, '$dontSelect');
  530. if (!dontSelectObject) {
  531. return;
  532. }
  533. // The dontSelect value must have precisely two keys - query and key
  534. var dontSelectValue = dontSelectObject['$dontSelect'];
  535. if (!dontSelectValue.query || !dontSelectValue.key || typeof dontSelectValue.query !== 'object' || !dontSelectValue.query.className || Object.keys(dontSelectValue).length !== 2) {
  536. throw new Parse.Error(Parse.Error.INVALID_QUERY, 'improper usage of $dontSelect');
  537. }
  538. const additionalOptions = {
  539. redirectClassNameForKey: dontSelectValue.query.redirectClassNameForKey
  540. };
  541. if (this.restOptions.subqueryReadPreference) {
  542. additionalOptions.readPreference = this.restOptions.subqueryReadPreference;
  543. additionalOptions.subqueryReadPreference = this.restOptions.subqueryReadPreference;
  544. } else if (this.restOptions.readPreference) {
  545. additionalOptions.readPreference = this.restOptions.readPreference;
  546. }
  547. const subquery = await RestQuery({
  548. method: RestQuery.Method.find,
  549. config: this.config,
  550. auth: this.auth,
  551. className: dontSelectValue.query.className,
  552. restWhere: dontSelectValue.query.where,
  553. restOptions: additionalOptions,
  554. context: this.context
  555. });
  556. return subquery.execute().then(response => {
  557. transformDontSelect(dontSelectObject, dontSelectValue.key, response.results);
  558. // Keep replacing $dontSelect clauses
  559. return this.replaceDontSelect();
  560. });
  561. };
  562. _UnsafeRestQuery.prototype.cleanResultAuthData = function (result) {
  563. delete result.password;
  564. if (result.authData) {
  565. Object.keys(result.authData).forEach(provider => {
  566. if (result.authData[provider] === null) {
  567. delete result.authData[provider];
  568. }
  569. });
  570. if (Object.keys(result.authData).length == 0) {
  571. delete result.authData;
  572. }
  573. }
  574. };
  575. const replaceEqualityConstraint = constraint => {
  576. if (typeof constraint !== 'object') {
  577. return constraint;
  578. }
  579. const equalToObject = {};
  580. let hasDirectConstraint = false;
  581. let hasOperatorConstraint = false;
  582. for (const key in constraint) {
  583. if (key.indexOf('$') !== 0) {
  584. hasDirectConstraint = true;
  585. equalToObject[key] = constraint[key];
  586. } else {
  587. hasOperatorConstraint = true;
  588. }
  589. }
  590. if (hasDirectConstraint && hasOperatorConstraint) {
  591. constraint['$eq'] = equalToObject;
  592. Object.keys(equalToObject).forEach(key => {
  593. delete constraint[key];
  594. });
  595. }
  596. return constraint;
  597. };
  598. _UnsafeRestQuery.prototype.replaceEquality = function () {
  599. if (typeof this.restWhere !== 'object') {
  600. return;
  601. }
  602. for (const key in this.restWhere) {
  603. this.restWhere[key] = replaceEqualityConstraint(this.restWhere[key]);
  604. }
  605. };
  606. // Returns a promise for whether it was successful.
  607. // Populates this.response with an object that only has 'results'.
  608. _UnsafeRestQuery.prototype.runFind = async function (options = {}) {
  609. if (this.findOptions.limit === 0) {
  610. this.response = {
  611. results: []
  612. };
  613. return Promise.resolve();
  614. }
  615. const findOptions = Object.assign({}, this.findOptions);
  616. if (this.keys) {
  617. findOptions.keys = this.keys.map(key => {
  618. return key.split('.')[0];
  619. });
  620. }
  621. if (options.op) {
  622. findOptions.op = options.op;
  623. }
  624. const results = await this.config.database.find(this.className, this.restWhere, findOptions, this.auth);
  625. if (this.className === '_User' && !findOptions.explain) {
  626. for (var result of results) {
  627. this.cleanResultAuthData(result);
  628. }
  629. }
  630. await this.config.filesController.expandFilesInObject(this.config, results);
  631. if (this.redirectClassName) {
  632. for (var r of results) {
  633. r.className = this.redirectClassName;
  634. }
  635. }
  636. this.response = {
  637. results: results
  638. };
  639. };
  640. // Returns a promise for whether it was successful.
  641. // Populates this.response.count with the count
  642. _UnsafeRestQuery.prototype.runCount = function () {
  643. if (!this.doCount) {
  644. return;
  645. }
  646. this.findOptions.count = true;
  647. delete this.findOptions.skip;
  648. delete this.findOptions.limit;
  649. return this.config.database.find(this.className, this.restWhere, this.findOptions).then(c => {
  650. this.response.count = c;
  651. });
  652. };
  653. _UnsafeRestQuery.prototype.denyProtectedFields = async function () {
  654. if (this.auth.isMaster) {
  655. return;
  656. }
  657. const schemaController = await this.config.database.loadSchema();
  658. const protectedFields = this.config.database.addProtectedFields(schemaController, this.className, this.restWhere, this.findOptions.acl, this.auth, this.findOptions) || [];
  659. for (const key of protectedFields) {
  660. if (this.restWhere[key]) {
  661. throw new Parse.Error(Parse.Error.OPERATION_FORBIDDEN, `This user is not allowed to query ${key} on class ${this.className}`);
  662. }
  663. }
  664. };
  665. // Augments this.response with all pointers on an object
  666. _UnsafeRestQuery.prototype.handleIncludeAll = function () {
  667. if (!this.includeAll) {
  668. return;
  669. }
  670. return this.config.database.loadSchema().then(schemaController => schemaController.getOneSchema(this.className)).then(schema => {
  671. const includeFields = [];
  672. const keyFields = [];
  673. for (const field in schema.fields) {
  674. if (schema.fields[field].type && schema.fields[field].type === 'Pointer' || schema.fields[field].type && schema.fields[field].type === 'Array') {
  675. includeFields.push([field]);
  676. keyFields.push(field);
  677. }
  678. }
  679. // Add fields to include, keys, remove dups
  680. this.include = [...new Set([...this.include, ...includeFields])];
  681. // if this.keys not set, then all keys are already included
  682. if (this.keys) {
  683. this.keys = [...new Set([...this.keys, ...keyFields])];
  684. }
  685. });
  686. };
  687. // Updates property `this.keys` to contain all keys but the ones unselected.
  688. _UnsafeRestQuery.prototype.handleExcludeKeys = function () {
  689. if (!this.excludeKeys) {
  690. return;
  691. }
  692. if (this.keys) {
  693. this.keys = this.keys.filter(k => !this.excludeKeys.includes(k));
  694. return;
  695. }
  696. return this.config.database.loadSchema().then(schemaController => schemaController.getOneSchema(this.className)).then(schema => {
  697. const fields = Object.keys(schema.fields);
  698. this.keys = fields.filter(k => !this.excludeKeys.includes(k));
  699. });
  700. };
  701. // Augments this.response with data at the paths provided in this.include.
  702. _UnsafeRestQuery.prototype.handleInclude = function () {
  703. if (this.include.length == 0) {
  704. return;
  705. }
  706. var pathResponse = includePath(this.config, this.auth, this.response, this.include[0], this.context, this.restOptions);
  707. if (pathResponse.then) {
  708. return pathResponse.then(newResponse => {
  709. this.response = newResponse;
  710. this.include = this.include.slice(1);
  711. return this.handleInclude();
  712. });
  713. } else if (this.include.length > 0) {
  714. this.include = this.include.slice(1);
  715. return this.handleInclude();
  716. }
  717. return pathResponse;
  718. };
  719. //Returns a promise of a processed set of results
  720. _UnsafeRestQuery.prototype.runAfterFindTrigger = function () {
  721. if (!this.response) {
  722. return;
  723. }
  724. if (!this.runAfterFind) {
  725. return;
  726. }
  727. // Avoid doing any setup for triggers if there is no 'afterFind' trigger for this class.
  728. const hasAfterFindHook = triggers.triggerExists(this.className, triggers.Types.afterFind, this.config.applicationId);
  729. if (!hasAfterFindHook) {
  730. return Promise.resolve();
  731. }
  732. // Skip Aggregate and Distinct Queries
  733. if (this.findOptions.pipeline || this.findOptions.distinct) {
  734. return Promise.resolve();
  735. }
  736. const json = Object.assign({}, this.restOptions);
  737. json.where = this.restWhere;
  738. const parseQuery = new Parse.Query(this.className);
  739. parseQuery.withJSON(json);
  740. // Run afterFind trigger and set the new results
  741. return triggers.maybeRunAfterFindTrigger(triggers.Types.afterFind, this.auth, this.className, this.response.results, this.config, parseQuery, this.context).then(results => {
  742. // Ensure we properly set the className back
  743. if (this.redirectClassName) {
  744. this.response.results = results.map(object => {
  745. if (object instanceof Parse.Object) {
  746. object = object.toJSON();
  747. }
  748. object.className = this.redirectClassName;
  749. return object;
  750. });
  751. } else {
  752. this.response.results = results;
  753. }
  754. });
  755. };
  756. _UnsafeRestQuery.prototype.handleAuthAdapters = async function () {
  757. if (this.className !== '_User' || this.findOptions.explain) {
  758. return;
  759. }
  760. await Promise.all(this.response.results.map(result => this.config.authDataManager.runAfterFind({
  761. config: this.config,
  762. auth: this.auth
  763. }, result.authData)));
  764. };
  765. // Adds included values to the response.
  766. // Path is a list of field names.
  767. // Returns a promise for an augmented response.
  768. function includePath(config, auth, response, path, context, restOptions = {}) {
  769. var pointers = findPointers(response.results, path);
  770. if (pointers.length == 0) {
  771. return response;
  772. }
  773. const pointersHash = {};
  774. for (var pointer of pointers) {
  775. if (!pointer) {
  776. continue;
  777. }
  778. const className = pointer.className;
  779. // only include the good pointers
  780. if (className) {
  781. pointersHash[className] = pointersHash[className] || new Set();
  782. pointersHash[className].add(pointer.objectId);
  783. }
  784. }
  785. const includeRestOptions = {};
  786. if (restOptions.keys) {
  787. const keys = new Set(restOptions.keys.split(','));
  788. const keySet = Array.from(keys).reduce((set, key) => {
  789. const keyPath = key.split('.');
  790. let i = 0;
  791. for (i; i < path.length; i++) {
  792. if (path[i] != keyPath[i]) {
  793. return set;
  794. }
  795. }
  796. if (i < keyPath.length) {
  797. set.add(keyPath[i]);
  798. }
  799. return set;
  800. }, new Set());
  801. if (keySet.size > 0) {
  802. includeRestOptions.keys = Array.from(keySet).join(',');
  803. }
  804. }
  805. if (restOptions.excludeKeys) {
  806. const excludeKeys = new Set(restOptions.excludeKeys.split(','));
  807. const excludeKeySet = Array.from(excludeKeys).reduce((set, key) => {
  808. const keyPath = key.split('.');
  809. let i = 0;
  810. for (i; i < path.length; i++) {
  811. if (path[i] != keyPath[i]) {
  812. return set;
  813. }
  814. }
  815. if (i == keyPath.length - 1) {
  816. set.add(keyPath[i]);
  817. }
  818. return set;
  819. }, new Set());
  820. if (excludeKeySet.size > 0) {
  821. includeRestOptions.excludeKeys = Array.from(excludeKeySet).join(',');
  822. }
  823. }
  824. if (restOptions.includeReadPreference) {
  825. includeRestOptions.readPreference = restOptions.includeReadPreference;
  826. includeRestOptions.includeReadPreference = restOptions.includeReadPreference;
  827. } else if (restOptions.readPreference) {
  828. includeRestOptions.readPreference = restOptions.readPreference;
  829. }
  830. const queryPromises = Object.keys(pointersHash).map(async className => {
  831. const objectIds = Array.from(pointersHash[className]);
  832. let where;
  833. if (objectIds.length === 1) {
  834. where = {
  835. objectId: objectIds[0]
  836. };
  837. } else {
  838. where = {
  839. objectId: {
  840. $in: objectIds
  841. }
  842. };
  843. }
  844. const query = await RestQuery({
  845. method: objectIds.length === 1 ? RestQuery.Method.get : RestQuery.Method.find,
  846. config,
  847. auth,
  848. className,
  849. restWhere: where,
  850. restOptions: includeRestOptions,
  851. context: context
  852. });
  853. return query.execute({
  854. op: 'get'
  855. }).then(results => {
  856. results.className = className;
  857. return Promise.resolve(results);
  858. });
  859. });
  860. // Get the objects for all these object ids
  861. return Promise.all(queryPromises).then(responses => {
  862. var replace = responses.reduce((replace, includeResponse) => {
  863. for (var obj of includeResponse.results) {
  864. obj.__type = 'Object';
  865. obj.className = includeResponse.className;
  866. if (obj.className == '_User' && !auth.isMaster) {
  867. delete obj.sessionToken;
  868. delete obj.authData;
  869. }
  870. replace[obj.objectId] = obj;
  871. }
  872. return replace;
  873. }, {});
  874. var resp = {
  875. results: replacePointers(response.results, path, replace)
  876. };
  877. if (response.count) {
  878. resp.count = response.count;
  879. }
  880. return resp;
  881. });
  882. }
  883. // Object may be a list of REST-format object to find pointers in, or
  884. // it may be a single object.
  885. // If the path yields things that aren't pointers, this throws an error.
  886. // Path is a list of fields to search into.
  887. // Returns a list of pointers in REST format.
  888. function findPointers(object, path) {
  889. if (object instanceof Array) {
  890. return object.map(x => findPointers(x, path)).flat();
  891. }
  892. if (typeof object !== 'object' || !object) {
  893. return [];
  894. }
  895. if (path.length == 0) {
  896. if (object === null || object.__type == 'Pointer') {
  897. return [object];
  898. }
  899. return [];
  900. }
  901. var subobject = object[path[0]];
  902. if (!subobject) {
  903. return [];
  904. }
  905. return findPointers(subobject, path.slice(1));
  906. }
  907. // Object may be a list of REST-format objects to replace pointers
  908. // in, or it may be a single object.
  909. // Path is a list of fields to search into.
  910. // replace is a map from object id -> object.
  911. // Returns something analogous to object, but with the appropriate
  912. // pointers inflated.
  913. function replacePointers(object, path, replace) {
  914. if (object instanceof Array) {
  915. return object.map(obj => replacePointers(obj, path, replace)).filter(obj => typeof obj !== 'undefined');
  916. }
  917. if (typeof object !== 'object' || !object) {
  918. return object;
  919. }
  920. if (path.length === 0) {
  921. if (object && object.__type === 'Pointer') {
  922. return replace[object.objectId];
  923. }
  924. return object;
  925. }
  926. var subobject = object[path[0]];
  927. if (!subobject) {
  928. return object;
  929. }
  930. var newsub = replacePointers(subobject, path.slice(1), replace);
  931. var answer = {};
  932. for (var key in object) {
  933. if (key == path[0]) {
  934. answer[key] = newsub;
  935. } else {
  936. answer[key] = object[key];
  937. }
  938. }
  939. return answer;
  940. }
  941. // Finds a subobject that has the given key, if there is one.
  942. // Returns undefined otherwise.
  943. function findObjectWithKey(root, key) {
  944. if (typeof root !== 'object') {
  945. return;
  946. }
  947. if (root instanceof Array) {
  948. for (var item of root) {
  949. const answer = findObjectWithKey(item, key);
  950. if (answer) {
  951. return answer;
  952. }
  953. }
  954. }
  955. if (root && root[key]) {
  956. return root;
  957. }
  958. for (var subkey in root) {
  959. const answer = findObjectWithKey(root[subkey], key);
  960. if (answer) {
  961. return answer;
  962. }
  963. }
  964. }
  965. module.exports = RestQuery;
  966. // For tests
  967. module.exports._UnsafeRestQuery = _UnsafeRestQuery;
  968. //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["SchemaController","require","Parse","triggers","continueWhile","AlwaysSelectedKeys","enforceRoleSecurity","RestQuery","method","config","auth","className","restWhere","restOptions","clientSDK","runAfterFind","runBeforeFind","context","Method","find","get","includes","Error","INVALID_QUERY","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","pathResponse","includePath","newResponse","hasAfterFindHook","triggerExists","afterFind","applicationId","pipeline","distinct","parseQuery","Query","withJSON","maybeRunAfterFindTrigger","object","toJSON","all","authDataManager","pointers","findPointers","pointersHash","pointer","add","includeRestOptions","keySet","set","keyPath","i","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');\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  enforceRoleSecurity(method, className, auth);\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      method === RestQuery.Method.get\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  );\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) {\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  if (!this.auth.isMaster) {\n    if (this.className == '_Session') {\n      if (!this.auth.user) {\n        throw new Parse.Error(Parse.Error.INVALID_SESSION_TOKEN, 'Invalid session token');\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 new Parse.Error(\n            Parse.Error.OPERATION_FORBIDDEN,\n            'This user is not allowed to access ' + 'non-existent class: ' + this.className\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 new Parse.Error(\n        Parse.Error.OPERATION_FORBIDDEN,\n        `This user is not allowed to query ${key} on class ${this.className}`\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 = function () {\n  if (this.include.length == 0) {\n    return;\n  }\n\n  var pathResponse = includePath(\n    this.config,\n    this.auth,\n    this.response,\n    this.include[0],\n    this.context,\n    this.restOptions\n  );\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\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    )\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\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\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":"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","ignoreList":[]}