Parse.Cloud.js 92 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701
  1. "use strict";
  2. var _node = require("parse/node");
  3. var triggers = _interopRequireWildcard(require("../triggers"));
  4. var _middlewares = require("../middlewares");
  5. function _getRequireWildcardCache(e) { if ("function" != typeof WeakMap) return null; var r = new WeakMap(), t = new WeakMap(); return (_getRequireWildcardCache = function (e) { return e ? t : r; })(e); }
  6. function _interopRequireWildcard(e, r) { if (!r && e && e.__esModule) return e; if (null === e || "object" != typeof e && "function" != typeof e) return { default: e }; var t = _getRequireWildcardCache(r); if (t && t.has(e)) return t.get(e); var n = { __proto__: null }, a = Object.defineProperty && Object.getOwnPropertyDescriptor; for (var u in e) if ("default" !== u && {}.hasOwnProperty.call(e, u)) { var i = a ? Object.getOwnPropertyDescriptor(e, u) : null; i && (i.get || i.set) ? Object.defineProperty(n, u, i) : n[u] = e[u]; } return n.default = e, t && t.set(e, n), n; }
  7. function ownKeys(e, r) { var t = Object.keys(e); if (Object.getOwnPropertySymbols) { var o = Object.getOwnPropertySymbols(e); r && (o = o.filter(function (r) { return Object.getOwnPropertyDescriptor(e, r).enumerable; })), t.push.apply(t, o); } return t; }
  8. function _objectSpread(e) { for (var r = 1; r < arguments.length; r++) { var t = null != arguments[r] ? arguments[r] : {}; r % 2 ? ownKeys(Object(t), !0).forEach(function (r) { _defineProperty(e, r, t[r]); }) : Object.getOwnPropertyDescriptors ? Object.defineProperties(e, Object.getOwnPropertyDescriptors(t)) : ownKeys(Object(t)).forEach(function (r) { Object.defineProperty(e, r, Object.getOwnPropertyDescriptor(t, r)); }); } return e; }
  9. function _defineProperty(e, r, t) { return (r = _toPropertyKey(r)) in e ? Object.defineProperty(e, r, { value: t, enumerable: !0, configurable: !0, writable: !0 }) : e[r] = t, e; }
  10. function _toPropertyKey(t) { var i = _toPrimitive(t, "string"); return "symbol" == typeof i ? i : i + ""; }
  11. function _toPrimitive(t, r) { if ("object" != typeof t || !t) return t; var e = t[Symbol.toPrimitive]; if (void 0 !== e) { var i = e.call(t, r || "default"); if ("object" != typeof i) return i; throw new TypeError("@@toPrimitive must return a primitive value."); } return ("string" === r ? String : Number)(t); }
  12. const Config = require('../Config');
  13. function isParseObjectConstructor(object) {
  14. return typeof object === 'function' && Object.prototype.hasOwnProperty.call(object, 'className');
  15. }
  16. function validateValidator(validator) {
  17. if (!validator || typeof validator === 'function') {
  18. return;
  19. }
  20. const fieldOptions = {
  21. type: ['Any'],
  22. constant: [Boolean],
  23. default: ['Any'],
  24. options: [Array, 'function', 'Any'],
  25. required: [Boolean],
  26. error: [String]
  27. };
  28. const allowedKeys = {
  29. requireUser: [Boolean],
  30. requireAnyUserRoles: [Array, 'function'],
  31. requireAllUserRoles: [Array, 'function'],
  32. requireMaster: [Boolean],
  33. validateMasterKey: [Boolean],
  34. skipWithMasterKey: [Boolean],
  35. requireUserKeys: [Array, Object],
  36. fields: [Array, Object],
  37. rateLimit: [Object]
  38. };
  39. const getType = fn => {
  40. if (Array.isArray(fn)) {
  41. return 'array';
  42. }
  43. if (fn === 'Any' || fn === 'function') {
  44. return fn;
  45. }
  46. const type = typeof fn;
  47. if (typeof fn === 'function') {
  48. const match = fn && fn.toString().match(/^\s*function (\w+)/);
  49. return (match ? match[1] : 'function').toLowerCase();
  50. }
  51. return type;
  52. };
  53. const checkKey = (key, data, validatorParam) => {
  54. const parameter = data[key];
  55. if (!parameter) {
  56. throw `${key} is not a supported parameter for Cloud Function validations.`;
  57. }
  58. const types = parameter.map(type => getType(type));
  59. const type = getType(validatorParam);
  60. if (!types.includes(type) && !types.includes('Any')) {
  61. throw `Invalid type for Cloud Function validation key ${key}. Expected ${types.join('|')}, actual ${type}`;
  62. }
  63. };
  64. for (const key in validator) {
  65. checkKey(key, allowedKeys, validator[key]);
  66. if (key === 'fields' || key === 'requireUserKeys') {
  67. const values = validator[key];
  68. if (Array.isArray(values)) {
  69. continue;
  70. }
  71. for (const value in values) {
  72. const data = values[value];
  73. for (const subKey in data) {
  74. checkKey(subKey, fieldOptions, data[subKey]);
  75. }
  76. }
  77. }
  78. }
  79. }
  80. const getRoute = parseClass => {
  81. const route = {
  82. _User: 'users',
  83. _Session: 'sessions',
  84. '@File': 'files',
  85. '@Config': 'config'
  86. }[parseClass] || 'classes';
  87. if (parseClass === '@File') {
  88. return `/${route}/:id?(.*)`;
  89. }
  90. if (parseClass === '@Config') {
  91. return `/${route}`;
  92. }
  93. return `/${route}/${parseClass}/:id?(.*)`;
  94. };
  95. /** @namespace
  96. * @name Parse
  97. * @description The Parse SDK.
  98. * see [api docs](https://docs.parseplatform.org/js/api) and [guide](https://docs.parseplatform.org/js/guide)
  99. */
  100. /** @namespace
  101. * @name Parse.Cloud
  102. * @memberof Parse
  103. * @description The Parse Cloud Code SDK.
  104. */
  105. var ParseCloud = {};
  106. /**
  107. * Defines a Cloud Function.
  108. *
  109. * **Available in Cloud Code only.**
  110. *
  111. * ```
  112. * Parse.Cloud.define('functionName', (request) => {
  113. * // code here
  114. * }, (request) => {
  115. * // validation code here
  116. * });
  117. *
  118. * Parse.Cloud.define('functionName', (request) => {
  119. * // code here
  120. * }, { ...validationObject });
  121. * ```
  122. *
  123. * @static
  124. * @memberof Parse.Cloud
  125. * @param {String} name The name of the Cloud Function
  126. * @param {Function} data The Cloud Function to register. This function can be an async function and should take one parameter a {@link Parse.Cloud.FunctionRequest}.
  127. * @param {(Object|Function)} validator An optional function to help validating cloud code. This function can be an async function and should take one parameter a {@link Parse.Cloud.FunctionRequest}, or a {@link Parse.Cloud.ValidatorObject}.
  128. */
  129. ParseCloud.define = function (functionName, handler, validationHandler) {
  130. validateValidator(validationHandler);
  131. triggers.addFunction(functionName, handler, validationHandler, _node.Parse.applicationId);
  132. if (validationHandler && validationHandler.rateLimit) {
  133. (0, _middlewares.addRateLimit)(_objectSpread({
  134. requestPath: `/functions/${functionName}`
  135. }, validationHandler.rateLimit), _node.Parse.applicationId, true);
  136. }
  137. };
  138. /**
  139. * Defines a Background Job.
  140. *
  141. * **Available in Cloud Code only.**
  142. *
  143. * @method job
  144. * @name Parse.Cloud.job
  145. * @param {String} name The name of the Background Job
  146. * @param {Function} func The Background Job to register. This function can be async should take a single parameters a {@link Parse.Cloud.JobRequest}
  147. *
  148. */
  149. ParseCloud.job = function (functionName, handler) {
  150. triggers.addJob(functionName, handler, _node.Parse.applicationId);
  151. };
  152. /**
  153. *
  154. * Registers a before save function.
  155. *
  156. * **Available in Cloud Code only.**
  157. *
  158. * If you want to use beforeSave for a predefined class in the Parse JavaScript SDK (e.g. {@link Parse.User} or {@link Parse.File}), you should pass the class itself and not the String for arg1.
  159. *
  160. * ```
  161. * Parse.Cloud.beforeSave('MyCustomClass', (request) => {
  162. * // code here
  163. * }, (request) => {
  164. * // validation code here
  165. * });
  166. *
  167. * Parse.Cloud.beforeSave(Parse.User, (request) => {
  168. * // code here
  169. * }, { ...validationObject })
  170. * ```
  171. *
  172. * @method beforeSave
  173. * @name Parse.Cloud.beforeSave
  174. * @param {(String|Parse.Object)} arg1 The Parse.Object subclass to register the after save function for. This can instead be a String that is the className of the subclass.
  175. * @param {Function} func The function to run before a save. This function can be async and should take one parameter a {@link Parse.Cloud.TriggerRequest};
  176. * @param {(Object|Function)} validator An optional function to help validating cloud code. This function can be an async function and should take one parameter a {@link Parse.Cloud.TriggerRequest}, or a {@link Parse.Cloud.ValidatorObject}.
  177. */
  178. ParseCloud.beforeSave = function (parseClass, handler, validationHandler) {
  179. const className = triggers.getClassName(parseClass);
  180. validateValidator(validationHandler);
  181. triggers.addTrigger(triggers.Types.beforeSave, className, handler, _node.Parse.applicationId, validationHandler);
  182. if (validationHandler && validationHandler.rateLimit) {
  183. (0, _middlewares.addRateLimit)(_objectSpread({
  184. requestPath: getRoute(className),
  185. requestMethods: ['POST', 'PUT']
  186. }, validationHandler.rateLimit), _node.Parse.applicationId, true);
  187. }
  188. };
  189. /**
  190. * Registers a before delete function.
  191. *
  192. * **Available in Cloud Code only.**
  193. *
  194. * If you want to use beforeDelete for a predefined class in the Parse JavaScript SDK (e.g. {@link Parse.User} or {@link Parse.File}), you should pass the class itself and not the String for arg1.
  195. * ```
  196. * Parse.Cloud.beforeDelete('MyCustomClass', (request) => {
  197. * // code here
  198. * }, (request) => {
  199. * // validation code here
  200. * });
  201. *
  202. * Parse.Cloud.beforeDelete(Parse.User, (request) => {
  203. * // code here
  204. * }, { ...validationObject })
  205. *```
  206. *
  207. * @method beforeDelete
  208. * @name Parse.Cloud.beforeDelete
  209. * @param {(String|Parse.Object)} arg1 The Parse.Object subclass to register the before delete function for. This can instead be a String that is the className of the subclass.
  210. * @param {Function} func The function to run before a delete. This function can be async and should take one parameter, a {@link Parse.Cloud.TriggerRequest}.
  211. * @param {(Object|Function)} validator An optional function to help validating cloud code. This function can be an async function and should take one parameter a {@link Parse.Cloud.TriggerRequest}, or a {@link Parse.Cloud.ValidatorObject}.
  212. */
  213. ParseCloud.beforeDelete = function (parseClass, handler, validationHandler) {
  214. const className = triggers.getClassName(parseClass);
  215. validateValidator(validationHandler);
  216. triggers.addTrigger(triggers.Types.beforeDelete, className, handler, _node.Parse.applicationId, validationHandler);
  217. if (validationHandler && validationHandler.rateLimit) {
  218. (0, _middlewares.addRateLimit)(_objectSpread({
  219. requestPath: getRoute(className),
  220. requestMethods: 'DELETE'
  221. }, validationHandler.rateLimit), _node.Parse.applicationId, true);
  222. }
  223. };
  224. /**
  225. *
  226. * Registers the before login function.
  227. *
  228. * **Available in Cloud Code only.**
  229. *
  230. * This function provides further control
  231. * in validating a login attempt. Specifically,
  232. * it is triggered after a user enters
  233. * correct credentials (or other valid authData),
  234. * but prior to a session being generated.
  235. *
  236. * ```
  237. * Parse.Cloud.beforeLogin((request) => {
  238. * // code here
  239. * })
  240. *
  241. * ```
  242. *
  243. * @method beforeLogin
  244. * @name Parse.Cloud.beforeLogin
  245. * @param {Function} func The function to run before a login. This function can be async and should take one parameter a {@link Parse.Cloud.TriggerRequest};
  246. */
  247. ParseCloud.beforeLogin = function (handler, validationHandler) {
  248. let className = '_User';
  249. if (typeof handler === 'string' || isParseObjectConstructor(handler)) {
  250. // validation will occur downstream, this is to maintain internal
  251. // code consistency with the other hook types.
  252. className = triggers.getClassName(handler);
  253. handler = arguments[1];
  254. validationHandler = arguments.length >= 2 ? arguments[2] : null;
  255. }
  256. triggers.addTrigger(triggers.Types.beforeLogin, className, handler, _node.Parse.applicationId);
  257. if (validationHandler && validationHandler.rateLimit) {
  258. (0, _middlewares.addRateLimit)(_objectSpread({
  259. requestPath: `/login`,
  260. requestMethods: 'POST'
  261. }, validationHandler.rateLimit), _node.Parse.applicationId, true);
  262. }
  263. };
  264. /**
  265. *
  266. * Registers the after login function.
  267. *
  268. * **Available in Cloud Code only.**
  269. *
  270. * This function is triggered after a user logs in successfully,
  271. * and after a _Session object has been created.
  272. *
  273. * ```
  274. * Parse.Cloud.afterLogin((request) => {
  275. * // code here
  276. * });
  277. * ```
  278. *
  279. * @method afterLogin
  280. * @name Parse.Cloud.afterLogin
  281. * @param {Function} func The function to run after a login. This function can be async and should take one parameter a {@link Parse.Cloud.TriggerRequest};
  282. */
  283. ParseCloud.afterLogin = function (handler) {
  284. let className = '_User';
  285. if (typeof handler === 'string' || isParseObjectConstructor(handler)) {
  286. // validation will occur downstream, this is to maintain internal
  287. // code consistency with the other hook types.
  288. className = triggers.getClassName(handler);
  289. handler = arguments[1];
  290. }
  291. triggers.addTrigger(triggers.Types.afterLogin, className, handler, _node.Parse.applicationId);
  292. };
  293. /**
  294. *
  295. * Registers the after logout function.
  296. *
  297. * **Available in Cloud Code only.**
  298. *
  299. * This function is triggered after a user logs out.
  300. *
  301. * ```
  302. * Parse.Cloud.afterLogout((request) => {
  303. * // code here
  304. * });
  305. * ```
  306. *
  307. * @method afterLogout
  308. * @name Parse.Cloud.afterLogout
  309. * @param {Function} func The function to run after a logout. This function can be async and should take one parameter a {@link Parse.Cloud.TriggerRequest};
  310. */
  311. ParseCloud.afterLogout = function (handler) {
  312. let className = '_Session';
  313. if (typeof handler === 'string' || isParseObjectConstructor(handler)) {
  314. // validation will occur downstream, this is to maintain internal
  315. // code consistency with the other hook types.
  316. className = triggers.getClassName(handler);
  317. handler = arguments[1];
  318. }
  319. triggers.addTrigger(triggers.Types.afterLogout, className, handler, _node.Parse.applicationId);
  320. };
  321. /**
  322. * Registers an after save function.
  323. *
  324. * **Available in Cloud Code only.**
  325. *
  326. * If you want to use afterSave for a predefined class in the Parse JavaScript SDK (e.g. {@link Parse.User} or {@link Parse.File}), you should pass the class itself and not the String for arg1.
  327. *
  328. * ```
  329. * Parse.Cloud.afterSave('MyCustomClass', async function(request) {
  330. * // code here
  331. * }, (request) => {
  332. * // validation code here
  333. * });
  334. *
  335. * Parse.Cloud.afterSave(Parse.User, async function(request) {
  336. * // code here
  337. * }, { ...validationObject });
  338. * ```
  339. *
  340. * @method afterSave
  341. * @name Parse.Cloud.afterSave
  342. * @param {(String|Parse.Object)} arg1 The Parse.Object subclass to register the after save function for. This can instead be a String that is the className of the subclass.
  343. * @param {Function} func The function to run after a save. This function can be an async function and should take just one parameter, {@link Parse.Cloud.TriggerRequest}.
  344. * @param {(Object|Function)} validator An optional function to help validating cloud code. This function can be an async function and should take one parameter a {@link Parse.Cloud.TriggerRequest}, or a {@link Parse.Cloud.ValidatorObject}.
  345. */
  346. ParseCloud.afterSave = function (parseClass, handler, validationHandler) {
  347. const className = triggers.getClassName(parseClass);
  348. validateValidator(validationHandler);
  349. triggers.addTrigger(triggers.Types.afterSave, className, handler, _node.Parse.applicationId, validationHandler);
  350. };
  351. /**
  352. * Registers an after delete function.
  353. *
  354. * **Available in Cloud Code only.**
  355. *
  356. * If you want to use afterDelete for a predefined class in the Parse JavaScript SDK (e.g. {@link Parse.User} or {@link Parse.File}), you should pass the class itself and not the String for arg1.
  357. * ```
  358. * Parse.Cloud.afterDelete('MyCustomClass', async (request) => {
  359. * // code here
  360. * }, (request) => {
  361. * // validation code here
  362. * });
  363. *
  364. * Parse.Cloud.afterDelete(Parse.User, async (request) => {
  365. * // code here
  366. * }, { ...validationObject });
  367. *```
  368. *
  369. * @method afterDelete
  370. * @name Parse.Cloud.afterDelete
  371. * @param {(String|Parse.Object)} arg1 The Parse.Object subclass to register the after delete function for. This can instead be a String that is the className of the subclass.
  372. * @param {Function} func The function to run after a delete. This function can be async and should take just one parameter, {@link Parse.Cloud.TriggerRequest}.
  373. * @param {(Object|Function)} validator An optional function to help validating cloud code. This function can be an async function and should take one parameter a {@link Parse.Cloud.TriggerRequest}, or a {@link Parse.Cloud.ValidatorObject}.
  374. */
  375. ParseCloud.afterDelete = function (parseClass, handler, validationHandler) {
  376. const className = triggers.getClassName(parseClass);
  377. validateValidator(validationHandler);
  378. triggers.addTrigger(triggers.Types.afterDelete, className, handler, _node.Parse.applicationId, validationHandler);
  379. };
  380. /**
  381. * Registers a before find function.
  382. *
  383. * **Available in Cloud Code only.**
  384. *
  385. * If you want to use beforeFind for a predefined class in the Parse JavaScript SDK (e.g. {@link Parse.User} or {@link Parse.File}), you should pass the class itself and not the String for arg1.
  386. * ```
  387. * Parse.Cloud.beforeFind('MyCustomClass', async (request) => {
  388. * // code here
  389. * }, (request) => {
  390. * // validation code here
  391. * });
  392. *
  393. * Parse.Cloud.beforeFind(Parse.User, async (request) => {
  394. * // code here
  395. * }, { ...validationObject });
  396. *```
  397. *
  398. * @method beforeFind
  399. * @name Parse.Cloud.beforeFind
  400. * @param {(String|Parse.Object)} arg1 The Parse.Object subclass to register the before find function for. This can instead be a String that is the className of the subclass.
  401. * @param {Function} func The function to run before a find. This function can be async and should take just one parameter, {@link Parse.Cloud.BeforeFindRequest}.
  402. * @param {(Object|Function)} validator An optional function to help validating cloud code. This function can be an async function and should take one parameter a {@link Parse.Cloud.BeforeFindRequest}, or a {@link Parse.Cloud.ValidatorObject}.
  403. */
  404. ParseCloud.beforeFind = function (parseClass, handler, validationHandler) {
  405. const className = triggers.getClassName(parseClass);
  406. validateValidator(validationHandler);
  407. triggers.addTrigger(triggers.Types.beforeFind, className, handler, _node.Parse.applicationId, validationHandler);
  408. if (validationHandler && validationHandler.rateLimit) {
  409. (0, _middlewares.addRateLimit)(_objectSpread({
  410. requestPath: getRoute(className),
  411. requestMethods: 'GET'
  412. }, validationHandler.rateLimit), _node.Parse.applicationId, true);
  413. }
  414. };
  415. /**
  416. * Registers an after find function.
  417. *
  418. * **Available in Cloud Code only.**
  419. *
  420. * If you want to use afterFind for a predefined class in the Parse JavaScript SDK (e.g. {@link Parse.User} or {@link Parse.File}), you should pass the class itself and not the String for arg1.
  421. * ```
  422. * Parse.Cloud.afterFind('MyCustomClass', async (request) => {
  423. * // code here
  424. * }, (request) => {
  425. * // validation code here
  426. * });
  427. *
  428. * Parse.Cloud.afterFind(Parse.User, async (request) => {
  429. * // code here
  430. * }, { ...validationObject });
  431. *```
  432. *
  433. * @method afterFind
  434. * @name Parse.Cloud.afterFind
  435. * @param {(String|Parse.Object)} arg1 The Parse.Object subclass to register the after find function for. This can instead be a String that is the className of the subclass.
  436. * @param {Function} func The function to run before a find. This function can be async and should take just one parameter, {@link Parse.Cloud.AfterFindRequest}.
  437. * @param {(Object|Function)} validator An optional function to help validating cloud code. This function can be an async function and should take one parameter a {@link Parse.Cloud.AfterFindRequest}, or a {@link Parse.Cloud.ValidatorObject}.
  438. */
  439. ParseCloud.afterFind = function (parseClass, handler, validationHandler) {
  440. const className = triggers.getClassName(parseClass);
  441. validateValidator(validationHandler);
  442. triggers.addTrigger(triggers.Types.afterFind, className, handler, _node.Parse.applicationId, validationHandler);
  443. };
  444. /**
  445. * Registers a before live query server connect function.
  446. *
  447. * **Available in Cloud Code only.**
  448. *
  449. * ```
  450. * Parse.Cloud.beforeConnect(async (request) => {
  451. * // code here
  452. * }, (request) => {
  453. * // validation code here
  454. * });
  455. *
  456. * Parse.Cloud.beforeConnect(async (request) => {
  457. * // code here
  458. * }, { ...validationObject });
  459. *```
  460. *
  461. * @method beforeConnect
  462. * @name Parse.Cloud.beforeConnect
  463. * @param {Function} func The function to before connection is made. This function can be async and should take just one parameter, {@link Parse.Cloud.ConnectTriggerRequest}.
  464. * @param {(Object|Function)} validator An optional function to help validating cloud code. This function can be an async function and should take one parameter a {@link Parse.Cloud.ConnectTriggerRequest}, or a {@link Parse.Cloud.ValidatorObject}.
  465. */
  466. ParseCloud.beforeConnect = function (handler, validationHandler) {
  467. validateValidator(validationHandler);
  468. triggers.addConnectTrigger(triggers.Types.beforeConnect, handler, _node.Parse.applicationId, validationHandler);
  469. };
  470. /**
  471. * Sends an email through the Parse Server mail adapter.
  472. *
  473. * **Available in Cloud Code only.**
  474. * **Requires a mail adapter to be configured for Parse Server.**
  475. *
  476. * ```
  477. * Parse.Cloud.sendEmail({
  478. * from: 'Example <test@example.com>',
  479. * to: 'contact@example.com',
  480. * subject: 'Test email',
  481. * text: 'This email is a test.'
  482. * });
  483. *```
  484. *
  485. * @method sendEmail
  486. * @name Parse.Cloud.sendEmail
  487. * @param {Object} data The object of the mail data to send.
  488. */
  489. ParseCloud.sendEmail = function (data) {
  490. const config = Config.get(_node.Parse.applicationId);
  491. const emailAdapter = config.userController.adapter;
  492. if (!emailAdapter) {
  493. config.loggerController.error('Failed to send email because no mail adapter is configured for Parse Server.');
  494. return;
  495. }
  496. return emailAdapter.sendMail(data);
  497. };
  498. /**
  499. * Registers a before live query subscription function.
  500. *
  501. * **Available in Cloud Code only.**
  502. *
  503. * If you want to use beforeSubscribe for a predefined class in the Parse JavaScript SDK (e.g. {@link Parse.User} or {@link Parse.File}), you should pass the class itself and not the String for arg1.
  504. * ```
  505. * Parse.Cloud.beforeSubscribe('MyCustomClass', (request) => {
  506. * // code here
  507. * }, (request) => {
  508. * // validation code here
  509. * });
  510. *
  511. * Parse.Cloud.beforeSubscribe(Parse.User, (request) => {
  512. * // code here
  513. * }, { ...validationObject });
  514. *```
  515. *
  516. * @method beforeSubscribe
  517. * @name Parse.Cloud.beforeSubscribe
  518. * @param {(String|Parse.Object)} arg1 The Parse.Object subclass to register the before subscription function for. This can instead be a String that is the className of the subclass.
  519. * @param {Function} func The function to run before a subscription. This function can be async and should take one parameter, a {@link Parse.Cloud.TriggerRequest}.
  520. * @param {(Object|Function)} validator An optional function to help validating cloud code. This function can be an async function and should take one parameter a {@link Parse.Cloud.TriggerRequest}, or a {@link Parse.Cloud.ValidatorObject}.
  521. */
  522. ParseCloud.beforeSubscribe = function (parseClass, handler, validationHandler) {
  523. validateValidator(validationHandler);
  524. const className = triggers.getClassName(parseClass);
  525. triggers.addTrigger(triggers.Types.beforeSubscribe, className, handler, _node.Parse.applicationId, validationHandler);
  526. };
  527. ParseCloud.onLiveQueryEvent = function (handler) {
  528. triggers.addLiveQueryEventHandler(handler, _node.Parse.applicationId);
  529. };
  530. /**
  531. * Registers an after live query server event function.
  532. *
  533. * **Available in Cloud Code only.**
  534. *
  535. * ```
  536. * Parse.Cloud.afterLiveQueryEvent('MyCustomClass', (request) => {
  537. * // code here
  538. * }, (request) => {
  539. * // validation code here
  540. * });
  541. *
  542. * Parse.Cloud.afterLiveQueryEvent('MyCustomClass', (request) => {
  543. * // code here
  544. * }, { ...validationObject });
  545. *```
  546. *
  547. * @method afterLiveQueryEvent
  548. * @name Parse.Cloud.afterLiveQueryEvent
  549. * @param {(String|Parse.Object)} arg1 The Parse.Object subclass to register the after live query event function for. This can instead be a String that is the className of the subclass.
  550. * @param {Function} func The function to run after a live query event. This function can be async and should take one parameter, a {@link Parse.Cloud.LiveQueryEventTrigger}.
  551. * @param {(Object|Function)} validator An optional function to help validating cloud code. This function can be an async function and should take one parameter a {@link Parse.Cloud.LiveQueryEventTrigger}, or a {@link Parse.Cloud.ValidatorObject}.
  552. */
  553. ParseCloud.afterLiveQueryEvent = function (parseClass, handler, validationHandler) {
  554. const className = triggers.getClassName(parseClass);
  555. validateValidator(validationHandler);
  556. triggers.addTrigger(triggers.Types.afterEvent, className, handler, _node.Parse.applicationId, validationHandler);
  557. };
  558. ParseCloud._removeAllHooks = () => {
  559. triggers._unregisterAll();
  560. const config = Config.get(_node.Parse.applicationId);
  561. config === null || config === void 0 || config.unregisterRateLimiters();
  562. };
  563. ParseCloud.useMasterKey = () => {
  564. // eslint-disable-next-line
  565. console.warn('Parse.Cloud.useMasterKey is deprecated (and has no effect anymore) on parse-server, please refer to the cloud code migration notes: http://docs.parseplatform.org/parse-server/guide/#master-key-must-be-passed-explicitly');
  566. };
  567. module.exports = ParseCloud;
  568. /**
  569. * @interface Parse.Cloud.TriggerRequest
  570. * @property {String} installationId If set, the installationId triggering the request.
  571. * @property {Boolean} master If true, means the master key was used.
  572. * @property {Boolean} isChallenge If true, means the current request is originally triggered by an auth challenge.
  573. * @property {Parse.User} user If set, the user that made the request.
  574. * @property {Parse.Object} object The object triggering the hook.
  575. * @property {String} ip The IP address of the client making the request. To ensure retrieving the correct IP address, set the Parse Server option `trustProxy: true` if Parse Server runs behind a proxy server, for example behind a load balancer.
  576. * @property {Object} headers The original HTTP headers for the request.
  577. * @property {String} triggerName The name of the trigger (`beforeSave`, `afterSave`, ...)
  578. * @property {Object} log The current logger inside Parse Server.
  579. * @property {Parse.Object} original If set, the object, as currently stored.
  580. */
  581. /**
  582. * @interface Parse.Cloud.FileTriggerRequest
  583. * @property {String} installationId If set, the installationId triggering the request.
  584. * @property {Boolean} master If true, means the master key was used.
  585. * @property {Parse.User} user If set, the user that made the request.
  586. * @property {Parse.File} file The file that triggered the hook.
  587. * @property {Integer} fileSize The size of the file in bytes.
  588. * @property {Integer} contentLength The value from Content-Length header
  589. * @property {String} ip The IP address of the client making the request.
  590. * @property {Object} headers The original HTTP headers for the request.
  591. * @property {String} triggerName The name of the trigger (`beforeSave`, `afterSave`)
  592. * @property {Object} log The current logger inside Parse Server.
  593. */
  594. /**
  595. * @interface Parse.Cloud.ConnectTriggerRequest
  596. * @property {String} installationId If set, the installationId triggering the request.
  597. * @property {Boolean} useMasterKey If true, means the master key was used.
  598. * @property {Parse.User} user If set, the user that made the request.
  599. * @property {Integer} clients The number of clients connected.
  600. * @property {Integer} subscriptions The number of subscriptions connected.
  601. * @property {String} sessionToken If set, the session of the user that made the request.
  602. */
  603. /**
  604. * @interface Parse.Cloud.LiveQueryEventTrigger
  605. * @property {String} installationId If set, the installationId triggering the request.
  606. * @property {Boolean} useMasterKey If true, means the master key was used.
  607. * @property {Parse.User} user If set, the user that made the request.
  608. * @property {String} sessionToken If set, the session of the user that made the request.
  609. * @property {String} event The live query event that triggered the request.
  610. * @property {Parse.Object} object The object triggering the hook.
  611. * @property {Parse.Object} original If set, the object, as currently stored.
  612. * @property {Integer} clients The number of clients connected.
  613. * @property {Integer} subscriptions The number of subscriptions connected.
  614. * @property {Boolean} sendEvent If the LiveQuery event should be sent to the client. Set to false to prevent LiveQuery from pushing to the client.
  615. */
  616. /**
  617. * @interface Parse.Cloud.BeforeFindRequest
  618. * @property {String} installationId If set, the installationId triggering the request.
  619. * @property {Boolean} master If true, means the master key was used.
  620. * @property {Parse.User} user If set, the user that made the request.
  621. * @property {Parse.Query} query The query triggering the hook.
  622. * @property {String} ip The IP address of the client making the request.
  623. * @property {Object} headers The original HTTP headers for the request.
  624. * @property {String} triggerName The name of the trigger (`beforeSave`, `afterSave`, ...)
  625. * @property {Object} log The current logger inside Parse Server.
  626. * @property {Boolean} isGet wether the query a `get` or a `find`
  627. */
  628. /**
  629. * @interface Parse.Cloud.AfterFindRequest
  630. * @property {String} installationId If set, the installationId triggering the request.
  631. * @property {Boolean} master If true, means the master key was used.
  632. * @property {Parse.User} user If set, the user that made the request.
  633. * @property {Parse.Query} query The query triggering the hook.
  634. * @property {Array<Parse.Object>} results The results the query yielded.
  635. * @property {String} ip The IP address of the client making the request.
  636. * @property {Object} headers The original HTTP headers for the request.
  637. * @property {String} triggerName The name of the trigger (`beforeSave`, `afterSave`, ...)
  638. * @property {Object} log The current logger inside Parse Server.
  639. */
  640. /**
  641. * @interface Parse.Cloud.FunctionRequest
  642. * @property {String} installationId If set, the installationId triggering the request.
  643. * @property {Boolean} master If true, means the master key was used.
  644. * @property {Parse.User} user If set, the user that made the request.
  645. * @property {Object} params The params passed to the cloud function.
  646. */
  647. /**
  648. * @interface Parse.Cloud.JobRequest
  649. * @property {Object} params The params passed to the background job.
  650. * @property {function} message If message is called with a string argument, will update the current message to be stored in the job status.
  651. */
  652. /**
  653. * @interface Parse.Cloud.ValidatorObject
  654. * @property {Boolean} requireUser whether the cloud trigger requires a user.
  655. * @property {Boolean} requireMaster whether the cloud trigger requires a master key.
  656. * @property {Boolean} validateMasterKey whether the validator should run if masterKey is provided. Defaults to false.
  657. * @property {Boolean} skipWithMasterKey whether the cloud code function should be ignored using a masterKey.
  658. *
  659. * @property {Array<String>|Object} requireUserKeys If set, keys required on request.user to make the request.
  660. * @property {String} requireUserKeys.field If requireUserKeys is an object, name of field to validate on request user
  661. * @property {Array|function|Any} requireUserKeys.field.options array of options that the field can be, function to validate field, or single value. Throw an error if value is invalid.
  662. * @property {String} requireUserKeys.field.error custom error message if field is invalid.
  663. *
  664. * @property {Array<String>|function}requireAnyUserRoles If set, request.user has to be part of at least one roles name to make the request. If set to a function, function must return role names.
  665. * @property {Array<String>|function}requireAllUserRoles If set, request.user has to be part all roles name to make the request. If set to a function, function must return role names.
  666. *
  667. * @property {Object|Array<String>} fields if an array of strings, validator will look for keys in request.params, and throw if not provided. If Object, fields to validate. If the trigger is a cloud function, `request.params` will be validated, otherwise `request.object`.
  668. * @property {String} fields.field name of field to validate.
  669. * @property {String} fields.field.type expected type of data for field.
  670. * @property {Boolean} fields.field.constant whether the field can be modified on the object.
  671. * @property {Any} fields.field.default default value if field is `null`, or initial value `constant` is `true`.
  672. * @property {Array|function|Any} fields.field.options array of options that the field can be, function to validate field, or single value. Throw an error if value is invalid.
  673. * @property {String} fields.field.error custom error message if field is invalid.
  674. */
  675. //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_node","require","triggers","_interopRequireWildcard","_middlewares","_getRequireWildcardCache","e","WeakMap","r","t","__esModule","default","has","get","n","__proto__","a","Object","defineProperty","getOwnPropertyDescriptor","u","hasOwnProperty","call","i","set","ownKeys","keys","getOwnPropertySymbols","o","filter","enumerable","push","apply","_objectSpread","arguments","length","forEach","_defineProperty","getOwnPropertyDescriptors","defineProperties","_toPropertyKey","value","configurable","writable","_toPrimitive","Symbol","toPrimitive","TypeError","String","Number","Config","isParseObjectConstructor","object","prototype","validateValidator","validator","fieldOptions","type","constant","Boolean","options","Array","required","error","allowedKeys","requireUser","requireAnyUserRoles","requireAllUserRoles","requireMaster","validateMasterKey","skipWithMasterKey","requireUserKeys","fields","rateLimit","getType","fn","isArray","match","toString","toLowerCase","checkKey","key","data","validatorParam","parameter","types","map","includes","join","values","subKey","getRoute","parseClass","route","_User","_Session","ParseCloud","define","functionName","handler","validationHandler","addFunction","Parse","applicationId","addRateLimit","requestPath","job","addJob","beforeSave","className","getClassName","addTrigger","Types","requestMethods","beforeDelete","beforeLogin","afterLogin","afterLogout","afterSave","afterDelete","beforeFind","afterFind","beforeConnect","addConnectTrigger","sendEmail","config","emailAdapter","userController","adapter","loggerController","sendMail","beforeSubscribe","onLiveQueryEvent","addLiveQueryEventHandler","afterLiveQueryEvent","afterEvent","_removeAllHooks","_unregisterAll","unregisterRateLimiters","useMasterKey","console","warn","module","exports"],"sources":["../../src/cloud-code/Parse.Cloud.js"],"sourcesContent":["import { Parse } from 'parse/node';\nimport * as triggers from '../triggers';\nimport { addRateLimit } from '../middlewares';\nconst Config = require('../Config');\n\nfunction isParseObjectConstructor(object) {\n  return typeof object === 'function' && Object.prototype.hasOwnProperty.call(object, 'className');\n}\n\nfunction validateValidator(validator) {\n  if (!validator || typeof validator === 'function') {\n    return;\n  }\n  const fieldOptions = {\n    type: ['Any'],\n    constant: [Boolean],\n    default: ['Any'],\n    options: [Array, 'function', 'Any'],\n    required: [Boolean],\n    error: [String],\n  };\n  const allowedKeys = {\n    requireUser: [Boolean],\n    requireAnyUserRoles: [Array, 'function'],\n    requireAllUserRoles: [Array, 'function'],\n    requireMaster: [Boolean],\n    validateMasterKey: [Boolean],\n    skipWithMasterKey: [Boolean],\n    requireUserKeys: [Array, Object],\n    fields: [Array, Object],\n    rateLimit: [Object],\n  };\n  const getType = fn => {\n    if (Array.isArray(fn)) {\n      return 'array';\n    }\n    if (fn === 'Any' || fn === 'function') {\n      return fn;\n    }\n    const type = typeof fn;\n    if (typeof fn === 'function') {\n      const match = fn && fn.toString().match(/^\\s*function (\\w+)/);\n      return (match ? match[1] : 'function').toLowerCase();\n    }\n    return type;\n  };\n  const checkKey = (key, data, validatorParam) => {\n    const parameter = data[key];\n    if (!parameter) {\n      throw `${key} is not a supported parameter for Cloud Function validations.`;\n    }\n    const types = parameter.map(type => getType(type));\n    const type = getType(validatorParam);\n    if (!types.includes(type) && !types.includes('Any')) {\n      throw `Invalid type for Cloud Function validation key ${key}. Expected ${types.join(\n        '|'\n      )}, actual ${type}`;\n    }\n  };\n  for (const key in validator) {\n    checkKey(key, allowedKeys, validator[key]);\n    if (key === 'fields' || key === 'requireUserKeys') {\n      const values = validator[key];\n      if (Array.isArray(values)) {\n        continue;\n      }\n      for (const value in values) {\n        const data = values[value];\n        for (const subKey in data) {\n          checkKey(subKey, fieldOptions, data[subKey]);\n        }\n      }\n    }\n  }\n}\nconst getRoute = parseClass => {\n  const route =\n    {\n      _User: 'users',\n      _Session: 'sessions',\n      '@File': 'files',\n      '@Config' : 'config',\n    }[parseClass] || 'classes';\n  if (parseClass === '@File') {\n    return `/${route}/:id?(.*)`;\n  }\n  if (parseClass === '@Config') {\n    return `/${route}`;\n  }\n  return `/${route}/${parseClass}/:id?(.*)`;\n};\n/** @namespace\n * @name Parse\n * @description The Parse SDK.\n *  see [api docs](https://docs.parseplatform.org/js/api) and [guide](https://docs.parseplatform.org/js/guide)\n */\n\n/** @namespace\n * @name Parse.Cloud\n * @memberof Parse\n * @description The Parse Cloud Code SDK.\n */\n\nvar ParseCloud = {};\n/**\n * Defines a Cloud Function.\n *\n * **Available in Cloud Code only.**\n *\n * ```\n * Parse.Cloud.define('functionName', (request) => {\n *   // code here\n * }, (request) => {\n *   // validation code here\n * });\n *\n * Parse.Cloud.define('functionName', (request) => {\n *   // code here\n * }, { ...validationObject });\n * ```\n *\n * @static\n * @memberof Parse.Cloud\n * @param {String} name The name of the Cloud Function\n * @param {Function} data The Cloud Function to register. This function can be an async function and should take one parameter a {@link Parse.Cloud.FunctionRequest}.\n * @param {(Object|Function)} validator An optional function to help validating cloud code. This function can be an async function and should take one parameter a {@link Parse.Cloud.FunctionRequest}, or a {@link Parse.Cloud.ValidatorObject}.\n */\nParseCloud.define = function (functionName, handler, validationHandler) {\n  validateValidator(validationHandler);\n  triggers.addFunction(functionName, handler, validationHandler, Parse.applicationId);\n  if (validationHandler && validationHandler.rateLimit) {\n    addRateLimit(\n      { requestPath: `/functions/${functionName}`, ...validationHandler.rateLimit },\n      Parse.applicationId,\n      true\n    );\n  }\n};\n\n/**\n * Defines a Background Job.\n *\n * **Available in Cloud Code only.**\n *\n * @method job\n * @name Parse.Cloud.job\n * @param {String} name The name of the Background Job\n * @param {Function} func The Background Job to register. This function can be async should take a single parameters a {@link Parse.Cloud.JobRequest}\n *\n */\nParseCloud.job = function (functionName, handler) {\n  triggers.addJob(functionName, handler, Parse.applicationId);\n};\n\n/**\n *\n * Registers a before save function.\n *\n * **Available in Cloud Code only.**\n *\n * If you want to use beforeSave for a predefined class in the Parse JavaScript SDK (e.g. {@link Parse.User} or {@link Parse.File}), you should pass the class itself and not the String for arg1.\n *\n * ```\n * Parse.Cloud.beforeSave('MyCustomClass', (request) => {\n *   // code here\n * }, (request) => {\n *   // validation code here\n * });\n *\n * Parse.Cloud.beforeSave(Parse.User, (request) => {\n *   // code here\n * }, { ...validationObject })\n * ```\n *\n * @method beforeSave\n * @name Parse.Cloud.beforeSave\n * @param {(String|Parse.Object)} arg1 The Parse.Object subclass to register the after save function for. This can instead be a String that is the className of the subclass.\n * @param {Function} func The function to run before a save. This function can be async and should take one parameter a {@link Parse.Cloud.TriggerRequest};\n * @param {(Object|Function)} validator An optional function to help validating cloud code. This function can be an async function and should take one parameter a {@link Parse.Cloud.TriggerRequest}, or a {@link Parse.Cloud.ValidatorObject}.\n */\nParseCloud.beforeSave = function (parseClass, handler, validationHandler) {\n  const className = triggers.getClassName(parseClass);\n  validateValidator(validationHandler);\n  triggers.addTrigger(\n    triggers.Types.beforeSave,\n    className,\n    handler,\n    Parse.applicationId,\n    validationHandler\n  );\n  if (validationHandler && validationHandler.rateLimit) {\n    addRateLimit(\n      {\n        requestPath: getRoute(className),\n        requestMethods: ['POST', 'PUT'],\n        ...validationHandler.rateLimit,\n      },\n      Parse.applicationId,\n      true\n    );\n  }\n};\n\n/**\n * Registers a before delete function.\n *\n * **Available in Cloud Code only.**\n *\n * If you want to use beforeDelete for a predefined class in the Parse JavaScript SDK (e.g. {@link Parse.User} or {@link Parse.File}), you should pass the class itself and not the String for arg1.\n * ```\n * Parse.Cloud.beforeDelete('MyCustomClass', (request) => {\n *   // code here\n * }, (request) => {\n *   // validation code here\n * });\n *\n * Parse.Cloud.beforeDelete(Parse.User, (request) => {\n *   // code here\n * }, { ...validationObject })\n *```\n *\n * @method beforeDelete\n * @name Parse.Cloud.beforeDelete\n * @param {(String|Parse.Object)} arg1 The Parse.Object subclass to register the before delete function for. This can instead be a String that is the className of the subclass.\n * @param {Function} func The function to run before a delete. This function can be async and should take one parameter, a {@link Parse.Cloud.TriggerRequest}.\n * @param {(Object|Function)} validator An optional function to help validating cloud code. This function can be an async function and should take one parameter a {@link Parse.Cloud.TriggerRequest}, or a {@link Parse.Cloud.ValidatorObject}.\n */\nParseCloud.beforeDelete = function (parseClass, handler, validationHandler) {\n  const className = triggers.getClassName(parseClass);\n  validateValidator(validationHandler);\n  triggers.addTrigger(\n    triggers.Types.beforeDelete,\n    className,\n    handler,\n    Parse.applicationId,\n    validationHandler\n  );\n  if (validationHandler && validationHandler.rateLimit) {\n    addRateLimit(\n      {\n        requestPath: getRoute(className),\n        requestMethods: 'DELETE',\n        ...validationHandler.rateLimit,\n      },\n      Parse.applicationId,\n      true\n    );\n  }\n};\n\n/**\n *\n * Registers the before login function.\n *\n * **Available in Cloud Code only.**\n *\n * This function provides further control\n * in validating a login attempt. Specifically,\n * it is triggered after a user enters\n * correct credentials (or other valid authData),\n * but prior to a session being generated.\n *\n * ```\n * Parse.Cloud.beforeLogin((request) => {\n *   // code here\n * })\n *\n * ```\n *\n * @method beforeLogin\n * @name Parse.Cloud.beforeLogin\n * @param {Function} func The function to run before a login. This function can be async and should take one parameter a {@link Parse.Cloud.TriggerRequest};\n */\nParseCloud.beforeLogin = function (handler, validationHandler) {\n  let className = '_User';\n  if (typeof handler === 'string' || isParseObjectConstructor(handler)) {\n    // validation will occur downstream, this is to maintain internal\n    // code consistency with the other hook types.\n    className = triggers.getClassName(handler);\n    handler = arguments[1];\n    validationHandler = arguments.length >= 2 ? arguments[2] : null;\n  }\n  triggers.addTrigger(triggers.Types.beforeLogin, className, handler, Parse.applicationId);\n  if (validationHandler && validationHandler.rateLimit) {\n    addRateLimit(\n      { requestPath: `/login`, requestMethods: 'POST', ...validationHandler.rateLimit },\n      Parse.applicationId,\n      true\n    );\n  }\n};\n\n/**\n *\n * Registers the after login function.\n *\n * **Available in Cloud Code only.**\n *\n * This function is triggered after a user logs in successfully,\n * and after a _Session object has been created.\n *\n * ```\n * Parse.Cloud.afterLogin((request) => {\n *   // code here\n * });\n * ```\n *\n * @method afterLogin\n * @name Parse.Cloud.afterLogin\n * @param {Function} func The function to run after a login. This function can be async and should take one parameter a {@link Parse.Cloud.TriggerRequest};\n */\nParseCloud.afterLogin = function (handler) {\n  let className = '_User';\n  if (typeof handler === 'string' || isParseObjectConstructor(handler)) {\n    // validation will occur downstream, this is to maintain internal\n    // code consistency with the other hook types.\n    className = triggers.getClassName(handler);\n    handler = arguments[1];\n  }\n  triggers.addTrigger(triggers.Types.afterLogin, className, handler, Parse.applicationId);\n};\n\n/**\n *\n * Registers the after logout function.\n *\n * **Available in Cloud Code only.**\n *\n * This function is triggered after a user logs out.\n *\n * ```\n * Parse.Cloud.afterLogout((request) => {\n *   // code here\n * });\n * ```\n *\n * @method afterLogout\n * @name Parse.Cloud.afterLogout\n * @param {Function} func The function to run after a logout. This function can be async and should take one parameter a {@link Parse.Cloud.TriggerRequest};\n */\nParseCloud.afterLogout = function (handler) {\n  let className = '_Session';\n  if (typeof handler === 'string' || isParseObjectConstructor(handler)) {\n    // validation will occur downstream, this is to maintain internal\n    // code consistency with the other hook types.\n    className = triggers.getClassName(handler);\n    handler = arguments[1];\n  }\n  triggers.addTrigger(triggers.Types.afterLogout, className, handler, Parse.applicationId);\n};\n\n/**\n * Registers an after save function.\n *\n * **Available in Cloud Code only.**\n *\n * If you want to use afterSave for a predefined class in the Parse JavaScript SDK (e.g. {@link Parse.User} or {@link Parse.File}), you should pass the class itself and not the String for arg1.\n *\n * ```\n * Parse.Cloud.afterSave('MyCustomClass', async function(request) {\n *   // code here\n * }, (request) => {\n *   // validation code here\n * });\n *\n * Parse.Cloud.afterSave(Parse.User, async function(request) {\n *   // code here\n * }, { ...validationObject });\n * ```\n *\n * @method afterSave\n * @name Parse.Cloud.afterSave\n * @param {(String|Parse.Object)} arg1 The Parse.Object subclass to register the after save function for. This can instead be a String that is the className of the subclass.\n * @param {Function} func The function to run after a save. This function can be an async function and should take just one parameter, {@link Parse.Cloud.TriggerRequest}.\n * @param {(Object|Function)} validator An optional function to help validating cloud code. This function can be an async function and should take one parameter a {@link Parse.Cloud.TriggerRequest}, or a {@link Parse.Cloud.ValidatorObject}.\n */\nParseCloud.afterSave = function (parseClass, handler, validationHandler) {\n  const className = triggers.getClassName(parseClass);\n  validateValidator(validationHandler);\n  triggers.addTrigger(\n    triggers.Types.afterSave,\n    className,\n    handler,\n    Parse.applicationId,\n    validationHandler\n  );\n};\n\n/**\n * Registers an after delete function.\n *\n * **Available in Cloud Code only.**\n *\n * If you want to use afterDelete for a predefined class in the Parse JavaScript SDK (e.g. {@link Parse.User} or {@link Parse.File}), you should pass the class itself and not the String for arg1.\n * ```\n * Parse.Cloud.afterDelete('MyCustomClass', async (request) => {\n *   // code here\n * }, (request) => {\n *   // validation code here\n * });\n *\n * Parse.Cloud.afterDelete(Parse.User, async (request) => {\n *   // code here\n * }, { ...validationObject });\n *```\n *\n * @method afterDelete\n * @name Parse.Cloud.afterDelete\n * @param {(String|Parse.Object)} arg1 The Parse.Object subclass to register the after delete function for. This can instead be a String that is the className of the subclass.\n * @param {Function} func The function to run after a delete. This function can be async and should take just one parameter, {@link Parse.Cloud.TriggerRequest}.\n * @param {(Object|Function)} validator An optional function to help validating cloud code. This function can be an async function and should take one parameter a {@link Parse.Cloud.TriggerRequest}, or a {@link Parse.Cloud.ValidatorObject}.\n */\nParseCloud.afterDelete = function (parseClass, handler, validationHandler) {\n  const className = triggers.getClassName(parseClass);\n  validateValidator(validationHandler);\n  triggers.addTrigger(\n    triggers.Types.afterDelete,\n    className,\n    handler,\n    Parse.applicationId,\n    validationHandler\n  );\n};\n\n/**\n * Registers a before find function.\n *\n * **Available in Cloud Code only.**\n *\n * If you want to use beforeFind for a predefined class in the Parse JavaScript SDK (e.g. {@link Parse.User} or {@link Parse.File}), you should pass the class itself and not the String for arg1.\n * ```\n * Parse.Cloud.beforeFind('MyCustomClass', async (request) => {\n *   // code here\n * }, (request) => {\n *   // validation code here\n * });\n *\n * Parse.Cloud.beforeFind(Parse.User, async (request) => {\n *   // code here\n * }, { ...validationObject });\n *```\n *\n * @method beforeFind\n * @name Parse.Cloud.beforeFind\n * @param {(String|Parse.Object)} arg1 The Parse.Object subclass to register the before find function for. This can instead be a String that is the className of the subclass.\n * @param {Function} func The function to run before a find. This function can be async and should take just one parameter, {@link Parse.Cloud.BeforeFindRequest}.\n * @param {(Object|Function)} validator An optional function to help validating cloud code. This function can be an async function and should take one parameter a {@link Parse.Cloud.BeforeFindRequest}, or a {@link Parse.Cloud.ValidatorObject}.\n */\nParseCloud.beforeFind = function (parseClass, handler, validationHandler) {\n  const className = triggers.getClassName(parseClass);\n  validateValidator(validationHandler);\n  triggers.addTrigger(\n    triggers.Types.beforeFind,\n    className,\n    handler,\n    Parse.applicationId,\n    validationHandler\n  );\n  if (validationHandler && validationHandler.rateLimit) {\n    addRateLimit(\n      {\n        requestPath: getRoute(className),\n        requestMethods: 'GET',\n        ...validationHandler.rateLimit,\n      },\n      Parse.applicationId,\n      true\n    );\n  }\n};\n\n/**\n * Registers an after find function.\n *\n * **Available in Cloud Code only.**\n *\n * If you want to use afterFind for a predefined class in the Parse JavaScript SDK (e.g. {@link Parse.User} or {@link Parse.File}), you should pass the class itself and not the String for arg1.\n * ```\n * Parse.Cloud.afterFind('MyCustomClass', async (request) => {\n *   // code here\n * }, (request) => {\n *   // validation code here\n * });\n *\n * Parse.Cloud.afterFind(Parse.User, async (request) => {\n *   // code here\n * }, { ...validationObject });\n *```\n *\n * @method afterFind\n * @name Parse.Cloud.afterFind\n * @param {(String|Parse.Object)} arg1 The Parse.Object subclass to register the after find function for. This can instead be a String that is the className of the subclass.\n * @param {Function} func The function to run before a find. This function can be async and should take just one parameter, {@link Parse.Cloud.AfterFindRequest}.\n * @param {(Object|Function)} validator An optional function to help validating cloud code. This function can be an async function and should take one parameter a {@link Parse.Cloud.AfterFindRequest}, or a {@link Parse.Cloud.ValidatorObject}.\n */\nParseCloud.afterFind = function (parseClass, handler, validationHandler) {\n  const className = triggers.getClassName(parseClass);\n  validateValidator(validationHandler);\n  triggers.addTrigger(\n    triggers.Types.afterFind,\n    className,\n    handler,\n    Parse.applicationId,\n    validationHandler\n  );\n};\n\n/**\n * Registers a before live query server connect function.\n *\n * **Available in Cloud Code only.**\n *\n * ```\n * Parse.Cloud.beforeConnect(async (request) => {\n *   // code here\n * }, (request) => {\n *   // validation code here\n * });\n *\n * Parse.Cloud.beforeConnect(async (request) => {\n *   // code here\n * }, { ...validationObject });\n *```\n *\n * @method beforeConnect\n * @name Parse.Cloud.beforeConnect\n * @param {Function} func The function to before connection is made. This function can be async and should take just one parameter, {@link Parse.Cloud.ConnectTriggerRequest}.\n * @param {(Object|Function)} validator An optional function to help validating cloud code. This function can be an async function and should take one parameter a {@link Parse.Cloud.ConnectTriggerRequest}, or a {@link Parse.Cloud.ValidatorObject}.\n */\nParseCloud.beforeConnect = function (handler, validationHandler) {\n  validateValidator(validationHandler);\n  triggers.addConnectTrigger(\n    triggers.Types.beforeConnect,\n    handler,\n    Parse.applicationId,\n    validationHandler\n  );\n};\n\n/**\n * Sends an email through the Parse Server mail adapter.\n *\n * **Available in Cloud Code only.**\n * **Requires a mail adapter to be configured for Parse Server.**\n *\n * ```\n * Parse.Cloud.sendEmail({\n *   from: 'Example <test@example.com>',\n *   to: 'contact@example.com',\n *   subject: 'Test email',\n *   text: 'This email is a test.'\n * });\n *```\n *\n * @method sendEmail\n * @name Parse.Cloud.sendEmail\n * @param {Object} data The object of the mail data to send.\n */\nParseCloud.sendEmail = function (data) {\n  const config = Config.get(Parse.applicationId);\n  const emailAdapter = config.userController.adapter;\n  if (!emailAdapter) {\n    config.loggerController.error(\n      'Failed to send email because no mail adapter is configured for Parse Server.'\n    );\n    return;\n  }\n  return emailAdapter.sendMail(data);\n};\n\n/**\n * Registers a before live query subscription function.\n *\n * **Available in Cloud Code only.**\n *\n * If you want to use beforeSubscribe for a predefined class in the Parse JavaScript SDK (e.g. {@link Parse.User} or {@link Parse.File}), you should pass the class itself and not the String for arg1.\n * ```\n * Parse.Cloud.beforeSubscribe('MyCustomClass', (request) => {\n *   // code here\n * }, (request) => {\n *   // validation code here\n * });\n *\n * Parse.Cloud.beforeSubscribe(Parse.User, (request) => {\n *   // code here\n * }, { ...validationObject });\n *```\n *\n * @method beforeSubscribe\n * @name Parse.Cloud.beforeSubscribe\n * @param {(String|Parse.Object)} arg1 The Parse.Object subclass to register the before subscription function for. This can instead be a String that is the className of the subclass.\n * @param {Function} func The function to run before a subscription. This function can be async and should take one parameter, a {@link Parse.Cloud.TriggerRequest}.\n * @param {(Object|Function)} validator An optional function to help validating cloud code. This function can be an async function and should take one parameter a {@link Parse.Cloud.TriggerRequest}, or a {@link Parse.Cloud.ValidatorObject}.\n */\nParseCloud.beforeSubscribe = function (parseClass, handler, validationHandler) {\n  validateValidator(validationHandler);\n  const className = triggers.getClassName(parseClass);\n  triggers.addTrigger(\n    triggers.Types.beforeSubscribe,\n    className,\n    handler,\n    Parse.applicationId,\n    validationHandler\n  );\n};\n\nParseCloud.onLiveQueryEvent = function (handler) {\n  triggers.addLiveQueryEventHandler(handler, Parse.applicationId);\n};\n\n/**\n * Registers an after live query server event function.\n *\n * **Available in Cloud Code only.**\n *\n * ```\n * Parse.Cloud.afterLiveQueryEvent('MyCustomClass', (request) => {\n *   // code here\n * }, (request) => {\n *   // validation code here\n * });\n *\n * Parse.Cloud.afterLiveQueryEvent('MyCustomClass', (request) => {\n *   // code here\n * }, { ...validationObject });\n *```\n *\n * @method afterLiveQueryEvent\n * @name Parse.Cloud.afterLiveQueryEvent\n * @param {(String|Parse.Object)} arg1 The Parse.Object subclass to register the after live query event function for. This can instead be a String that is the className of the subclass.\n * @param {Function} func The function to run after a live query event. This function can be async and should take one parameter, a {@link Parse.Cloud.LiveQueryEventTrigger}.\n * @param {(Object|Function)} validator An optional function to help validating cloud code. This function can be an async function and should take one parameter a {@link Parse.Cloud.LiveQueryEventTrigger}, or a {@link Parse.Cloud.ValidatorObject}.\n */\nParseCloud.afterLiveQueryEvent = function (parseClass, handler, validationHandler) {\n  const className = triggers.getClassName(parseClass);\n  validateValidator(validationHandler);\n  triggers.addTrigger(\n    triggers.Types.afterEvent,\n    className,\n    handler,\n    Parse.applicationId,\n    validationHandler\n  );\n};\n\nParseCloud._removeAllHooks = () => {\n  triggers._unregisterAll();\n  const config = Config.get(Parse.applicationId);\n  config?.unregisterRateLimiters();\n};\n\nParseCloud.useMasterKey = () => {\n  // eslint-disable-next-line\n  console.warn(\n    'Parse.Cloud.useMasterKey is deprecated (and has no effect anymore) on parse-server, please refer to the cloud code migration notes: http://docs.parseplatform.org/parse-server/guide/#master-key-must-be-passed-explicitly'\n  );\n};\n\nmodule.exports = ParseCloud;\n\n/**\n * @interface Parse.Cloud.TriggerRequest\n * @property {String} installationId If set, the installationId triggering the request.\n * @property {Boolean} master If true, means the master key was used.\n * @property {Boolean} isChallenge If true, means the current request is originally triggered by an auth challenge.\n * @property {Parse.User} user If set, the user that made the request.\n * @property {Parse.Object} object The object triggering the hook.\n * @property {String} ip The IP address of the client making the request. To ensure retrieving the correct IP address, set the Parse Server option `trustProxy: true` if Parse Server runs behind a proxy server, for example behind a load balancer.\n * @property {Object} headers The original HTTP headers for the request.\n * @property {String} triggerName The name of the trigger (`beforeSave`, `afterSave`, ...)\n * @property {Object} log The current logger inside Parse Server.\n * @property {Parse.Object} original If set, the object, as currently stored.\n */\n\n/**\n * @interface Parse.Cloud.FileTriggerRequest\n * @property {String} installationId If set, the installationId triggering the request.\n * @property {Boolean} master If true, means the master key was used.\n * @property {Parse.User} user If set, the user that made the request.\n * @property {Parse.File} file The file that triggered the hook.\n * @property {Integer} fileSize The size of the file in bytes.\n * @property {Integer} contentLength The value from Content-Length header\n * @property {String} ip The IP address of the client making the request.\n * @property {Object} headers The original HTTP headers for the request.\n * @property {String} triggerName The name of the trigger (`beforeSave`, `afterSave`)\n * @property {Object} log The current logger inside Parse Server.\n */\n\n/**\n * @interface Parse.Cloud.ConnectTriggerRequest\n * @property {String} installationId If set, the installationId triggering the request.\n * @property {Boolean} useMasterKey If true, means the master key was used.\n * @property {Parse.User} user If set, the user that made the request.\n * @property {Integer} clients The number of clients connected.\n * @property {Integer} subscriptions The number of subscriptions connected.\n * @property {String} sessionToken If set, the session of the user that made the request.\n */\n\n/**\n * @interface Parse.Cloud.LiveQueryEventTrigger\n * @property {String} installationId If set, the installationId triggering the request.\n * @property {Boolean} useMasterKey If true, means the master key was used.\n * @property {Parse.User} user If set, the user that made the request.\n * @property {String} sessionToken If set, the session of the user that made the request.\n * @property {String} event The live query event that triggered the request.\n * @property {Parse.Object} object The object triggering the hook.\n * @property {Parse.Object} original If set, the object, as currently stored.\n * @property {Integer} clients The number of clients connected.\n * @property {Integer} subscriptions The number of subscriptions connected.\n * @property {Boolean} sendEvent If the LiveQuery event should be sent to the client. Set to false to prevent LiveQuery from pushing to the client.\n */\n\n/**\n * @interface Parse.Cloud.BeforeFindRequest\n * @property {String} installationId If set, the installationId triggering the request.\n * @property {Boolean} master If true, means the master key was used.\n * @property {Parse.User} user If set, the user that made the request.\n * @property {Parse.Query} query The query triggering the hook.\n * @property {String} ip The IP address of the client making the request.\n * @property {Object} headers The original HTTP headers for the request.\n * @property {String} triggerName The name of the trigger (`beforeSave`, `afterSave`, ...)\n * @property {Object} log The current logger inside Parse Server.\n * @property {Boolean} isGet wether the query a `get` or a `find`\n */\n\n/**\n * @interface Parse.Cloud.AfterFindRequest\n * @property {String} installationId If set, the installationId triggering the request.\n * @property {Boolean} master If true, means the master key was used.\n * @property {Parse.User} user If set, the user that made the request.\n * @property {Parse.Query} query The query triggering the hook.\n * @property {Array<Parse.Object>} results The results the query yielded.\n * @property {String} ip The IP address of the client making the request.\n * @property {Object} headers The original HTTP headers for the request.\n * @property {String} triggerName The name of the trigger (`beforeSave`, `afterSave`, ...)\n * @property {Object} log The current logger inside Parse Server.\n */\n\n/**\n * @interface Parse.Cloud.FunctionRequest\n * @property {String} installationId If set, the installationId triggering the request.\n * @property {Boolean} master If true, means the master key was used.\n * @property {Parse.User} user If set, the user that made the request.\n * @property {Object} params The params passed to the cloud function.\n */\n\n/**\n * @interface Parse.Cloud.JobRequest\n * @property {Object} params The params passed to the background job.\n * @property {function} message If message is called with a string argument, will update the current message to be stored in the job status.\n */\n\n/**\n * @interface Parse.Cloud.ValidatorObject\n * @property {Boolean} requireUser whether the cloud trigger requires a user.\n * @property {Boolean} requireMaster whether the cloud trigger requires a master key.\n * @property {Boolean} validateMasterKey whether the validator should run if masterKey is provided. Defaults to false.\n * @property {Boolean} skipWithMasterKey whether the cloud code function should be ignored using a masterKey.\n *\n * @property {Array<String>|Object} requireUserKeys If set, keys required on request.user to make the request.\n * @property {String} requireUserKeys.field If requireUserKeys is an object, name of field to validate on request user\n * @property {Array|function|Any} requireUserKeys.field.options array of options that the field can be, function to validate field, or single value. Throw an error if value is invalid.\n * @property {String} requireUserKeys.field.error custom error message if field is invalid.\n *\n * @property {Array<String>|function}requireAnyUserRoles If set, request.user has to be part of at least one roles name to make the request. If set to a function, function must return role names.\n * @property {Array<String>|function}requireAllUserRoles If set, request.user has to be part all roles name to make the request. If set to a function, function must return role names.\n *\n * @property {Object|Array<String>} fields if an array of strings, validator will look for keys in request.params, and throw if not provided. If Object, fields to validate. If the trigger is a cloud function, `request.params` will be validated, otherwise `request.object`.\n * @property {String} fields.field name of field to validate.\n * @property {String} fields.field.type expected type of data for field.\n * @property {Boolean} fields.field.constant whether the field can be modified on the object.\n * @property {Any} fields.field.default default value if field is `null`, or initial value `constant` is `true`.\n * @property {Array|function|Any} fields.field.options array of options that the field can be, function to validate field, or single value. Throw an error if value is invalid.\n * @property {String} fields.field.error custom error message if field is invalid.\n */\n"],"mappings":"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EAAEG,WAAK,CAACC,aAAa,CAAC;AACjE,CAAC;;AAED;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACAP,UAAU,CAACsC,mBAAmB,GAAG,UAAU1C,UAAU,EAAEO,OAAO,EAAEC,iBAAiB,EAAE;EACjF,MAAMS,SAAS,GAAG3G,QAAQ,CAAC4G,YAAY,CAAClB,UAAU,CAAC;EACnDtC,iBAAiB,CAAC8C,iBAAiB,CAAC;EACpClG,QAAQ,CAAC6G,UAAU,CACjB7G,QAAQ,CAAC8G,KAAK,CAACuB,UAAU,EACzB1B,SAAS,EACTV,OAAO,EACPG,WAAK,CAACC,aAAa,EACnBH,iBACF,CAAC;AACH,CAAC;AAEDJ,UAAU,CAACwC,eAAe,GAAG,MAAM;EACjCtI,QAAQ,CAACuI,cAAc,CAAC,CAAC;EACzB,MAAMZ,MAAM,GAAG3E,MAAM,CAACrC,GAAG,CAACyF,WAAK,CAACC,aAAa,CAAC;EAC9CsB,MAAM,aAANA,MAAM,eAANA,MAAM,CAAEa,sBAAsB,CAAC,CAAC;AAClC,CAAC;AAED1C,UAAU,CAAC2C,YAAY,GAAG,MAAM;EAC9B;EACAC,OAAO,CAACC,IAAI,CACV,4NACF,CAAC;AACH,CAAC;AAEDC,MAAM,CAACC,OAAO,GAAG/C,UAAU;;AAE3B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","ignoreList":[]}