rest.js 26 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192
  1. "use strict";
  2. // This file contains helpers for running operations in REST format.
  3. // The goal is that handlers that explicitly handle an express route
  4. // should just be shallow wrappers around things in this file, but
  5. // these functions should not explicitly depend on the request
  6. // object.
  7. // This means that one of these handlers can support multiple
  8. // routes. That's useful for the routes that do really similar
  9. // things.
  10. var Parse = require('parse/node').Parse;
  11. var RestQuery = require('./RestQuery');
  12. var RestWrite = require('./RestWrite');
  13. var triggers = require('./triggers');
  14. const {
  15. enforceRoleSecurity
  16. } = require('./SharedRest');
  17. function checkTriggers(className, config, types) {
  18. return types.some(triggerType => {
  19. return triggers.getTrigger(className, triggers.Types[triggerType], config.applicationId);
  20. });
  21. }
  22. function checkLiveQuery(className, config) {
  23. return config.liveQueryController && config.liveQueryController.hasLiveQuery(className);
  24. }
  25. // Returns a promise for an object with optional keys 'results' and 'count'.
  26. const find = async (config, auth, className, restWhere, restOptions, clientSDK, context) => {
  27. const query = await RestQuery({
  28. method: RestQuery.Method.find,
  29. config,
  30. auth,
  31. className,
  32. restWhere,
  33. restOptions,
  34. clientSDK,
  35. context
  36. });
  37. return query.execute();
  38. };
  39. // get is just like find but only queries an objectId.
  40. const get = async (config, auth, className, objectId, restOptions, clientSDK, context) => {
  41. var restWhere = {
  42. objectId
  43. };
  44. const query = await RestQuery({
  45. method: RestQuery.Method.get,
  46. config,
  47. auth,
  48. className,
  49. restWhere,
  50. restOptions,
  51. clientSDK,
  52. context
  53. });
  54. return query.execute();
  55. };
  56. // Returns a promise that doesn't resolve to any useful value.
  57. function del(config, auth, className, objectId, context) {
  58. if (typeof objectId !== 'string') {
  59. throw new Parse.Error(Parse.Error.INVALID_JSON, 'bad objectId');
  60. }
  61. if (className === '_User' && auth.isUnauthenticated()) {
  62. throw new Parse.Error(Parse.Error.SESSION_MISSING, 'Insufficient auth to delete user');
  63. }
  64. enforceRoleSecurity('delete', className, auth);
  65. let inflatedObject;
  66. let schemaController;
  67. return Promise.resolve().then(async () => {
  68. const hasTriggers = checkTriggers(className, config, ['beforeDelete', 'afterDelete']);
  69. const hasLiveQuery = checkLiveQuery(className, config);
  70. if (hasTriggers || hasLiveQuery || className == '_Session') {
  71. const query = await RestQuery({
  72. method: RestQuery.Method.get,
  73. config,
  74. auth,
  75. className,
  76. restWhere: {
  77. objectId
  78. }
  79. });
  80. return query.execute({
  81. op: 'delete'
  82. }).then(response => {
  83. if (response && response.results && response.results.length) {
  84. const firstResult = response.results[0];
  85. firstResult.className = className;
  86. if (className === '_Session' && !auth.isMaster && !auth.isMaintenance) {
  87. if (!auth.user || firstResult.user.objectId !== auth.user.id) {
  88. throw new Parse.Error(Parse.Error.INVALID_SESSION_TOKEN, 'Invalid session token');
  89. }
  90. }
  91. var cacheAdapter = config.cacheController;
  92. cacheAdapter.user.del(firstResult.sessionToken);
  93. inflatedObject = Parse.Object.fromJSON(firstResult);
  94. return triggers.maybeRunTrigger(triggers.Types.beforeDelete, auth, inflatedObject, null, config, context);
  95. }
  96. throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Object not found for delete.');
  97. });
  98. }
  99. return Promise.resolve({});
  100. }).then(() => {
  101. if (!auth.isMaster && !auth.isMaintenance) {
  102. return auth.getUserRoles();
  103. } else {
  104. return;
  105. }
  106. }).then(() => config.database.loadSchema()).then(s => {
  107. schemaController = s;
  108. const options = {};
  109. if (!auth.isMaster && !auth.isMaintenance) {
  110. options.acl = ['*'];
  111. if (auth.user) {
  112. options.acl.push(auth.user.id);
  113. options.acl = options.acl.concat(auth.userRoles);
  114. }
  115. }
  116. return config.database.destroy(className, {
  117. objectId: objectId
  118. }, options, schemaController);
  119. }).then(() => {
  120. // Notify LiveQuery server if possible
  121. const perms = schemaController.getClassLevelPermissions(className);
  122. config.liveQueryController.onAfterDelete(className, inflatedObject, null, perms);
  123. return triggers.maybeRunTrigger(triggers.Types.afterDelete, auth, inflatedObject, null, config, context);
  124. }).catch(error => {
  125. handleSessionMissingError(error, className, auth);
  126. });
  127. }
  128. // Returns a promise for a {response, status, location} object.
  129. function create(config, auth, className, restObject, clientSDK, context) {
  130. enforceRoleSecurity('create', className, auth);
  131. var write = new RestWrite(config, auth, className, null, restObject, null, clientSDK, context);
  132. return write.execute();
  133. }
  134. // Returns a promise that contains the fields of the update that the
  135. // REST API is supposed to return.
  136. // Usually, this is just updatedAt.
  137. function update(config, auth, className, restWhere, restObject, clientSDK, context) {
  138. enforceRoleSecurity('update', className, auth);
  139. return Promise.resolve().then(async () => {
  140. const hasTriggers = checkTriggers(className, config, ['beforeSave', 'afterSave']);
  141. const hasLiveQuery = checkLiveQuery(className, config);
  142. if (hasTriggers || hasLiveQuery) {
  143. // Do not use find, as it runs the before finds
  144. const query = await RestQuery({
  145. method: RestQuery.Method.get,
  146. config,
  147. auth,
  148. className,
  149. restWhere,
  150. runAfterFind: false,
  151. runBeforeFind: false,
  152. context
  153. });
  154. return query.execute({
  155. op: 'update'
  156. });
  157. }
  158. return Promise.resolve({});
  159. }).then(({
  160. results
  161. }) => {
  162. var originalRestObject;
  163. if (results && results.length) {
  164. originalRestObject = results[0];
  165. }
  166. return new RestWrite(config, auth, className, restWhere, restObject, originalRestObject, clientSDK, context, 'update').execute();
  167. }).catch(error => {
  168. handleSessionMissingError(error, className, auth);
  169. });
  170. }
  171. function handleSessionMissingError(error, className, auth) {
  172. // If we're trying to update a user without / with bad session token
  173. if (className === '_User' && error.code === Parse.Error.OBJECT_NOT_FOUND && !auth.isMaster && !auth.isMaintenance) {
  174. throw new Parse.Error(Parse.Error.SESSION_MISSING, 'Insufficient auth.');
  175. }
  176. throw error;
  177. }
  178. module.exports = {
  179. create,
  180. del,
  181. find,
  182. get,
  183. update
  184. };
  185. //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["Parse","require","RestQuery","RestWrite","triggers","enforceRoleSecurity","checkTriggers","className","config","types","some","triggerType","getTrigger","Types","applicationId","checkLiveQuery","liveQueryController","hasLiveQuery","find","auth","restWhere","restOptions","clientSDK","context","query","method","Method","execute","get","objectId","del","Error","INVALID_JSON","isUnauthenticated","SESSION_MISSING","inflatedObject","schemaController","Promise","resolve","then","hasTriggers","op","response","results","length","firstResult","isMaster","isMaintenance","user","id","INVALID_SESSION_TOKEN","cacheAdapter","cacheController","sessionToken","Object","fromJSON","maybeRunTrigger","beforeDelete","OBJECT_NOT_FOUND","getUserRoles","database","loadSchema","s","options","acl","push","concat","userRoles","destroy","perms","getClassLevelPermissions","onAfterDelete","afterDelete","catch","error","handleSessionMissingError","create","restObject","write","update","runAfterFind","runBeforeFind","originalRestObject","code","module","exports"],"sources":["../src/rest.js"],"sourcesContent":["// This file contains helpers for running operations in REST format.\n// The goal is that handlers that explicitly handle an express route\n// should just be shallow wrappers around things in this file, but\n// these functions should not explicitly depend on the request\n// object.\n// This means that one of these handlers can support multiple\n// routes. That's useful for the routes that do really similar\n// things.\n\nvar Parse = require('parse/node').Parse;\n\nvar RestQuery = require('./RestQuery');\nvar RestWrite = require('./RestWrite');\nvar triggers = require('./triggers');\nconst { enforceRoleSecurity } = require('./SharedRest');\n\nfunction checkTriggers(className, config, types) {\n  return types.some(triggerType => {\n    return triggers.getTrigger(className, triggers.Types[triggerType], config.applicationId);\n  });\n}\n\nfunction checkLiveQuery(className, config) {\n  return config.liveQueryController && config.liveQueryController.hasLiveQuery(className);\n}\n\n// Returns a promise for an object with optional keys 'results' and 'count'.\nconst find = async (config, auth, className, restWhere, restOptions, clientSDK, context) => {\n  const query = await RestQuery({\n    method: RestQuery.Method.find,\n    config,\n    auth,\n    className,\n    restWhere,\n    restOptions,\n    clientSDK,\n    context,\n  });\n  return query.execute();\n};\n\n// get is just like find but only queries an objectId.\nconst get = async (config, auth, className, objectId, restOptions, clientSDK, context) => {\n  var restWhere = { objectId };\n  const query = await RestQuery({\n    method: RestQuery.Method.get,\n    config,\n    auth,\n    className,\n    restWhere,\n    restOptions,\n    clientSDK,\n    context,\n  });\n  return query.execute();\n};\n\n// Returns a promise that doesn't resolve to any useful value.\nfunction del(config, auth, className, objectId, context) {\n  if (typeof objectId !== 'string') {\n    throw new Parse.Error(Parse.Error.INVALID_JSON, 'bad objectId');\n  }\n\n  if (className === '_User' && auth.isUnauthenticated()) {\n    throw new Parse.Error(Parse.Error.SESSION_MISSING, 'Insufficient auth to delete user');\n  }\n\n  enforceRoleSecurity('delete', className, auth);\n\n  let inflatedObject;\n  let schemaController;\n\n  return Promise.resolve()\n    .then(async () => {\n      const hasTriggers = checkTriggers(className, config, ['beforeDelete', 'afterDelete']);\n      const hasLiveQuery = checkLiveQuery(className, config);\n      if (hasTriggers || hasLiveQuery || className == '_Session') {\n        const query = await RestQuery({\n          method: RestQuery.Method.get,\n          config,\n          auth,\n          className,\n          restWhere: { objectId },\n        });\n        return query.execute({ op: 'delete' }).then(response => {\n          if (response && response.results && response.results.length) {\n            const firstResult = response.results[0];\n            firstResult.className = className;\n            if (className === '_Session' && !auth.isMaster && !auth.isMaintenance) {\n              if (!auth.user || firstResult.user.objectId !== auth.user.id) {\n                throw new Parse.Error(Parse.Error.INVALID_SESSION_TOKEN, 'Invalid session token');\n              }\n            }\n            var cacheAdapter = config.cacheController;\n            cacheAdapter.user.del(firstResult.sessionToken);\n            inflatedObject = Parse.Object.fromJSON(firstResult);\n            return triggers.maybeRunTrigger(\n              triggers.Types.beforeDelete,\n              auth,\n              inflatedObject,\n              null,\n              config,\n              context\n            );\n          }\n          throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Object not found for delete.');\n        });\n      }\n      return Promise.resolve({});\n    })\n    .then(() => {\n      if (!auth.isMaster && !auth.isMaintenance) {\n        return auth.getUserRoles();\n      } else {\n        return;\n      }\n    })\n    .then(() => config.database.loadSchema())\n    .then(s => {\n      schemaController = s;\n      const options = {};\n      if (!auth.isMaster && !auth.isMaintenance) {\n        options.acl = ['*'];\n        if (auth.user) {\n          options.acl.push(auth.user.id);\n          options.acl = options.acl.concat(auth.userRoles);\n        }\n      }\n\n      return config.database.destroy(\n        className,\n        {\n          objectId: objectId,\n        },\n        options,\n        schemaController\n      );\n    })\n    .then(() => {\n      // Notify LiveQuery server if possible\n      const perms = schemaController.getClassLevelPermissions(className);\n      config.liveQueryController.onAfterDelete(className, inflatedObject, null, perms);\n      return triggers.maybeRunTrigger(\n        triggers.Types.afterDelete,\n        auth,\n        inflatedObject,\n        null,\n        config,\n        context\n      );\n    })\n    .catch(error => {\n      handleSessionMissingError(error, className, auth);\n    });\n}\n\n// Returns a promise for a {response, status, location} object.\nfunction create(config, auth, className, restObject, clientSDK, context) {\n  enforceRoleSecurity('create', className, auth);\n  var write = new RestWrite(config, auth, className, null, restObject, null, clientSDK, context);\n  return write.execute();\n}\n\n// Returns a promise that contains the fields of the update that the\n// REST API is supposed to return.\n// Usually, this is just updatedAt.\nfunction update(config, auth, className, restWhere, restObject, clientSDK, context) {\n  enforceRoleSecurity('update', className, auth);\n\n  return Promise.resolve()\n    .then(async () => {\n      const hasTriggers = checkTriggers(className, config, ['beforeSave', 'afterSave']);\n      const hasLiveQuery = checkLiveQuery(className, config);\n      if (hasTriggers || hasLiveQuery) {\n        // Do not use find, as it runs the before finds\n        const query = await RestQuery({\n          method: RestQuery.Method.get,\n          config,\n          auth,\n          className,\n          restWhere,\n          runAfterFind: false,\n          runBeforeFind: false,\n          context,\n        });\n        return query.execute({\n          op: 'update',\n        });\n      }\n      return Promise.resolve({});\n    })\n    .then(({ results }) => {\n      var originalRestObject;\n      if (results && results.length) {\n        originalRestObject = results[0];\n      }\n      return new RestWrite(\n        config,\n        auth,\n        className,\n        restWhere,\n        restObject,\n        originalRestObject,\n        clientSDK,\n        context,\n        'update'\n      ).execute();\n    })\n    .catch(error => {\n      handleSessionMissingError(error, className, auth);\n    });\n}\n\nfunction handleSessionMissingError(error, className, auth) {\n  // If we're trying to update a user without / with bad session token\n  if (\n    className === '_User' &&\n    error.code === Parse.Error.OBJECT_NOT_FOUND &&\n    !auth.isMaster &&\n    !auth.isMaintenance\n  ) {\n    throw new Parse.Error(Parse.Error.SESSION_MISSING, 'Insufficient auth.');\n  }\n  throw error;\n}\n\nmodule.exports = {\n  create,\n  del,\n  find,\n  get,\n  update,\n};\n"],"mappings":"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","ignoreList":[]}