Auth.js 73 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608
  1. "use strict";
  2. var _util = require("util");
  3. var _triggers = require("./triggers");
  4. var _logger = require("./logger");
  5. var _RestQuery = _interopRequireDefault(require("./RestQuery"));
  6. var _RestWrite = _interopRequireDefault(require("./RestWrite"));
  7. function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
  8. 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; }
  9. 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; }
  10. 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; }
  11. function _toPropertyKey(t) { var i = _toPrimitive(t, "string"); return "symbol" == typeof i ? i : i + ""; }
  12. 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); }
  13. const Parse = require('parse/node');
  14. // An Auth object tells you who is requesting something and whether
  15. // the master key was used.
  16. // userObject is a Parse.User and can be null if there's no user.
  17. function Auth({
  18. config,
  19. cacheController = undefined,
  20. isMaster = false,
  21. isMaintenance = false,
  22. isReadOnly = false,
  23. user,
  24. installationId
  25. }) {
  26. this.config = config;
  27. this.cacheController = cacheController || config && config.cacheController;
  28. this.installationId = installationId;
  29. this.isMaster = isMaster;
  30. this.isMaintenance = isMaintenance;
  31. this.user = user;
  32. this.isReadOnly = isReadOnly;
  33. // Assuming a users roles won't change during a single request, we'll
  34. // only load them once.
  35. this.userRoles = [];
  36. this.fetchedRoles = false;
  37. this.rolePromise = null;
  38. }
  39. // Whether this auth could possibly modify the given user id.
  40. // It still could be forbidden via ACLs even if this returns true.
  41. Auth.prototype.isUnauthenticated = function () {
  42. if (this.isMaster) {
  43. return false;
  44. }
  45. if (this.isMaintenance) {
  46. return false;
  47. }
  48. if (this.user) {
  49. return false;
  50. }
  51. return true;
  52. };
  53. // A helper to get a master-level Auth object
  54. function master(config) {
  55. return new Auth({
  56. config,
  57. isMaster: true
  58. });
  59. }
  60. // A helper to get a maintenance-level Auth object
  61. function maintenance(config) {
  62. return new Auth({
  63. config,
  64. isMaintenance: true
  65. });
  66. }
  67. // A helper to get a master-level Auth object
  68. function readOnly(config) {
  69. return new Auth({
  70. config,
  71. isMaster: true,
  72. isReadOnly: true
  73. });
  74. }
  75. // A helper to get a nobody-level Auth object
  76. function nobody(config) {
  77. return new Auth({
  78. config,
  79. isMaster: false
  80. });
  81. }
  82. /**
  83. * Checks whether session should be updated based on last update time & session length.
  84. */
  85. function shouldUpdateSessionExpiry(config, session) {
  86. const resetAfter = config.sessionLength / 2;
  87. const lastUpdated = new Date(session === null || session === void 0 ? void 0 : session.updatedAt);
  88. const skipRange = new Date();
  89. skipRange.setTime(skipRange.getTime() - resetAfter * 1000);
  90. return lastUpdated <= skipRange;
  91. }
  92. const throttle = {};
  93. const renewSessionIfNeeded = async ({
  94. config,
  95. session,
  96. sessionToken
  97. }) => {
  98. if (!(config !== null && config !== void 0 && config.extendSessionOnUse)) {
  99. return;
  100. }
  101. clearTimeout(throttle[sessionToken]);
  102. throttle[sessionToken] = setTimeout(async () => {
  103. try {
  104. if (!session) {
  105. const query = await (0, _RestQuery.default)({
  106. method: _RestQuery.default.Method.get,
  107. config,
  108. auth: master(config),
  109. runBeforeFind: false,
  110. className: '_Session',
  111. restWhere: {
  112. sessionToken
  113. },
  114. restOptions: {
  115. limit: 1
  116. }
  117. });
  118. const {
  119. results
  120. } = await query.execute();
  121. session = results[0];
  122. }
  123. if (!shouldUpdateSessionExpiry(config, session) || !session) {
  124. return;
  125. }
  126. const expiresAt = config.generateSessionExpiresAt();
  127. await new _RestWrite.default(config, master(config), '_Session', {
  128. objectId: session.objectId
  129. }, {
  130. expiresAt: Parse._encode(expiresAt)
  131. }).execute();
  132. } catch (e) {
  133. if ((e === null || e === void 0 ? void 0 : e.code) !== Parse.Error.OBJECT_NOT_FOUND) {
  134. _logger.logger.error('Could not update session expiry: ', e);
  135. }
  136. }
  137. }, 500);
  138. };
  139. // Returns a promise that resolves to an Auth object
  140. const getAuthForSessionToken = async function ({
  141. config,
  142. cacheController,
  143. sessionToken,
  144. installationId
  145. }) {
  146. cacheController = cacheController || config && config.cacheController;
  147. if (cacheController) {
  148. const userJSON = await cacheController.user.get(sessionToken);
  149. if (userJSON) {
  150. const cachedUser = Parse.Object.fromJSON(userJSON);
  151. renewSessionIfNeeded({
  152. config,
  153. sessionToken
  154. });
  155. return Promise.resolve(new Auth({
  156. config,
  157. cacheController,
  158. isMaster: false,
  159. installationId,
  160. user: cachedUser
  161. }));
  162. }
  163. }
  164. let results;
  165. if (config) {
  166. const restOptions = {
  167. limit: 1,
  168. include: 'user'
  169. };
  170. const RestQuery = require('./RestQuery');
  171. const query = await RestQuery({
  172. method: RestQuery.Method.get,
  173. config,
  174. runBeforeFind: false,
  175. auth: master(config),
  176. className: '_Session',
  177. restWhere: {
  178. sessionToken
  179. },
  180. restOptions
  181. });
  182. results = (await query.execute()).results;
  183. } else {
  184. results = (await new Parse.Query(Parse.Session).limit(1).include('user').equalTo('sessionToken', sessionToken).find({
  185. useMasterKey: true
  186. })).map(obj => obj.toJSON());
  187. }
  188. if (results.length !== 1 || !results[0]['user']) {
  189. throw new Parse.Error(Parse.Error.INVALID_SESSION_TOKEN, 'Invalid session token');
  190. }
  191. const session = results[0];
  192. const now = new Date(),
  193. expiresAt = session.expiresAt ? new Date(session.expiresAt.iso) : undefined;
  194. if (expiresAt < now) {
  195. throw new Parse.Error(Parse.Error.INVALID_SESSION_TOKEN, 'Session token is expired.');
  196. }
  197. const obj = session.user;
  198. if (typeof obj['objectId'] === 'string' && obj['objectId'].startsWith('role:')) {
  199. throw new Parse.Error(Parse.Error.INTERNAL_SERVER_ERROR, 'Invalid object ID.');
  200. }
  201. delete obj.password;
  202. obj['className'] = '_User';
  203. obj['sessionToken'] = sessionToken;
  204. if (cacheController) {
  205. cacheController.user.put(sessionToken, obj);
  206. }
  207. renewSessionIfNeeded({
  208. config,
  209. session,
  210. sessionToken
  211. });
  212. const userObject = Parse.Object.fromJSON(obj);
  213. return new Auth({
  214. config,
  215. cacheController,
  216. isMaster: false,
  217. installationId,
  218. user: userObject
  219. });
  220. };
  221. var getAuthForLegacySessionToken = async function ({
  222. config,
  223. sessionToken,
  224. installationId
  225. }) {
  226. var restOptions = {
  227. limit: 1
  228. };
  229. const RestQuery = require('./RestQuery');
  230. var query = await RestQuery({
  231. method: RestQuery.Method.get,
  232. config,
  233. runBeforeFind: false,
  234. auth: master(config),
  235. className: '_User',
  236. restWhere: {
  237. _session_token: sessionToken
  238. },
  239. restOptions
  240. });
  241. return query.execute().then(response => {
  242. var results = response.results;
  243. if (results.length !== 1) {
  244. throw new Parse.Error(Parse.Error.INVALID_SESSION_TOKEN, 'invalid legacy session token');
  245. }
  246. const obj = results[0];
  247. obj.className = '_User';
  248. const userObject = Parse.Object.fromJSON(obj);
  249. return new Auth({
  250. config,
  251. isMaster: false,
  252. installationId,
  253. user: userObject
  254. });
  255. });
  256. };
  257. // Returns a promise that resolves to an array of role names
  258. Auth.prototype.getUserRoles = function () {
  259. if (this.isMaster || this.isMaintenance || !this.user) {
  260. return Promise.resolve([]);
  261. }
  262. if (this.fetchedRoles) {
  263. return Promise.resolve(this.userRoles);
  264. }
  265. if (this.rolePromise) {
  266. return this.rolePromise;
  267. }
  268. this.rolePromise = this._loadRoles();
  269. return this.rolePromise;
  270. };
  271. Auth.prototype.getRolesForUser = async function () {
  272. //Stack all Parse.Role
  273. const results = [];
  274. if (this.config) {
  275. const restWhere = {
  276. users: {
  277. __type: 'Pointer',
  278. className: '_User',
  279. objectId: this.user.id
  280. }
  281. };
  282. const RestQuery = require('./RestQuery');
  283. const query = await RestQuery({
  284. method: RestQuery.Method.find,
  285. runBeforeFind: false,
  286. config: this.config,
  287. auth: master(this.config),
  288. className: '_Role',
  289. restWhere
  290. });
  291. await query.each(result => results.push(result));
  292. } else {
  293. await new Parse.Query(Parse.Role).equalTo('users', this.user).each(result => results.push(result.toJSON()), {
  294. useMasterKey: true
  295. });
  296. }
  297. return results;
  298. };
  299. // Iterates through the role tree and compiles a user's roles
  300. Auth.prototype._loadRoles = async function () {
  301. if (this.cacheController) {
  302. const cachedRoles = await this.cacheController.role.get(this.user.id);
  303. if (cachedRoles != null) {
  304. this.fetchedRoles = true;
  305. this.userRoles = cachedRoles;
  306. return cachedRoles;
  307. }
  308. }
  309. // First get the role ids this user is directly a member of
  310. const results = await this.getRolesForUser();
  311. if (!results.length) {
  312. this.userRoles = [];
  313. this.fetchedRoles = true;
  314. this.rolePromise = null;
  315. this.cacheRoles();
  316. return this.userRoles;
  317. }
  318. const rolesMap = results.reduce((m, r) => {
  319. m.names.push(r.name);
  320. m.ids.push(r.objectId);
  321. return m;
  322. }, {
  323. ids: [],
  324. names: []
  325. });
  326. // run the recursive finding
  327. const roleNames = await this._getAllRolesNamesForRoleIds(rolesMap.ids, rolesMap.names);
  328. this.userRoles = roleNames.map(r => {
  329. return 'role:' + r;
  330. });
  331. this.fetchedRoles = true;
  332. this.rolePromise = null;
  333. this.cacheRoles();
  334. return this.userRoles;
  335. };
  336. Auth.prototype.cacheRoles = function () {
  337. if (!this.cacheController) {
  338. return false;
  339. }
  340. this.cacheController.role.put(this.user.id, Array(...this.userRoles));
  341. return true;
  342. };
  343. Auth.prototype.clearRoleCache = function (sessionToken) {
  344. if (!this.cacheController) {
  345. return false;
  346. }
  347. this.cacheController.role.del(this.user.id);
  348. this.cacheController.user.del(sessionToken);
  349. return true;
  350. };
  351. Auth.prototype.getRolesByIds = async function (ins) {
  352. const results = [];
  353. // Build an OR query across all parentRoles
  354. if (!this.config) {
  355. await new Parse.Query(Parse.Role).containedIn('roles', ins.map(id => {
  356. const role = new Parse.Object(Parse.Role);
  357. role.id = id;
  358. return role;
  359. })).each(result => results.push(result.toJSON()), {
  360. useMasterKey: true
  361. });
  362. } else {
  363. const roles = ins.map(id => {
  364. return {
  365. __type: 'Pointer',
  366. className: '_Role',
  367. objectId: id
  368. };
  369. });
  370. const restWhere = {
  371. roles: {
  372. $in: roles
  373. }
  374. };
  375. const RestQuery = require('./RestQuery');
  376. const query = await RestQuery({
  377. method: RestQuery.Method.find,
  378. config: this.config,
  379. runBeforeFind: false,
  380. auth: master(this.config),
  381. className: '_Role',
  382. restWhere
  383. });
  384. await query.each(result => results.push(result));
  385. }
  386. return results;
  387. };
  388. // Given a list of roleIds, find all the parent roles, returns a promise with all names
  389. Auth.prototype._getAllRolesNamesForRoleIds = function (roleIDs, names = [], queriedRoles = {}) {
  390. const ins = roleIDs.filter(roleID => {
  391. const wasQueried = queriedRoles[roleID] !== true;
  392. queriedRoles[roleID] = true;
  393. return wasQueried;
  394. });
  395. // all roles are accounted for, return the names
  396. if (ins.length == 0) {
  397. return Promise.resolve([...new Set(names)]);
  398. }
  399. return this.getRolesByIds(ins).then(results => {
  400. // Nothing found
  401. if (!results.length) {
  402. return Promise.resolve(names);
  403. }
  404. // Map the results with all Ids and names
  405. const resultMap = results.reduce((memo, role) => {
  406. memo.names.push(role.name);
  407. memo.ids.push(role.objectId);
  408. return memo;
  409. }, {
  410. ids: [],
  411. names: []
  412. });
  413. // store the new found names
  414. names = names.concat(resultMap.names);
  415. // find the next ones, circular roles will be cut
  416. return this._getAllRolesNamesForRoleIds(resultMap.ids, names, queriedRoles);
  417. }).then(names => {
  418. return Promise.resolve([...new Set(names)]);
  419. });
  420. };
  421. const findUsersWithAuthData = (config, authData) => {
  422. const providers = Object.keys(authData);
  423. const query = providers.reduce((memo, provider) => {
  424. if (!authData[provider] || authData && !authData[provider].id) {
  425. return memo;
  426. }
  427. const queryKey = `authData.${provider}.id`;
  428. const query = {};
  429. query[queryKey] = authData[provider].id;
  430. memo.push(query);
  431. return memo;
  432. }, []).filter(q => {
  433. return typeof q !== 'undefined';
  434. });
  435. return query.length > 0 ? config.database.find('_User', {
  436. $or: query
  437. }, {
  438. limit: 2
  439. }) : Promise.resolve([]);
  440. };
  441. const hasMutatedAuthData = (authData, userAuthData) => {
  442. if (!userAuthData) return {
  443. hasMutatedAuthData: true,
  444. mutatedAuthData: authData
  445. };
  446. const mutatedAuthData = {};
  447. Object.keys(authData).forEach(provider => {
  448. // Anonymous provider is not handled this way
  449. if (provider === 'anonymous') return;
  450. const providerData = authData[provider];
  451. const userProviderAuthData = userAuthData[provider];
  452. if (!(0, _util.isDeepStrictEqual)(providerData, userProviderAuthData)) {
  453. mutatedAuthData[provider] = providerData;
  454. }
  455. });
  456. const hasMutatedAuthData = Object.keys(mutatedAuthData).length !== 0;
  457. return {
  458. hasMutatedAuthData,
  459. mutatedAuthData
  460. };
  461. };
  462. const checkIfUserHasProvidedConfiguredProvidersForLogin = (req = {}, authData = {}, userAuthData = {}, config) => {
  463. const savedUserProviders = Object.keys(userAuthData).map(provider => ({
  464. name: provider,
  465. adapter: config.authDataManager.getValidatorForProvider(provider).adapter
  466. }));
  467. const hasProvidedASoloProvider = savedUserProviders.some(provider => provider && provider.adapter && provider.adapter.policy === 'solo' && authData[provider.name]);
  468. // Solo providers can be considered as safe, so we do not have to check if the user needs
  469. // to provide an additional provider to login. An auth adapter with "solo" (like webauthn) means
  470. // no "additional" auth needs to be provided to login (like OTP, MFA)
  471. if (hasProvidedASoloProvider) {
  472. return;
  473. }
  474. const additionProvidersNotFound = [];
  475. const hasProvidedAtLeastOneAdditionalProvider = savedUserProviders.some(provider => {
  476. let policy = provider.adapter.policy;
  477. if (typeof policy === 'function') {
  478. const requestObject = {
  479. ip: req.config.ip,
  480. user: req.auth.user,
  481. master: req.auth.isMaster
  482. };
  483. policy = policy.call(provider.adapter, requestObject, userAuthData[provider.name]);
  484. }
  485. if (policy === 'additional') {
  486. if (authData[provider.name]) {
  487. return true;
  488. } else {
  489. // Push missing provider for error message
  490. additionProvidersNotFound.push(provider.name);
  491. }
  492. }
  493. });
  494. if (hasProvidedAtLeastOneAdditionalProvider || !additionProvidersNotFound.length) {
  495. return;
  496. }
  497. throw new Parse.Error(Parse.Error.OTHER_CAUSE, `Missing additional authData ${additionProvidersNotFound.join(',')}`);
  498. };
  499. // Validate each authData step-by-step and return the provider responses
  500. const handleAuthDataValidation = async (authData, req, foundUser) => {
  501. let user;
  502. if (foundUser) {
  503. user = Parse.User.fromJSON(_objectSpread({
  504. className: '_User'
  505. }, foundUser));
  506. // Find user by session and current objectId; only pass user if it's the current user or master key is provided
  507. } else if (req.auth && req.auth.user && typeof req.getUserId === 'function' && req.getUserId() === req.auth.user.id || req.auth && req.auth.isMaster && typeof req.getUserId === 'function' && req.getUserId()) {
  508. user = new Parse.User();
  509. user.id = req.auth.isMaster ? req.getUserId() : req.auth.user.id;
  510. await user.fetch({
  511. useMasterKey: true
  512. });
  513. }
  514. const {
  515. updatedObject
  516. } = req.buildParseObjects();
  517. const requestObject = (0, _triggers.getRequestObject)(undefined, req.auth, updatedObject, user, req.config);
  518. // Perform validation as step-by-step pipeline for better error consistency
  519. // and also to avoid to trigger a provider (like OTP SMS) if another one fails
  520. const acc = {
  521. authData: {},
  522. authDataResponse: {}
  523. };
  524. const authKeys = Object.keys(authData).sort();
  525. for (const provider of authKeys) {
  526. let method = '';
  527. try {
  528. if (authData[provider] === null) {
  529. acc.authData[provider] = null;
  530. continue;
  531. }
  532. const {
  533. validator
  534. } = req.config.authDataManager.getValidatorForProvider(provider);
  535. const authProvider = (req.config.auth || {})[provider] || {};
  536. if (!validator || authProvider.enabled === false) {
  537. throw new Parse.Error(Parse.Error.UNSUPPORTED_SERVICE, 'This authentication method is unsupported.');
  538. }
  539. let validationResult = await validator(authData[provider], req, user, requestObject);
  540. method = validationResult && validationResult.method;
  541. requestObject.triggerName = method;
  542. if (validationResult && validationResult.validator) {
  543. validationResult = await validationResult.validator();
  544. }
  545. if (!validationResult) {
  546. acc.authData[provider] = authData[provider];
  547. continue;
  548. }
  549. if (!Object.keys(validationResult).length) {
  550. acc.authData[provider] = authData[provider];
  551. continue;
  552. }
  553. if (validationResult.response) {
  554. acc.authDataResponse[provider] = validationResult.response;
  555. }
  556. // Some auth providers after initialization will avoid to replace authData already stored
  557. if (!validationResult.doNotSave) {
  558. acc.authData[provider] = validationResult.save || authData[provider];
  559. }
  560. } catch (err) {
  561. const e = (0, _triggers.resolveError)(err, {
  562. code: Parse.Error.SCRIPT_FAILED,
  563. message: 'Auth failed. Unknown error.'
  564. });
  565. const userString = req.auth && req.auth.user ? req.auth.user.id : req.data.objectId || undefined;
  566. _logger.logger.error(`Failed running auth step ${method} for ${provider} for user ${userString} with Error: ` + JSON.stringify(e), {
  567. authenticationStep: method,
  568. error: e,
  569. user: userString,
  570. provider
  571. });
  572. throw e;
  573. }
  574. }
  575. return acc;
  576. };
  577. module.exports = {
  578. Auth,
  579. master,
  580. maintenance,
  581. nobody,
  582. readOnly,
  583. shouldUpdateSessionExpiry,
  584. getAuthForSessionToken,
  585. getAuthForLegacySessionToken,
  586. findUsersWithAuthData,
  587. hasMutatedAuthData,
  588. checkIfUserHasProvidedConfiguredProvidersForLogin,
  589. handleAuthDataValidation
  590. };
  591. //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_util","require","_triggers","_logger","_RestQuery","_interopRequireDefault","_RestWrite","e","__esModule","default","ownKeys","r","t","Object","keys","getOwnPropertySymbols","o","filter","getOwnPropertyDescriptor","enumerable","push","apply","_objectSpread","arguments","length","forEach","_defineProperty","getOwnPropertyDescriptors","defineProperties","defineProperty","_toPropertyKey","value","configurable","writable","i","_toPrimitive","Symbol","toPrimitive","call","TypeError","String","Number","Parse","Auth","config","cacheController","undefined","isMaster","isMaintenance","isReadOnly","user","installationId","userRoles","fetchedRoles","rolePromise","prototype","isUnauthenticated","master","maintenance","readOnly","nobody","shouldUpdateSessionExpiry","session","resetAfter","sessionLength","lastUpdated","Date","updatedAt","skipRange","setTime","getTime","throttle","renewSessionIfNeeded","sessionToken","extendSessionOnUse","clearTimeout","setTimeout","query","RestQuery","method","Method","get","auth","runBeforeFind","className","restWhere","restOptions","limit","results","execute","expiresAt","generateSessionExpiresAt","RestWrite","objectId","_encode","code","Error","OBJECT_NOT_FOUND","logger","error","getAuthForSessionToken","userJSON","cachedUser","fromJSON","Promise","resolve","include","Query","Session","equalTo","find","useMasterKey","map","obj","toJSON","INVALID_SESSION_TOKEN","now","iso","startsWith","INTERNAL_SERVER_ERROR","password","put","userObject","getAuthForLegacySessionToken","_session_token","then","response","getUserRoles","_loadRoles","getRolesForUser","users","__type","id","each","result","Role","cachedRoles","role","cacheRoles","rolesMap","reduce","m","names","name","ids","roleNames","_getAllRolesNamesForRoleIds","Array","clearRoleCache","del","getRolesByIds","ins","containedIn","roles","$in","roleIDs","queriedRoles","roleID","wasQueried","Set","resultMap","memo","concat","findUsersWithAuthData","authData","providers","provider","queryKey","q","database","$or","hasMutatedAuthData","userAuthData","mutatedAuthData","providerData","userProviderAuthData","isDeepStrictEqual","checkIfUserHasProvidedConfiguredProvidersForLogin","req","savedUserProviders","adapter","authDataManager","getValidatorForProvider","hasProvidedASoloProvider","some","policy","additionProvidersNotFound","hasProvidedAtLeastOneAdditionalProvider","requestObject","ip","OTHER_CAUSE","join","handleAuthDataValidation","foundUser","User","getUserId","fetch","updatedObject","buildParseObjects","getRequestObject","acc","authDataResponse","authKeys","sort","validator","authProvider","enabled","UNSUPPORTED_SERVICE","validationResult","triggerName","doNotSave","save","err","resolveError","SCRIPT_FAILED","message","userString","data","JSON","stringify","authenticationStep","module","exports"],"sources":["../src/Auth.js"],"sourcesContent":["const Parse = require('parse/node');\nimport { isDeepStrictEqual } from 'util';\nimport { getRequestObject, resolveError } from './triggers';\nimport { logger } from './logger';\nimport RestQuery from './RestQuery';\nimport RestWrite from './RestWrite';\n\n// An Auth object tells you who is requesting something and whether\n// the master key was used.\n// userObject is a Parse.User and can be null if there's no user.\nfunction Auth({\n  config,\n  cacheController = undefined,\n  isMaster = false,\n  isMaintenance = false,\n  isReadOnly = false,\n  user,\n  installationId,\n}) {\n  this.config = config;\n  this.cacheController = cacheController || (config && config.cacheController);\n  this.installationId = installationId;\n  this.isMaster = isMaster;\n  this.isMaintenance = isMaintenance;\n  this.user = user;\n  this.isReadOnly = isReadOnly;\n\n  // Assuming a users roles won't change during a single request, we'll\n  // only load them once.\n  this.userRoles = [];\n  this.fetchedRoles = false;\n  this.rolePromise = null;\n}\n\n// Whether this auth could possibly modify the given user id.\n// It still could be forbidden via ACLs even if this returns true.\nAuth.prototype.isUnauthenticated = function () {\n  if (this.isMaster) {\n    return false;\n  }\n  if (this.isMaintenance) {\n    return false;\n  }\n  if (this.user) {\n    return false;\n  }\n  return true;\n};\n\n// A helper to get a master-level Auth object\nfunction master(config) {\n  return new Auth({ config, isMaster: true });\n}\n\n// A helper to get a maintenance-level Auth object\nfunction maintenance(config) {\n  return new Auth({ config, isMaintenance: true });\n}\n\n// A helper to get a master-level Auth object\nfunction readOnly(config) {\n  return new Auth({ config, isMaster: true, isReadOnly: true });\n}\n\n// A helper to get a nobody-level Auth object\nfunction nobody(config) {\n  return new Auth({ config, isMaster: false });\n}\n\n/**\n * Checks whether session should be updated based on last update time & session length.\n */\nfunction shouldUpdateSessionExpiry(config, session) {\n  const resetAfter = config.sessionLength / 2;\n  const lastUpdated = new Date(session?.updatedAt);\n  const skipRange = new Date();\n  skipRange.setTime(skipRange.getTime() - resetAfter * 1000);\n  return lastUpdated <= skipRange;\n}\n\nconst throttle = {};\nconst renewSessionIfNeeded = async ({ config, session, sessionToken }) => {\n  if (!config?.extendSessionOnUse) {\n    return;\n  }\n  clearTimeout(throttle[sessionToken]);\n  throttle[sessionToken] = setTimeout(async () => {\n    try {\n      if (!session) {\n        const query = await RestQuery({\n          method: RestQuery.Method.get,\n          config,\n          auth: master(config),\n          runBeforeFind: false,\n          className: '_Session',\n          restWhere: { sessionToken },\n          restOptions: { limit: 1 },\n        });\n        const { results } = await query.execute();\n        session = results[0];\n      }\n      if (!shouldUpdateSessionExpiry(config, session) || !session) {\n        return;\n      }\n      const expiresAt = config.generateSessionExpiresAt();\n      await new RestWrite(\n        config,\n        master(config),\n        '_Session',\n        { objectId: session.objectId },\n        { expiresAt: Parse._encode(expiresAt) }\n      ).execute();\n    } catch (e) {\n      if (e?.code !== Parse.Error.OBJECT_NOT_FOUND) {\n        logger.error('Could not update session expiry: ', e);\n      }\n    }\n  }, 500);\n};\n\n// Returns a promise that resolves to an Auth object\nconst getAuthForSessionToken = async function ({\n  config,\n  cacheController,\n  sessionToken,\n  installationId,\n}) {\n  cacheController = cacheController || (config && config.cacheController);\n  if (cacheController) {\n    const userJSON = await cacheController.user.get(sessionToken);\n    if (userJSON) {\n      const cachedUser = Parse.Object.fromJSON(userJSON);\n      renewSessionIfNeeded({ config, sessionToken });\n      return Promise.resolve(\n        new Auth({\n          config,\n          cacheController,\n          isMaster: false,\n          installationId,\n          user: cachedUser,\n        })\n      );\n    }\n  }\n\n  let results;\n  if (config) {\n    const restOptions = {\n      limit: 1,\n      include: 'user',\n    };\n    const RestQuery = require('./RestQuery');\n    const query = await RestQuery({\n      method: RestQuery.Method.get,\n      config,\n      runBeforeFind: false,\n      auth: master(config),\n      className: '_Session',\n      restWhere: { sessionToken },\n      restOptions,\n    });\n    results = (await query.execute()).results;\n  } else {\n    results = (\n      await new Parse.Query(Parse.Session)\n        .limit(1)\n        .include('user')\n        .equalTo('sessionToken', sessionToken)\n        .find({ useMasterKey: true })\n    ).map(obj => obj.toJSON());\n  }\n\n  if (results.length !== 1 || !results[0]['user']) {\n    throw new Parse.Error(Parse.Error.INVALID_SESSION_TOKEN, 'Invalid session token');\n  }\n  const session = results[0];\n  const now = new Date(),\n    expiresAt = session.expiresAt ? new Date(session.expiresAt.iso) : undefined;\n  if (expiresAt < now) {\n    throw new Parse.Error(Parse.Error.INVALID_SESSION_TOKEN, 'Session token is expired.');\n  }\n  const obj = session.user;\n\n  if (typeof obj['objectId'] === 'string' && obj['objectId'].startsWith('role:')) {\n    throw new Parse.Error(Parse.Error.INTERNAL_SERVER_ERROR, 'Invalid object ID.');\n  }\n\n  delete obj.password;\n  obj['className'] = '_User';\n  obj['sessionToken'] = sessionToken;\n  if (cacheController) {\n    cacheController.user.put(sessionToken, obj);\n  }\n  renewSessionIfNeeded({ config, session, sessionToken });\n  const userObject = Parse.Object.fromJSON(obj);\n  return new Auth({\n    config,\n    cacheController,\n    isMaster: false,\n    installationId,\n    user: userObject,\n  });\n};\n\nvar getAuthForLegacySessionToken = async function ({ config, sessionToken, installationId }) {\n  var restOptions = {\n    limit: 1,\n  };\n  const RestQuery = require('./RestQuery');\n  var query = await RestQuery({\n    method: RestQuery.Method.get,\n    config,\n    runBeforeFind: false,\n    auth: master(config),\n    className: '_User',\n    restWhere: { _session_token: sessionToken },\n    restOptions,\n  });\n  return query.execute().then(response => {\n    var results = response.results;\n    if (results.length !== 1) {\n      throw new Parse.Error(Parse.Error.INVALID_SESSION_TOKEN, 'invalid legacy session token');\n    }\n    const obj = results[0];\n    obj.className = '_User';\n    const userObject = Parse.Object.fromJSON(obj);\n    return new Auth({\n      config,\n      isMaster: false,\n      installationId,\n      user: userObject,\n    });\n  });\n};\n\n// Returns a promise that resolves to an array of role names\nAuth.prototype.getUserRoles = function () {\n  if (this.isMaster || this.isMaintenance || !this.user) {\n    return Promise.resolve([]);\n  }\n  if (this.fetchedRoles) {\n    return Promise.resolve(this.userRoles);\n  }\n  if (this.rolePromise) {\n    return this.rolePromise;\n  }\n  this.rolePromise = this._loadRoles();\n  return this.rolePromise;\n};\n\nAuth.prototype.getRolesForUser = async function () {\n  //Stack all Parse.Role\n  const results = [];\n  if (this.config) {\n    const restWhere = {\n      users: {\n        __type: 'Pointer',\n        className: '_User',\n        objectId: this.user.id,\n      },\n    };\n    const RestQuery = require('./RestQuery');\n    const query = await RestQuery({\n      method: RestQuery.Method.find,\n      runBeforeFind: false,\n      config: this.config,\n      auth: master(this.config),\n      className: '_Role',\n      restWhere,\n    });\n    await query.each(result => results.push(result));\n  } else {\n    await new Parse.Query(Parse.Role)\n      .equalTo('users', this.user)\n      .each(result => results.push(result.toJSON()), { useMasterKey: true });\n  }\n  return results;\n};\n\n// Iterates through the role tree and compiles a user's roles\nAuth.prototype._loadRoles = async function () {\n  if (this.cacheController) {\n    const cachedRoles = await this.cacheController.role.get(this.user.id);\n    if (cachedRoles != null) {\n      this.fetchedRoles = true;\n      this.userRoles = cachedRoles;\n      return cachedRoles;\n    }\n  }\n\n  // First get the role ids this user is directly a member of\n  const results = await this.getRolesForUser();\n  if (!results.length) {\n    this.userRoles = [];\n    this.fetchedRoles = true;\n    this.rolePromise = null;\n\n    this.cacheRoles();\n    return this.userRoles;\n  }\n\n  const rolesMap = results.reduce(\n    (m, r) => {\n      m.names.push(r.name);\n      m.ids.push(r.objectId);\n      return m;\n    },\n    { ids: [], names: [] }\n  );\n\n  // run the recursive finding\n  const roleNames = await this._getAllRolesNamesForRoleIds(rolesMap.ids, rolesMap.names);\n  this.userRoles = roleNames.map(r => {\n    return 'role:' + r;\n  });\n  this.fetchedRoles = true;\n  this.rolePromise = null;\n  this.cacheRoles();\n  return this.userRoles;\n};\n\nAuth.prototype.cacheRoles = function () {\n  if (!this.cacheController) {\n    return false;\n  }\n  this.cacheController.role.put(this.user.id, Array(...this.userRoles));\n  return true;\n};\n\nAuth.prototype.clearRoleCache = function (sessionToken) {\n  if (!this.cacheController) {\n    return false;\n  }\n  this.cacheController.role.del(this.user.id);\n  this.cacheController.user.del(sessionToken);\n  return true;\n};\n\nAuth.prototype.getRolesByIds = async function (ins) {\n  const results = [];\n  // Build an OR query across all parentRoles\n  if (!this.config) {\n    await new Parse.Query(Parse.Role)\n      .containedIn(\n        'roles',\n        ins.map(id => {\n          const role = new Parse.Object(Parse.Role);\n          role.id = id;\n          return role;\n        })\n      )\n      .each(result => results.push(result.toJSON()), { useMasterKey: true });\n  } else {\n    const roles = ins.map(id => {\n      return {\n        __type: 'Pointer',\n        className: '_Role',\n        objectId: id,\n      };\n    });\n    const restWhere = { roles: { $in: roles } };\n    const RestQuery = require('./RestQuery');\n    const query = await RestQuery({\n      method: RestQuery.Method.find,\n      config: this.config,\n      runBeforeFind: false,\n      auth: master(this.config),\n      className: '_Role',\n      restWhere,\n    });\n    await query.each(result => results.push(result));\n  }\n  return results;\n};\n\n// Given a list of roleIds, find all the parent roles, returns a promise with all names\nAuth.prototype._getAllRolesNamesForRoleIds = function (roleIDs, names = [], queriedRoles = {}) {\n  const ins = roleIDs.filter(roleID => {\n    const wasQueried = queriedRoles[roleID] !== true;\n    queriedRoles[roleID] = true;\n    return wasQueried;\n  });\n\n  // all roles are accounted for, return the names\n  if (ins.length == 0) {\n    return Promise.resolve([...new Set(names)]);\n  }\n\n  return this.getRolesByIds(ins)\n    .then(results => {\n      // Nothing found\n      if (!results.length) {\n        return Promise.resolve(names);\n      }\n      // Map the results with all Ids and names\n      const resultMap = results.reduce(\n        (memo, role) => {\n          memo.names.push(role.name);\n          memo.ids.push(role.objectId);\n          return memo;\n        },\n        { ids: [], names: [] }\n      );\n      // store the new found names\n      names = names.concat(resultMap.names);\n      // find the next ones, circular roles will be cut\n      return this._getAllRolesNamesForRoleIds(resultMap.ids, names, queriedRoles);\n    })\n    .then(names => {\n      return Promise.resolve([...new Set(names)]);\n    });\n};\n\nconst findUsersWithAuthData = (config, authData) => {\n  const providers = Object.keys(authData);\n  const query = providers\n    .reduce((memo, provider) => {\n      if (!authData[provider] || (authData && !authData[provider].id)) {\n        return memo;\n      }\n      const queryKey = `authData.${provider}.id`;\n      const query = {};\n      query[queryKey] = authData[provider].id;\n      memo.push(query);\n      return memo;\n    }, [])\n    .filter(q => {\n      return typeof q !== 'undefined';\n    });\n\n  return query.length > 0\n    ? config.database.find('_User', { $or: query }, { limit: 2 })\n    : Promise.resolve([]);\n};\n\nconst hasMutatedAuthData = (authData, userAuthData) => {\n  if (!userAuthData) return { hasMutatedAuthData: true, mutatedAuthData: authData };\n  const mutatedAuthData = {};\n  Object.keys(authData).forEach(provider => {\n    // Anonymous provider is not handled this way\n    if (provider === 'anonymous') return;\n    const providerData = authData[provider];\n    const userProviderAuthData = userAuthData[provider];\n    if (!isDeepStrictEqual(providerData, userProviderAuthData)) {\n      mutatedAuthData[provider] = providerData;\n    }\n  });\n  const hasMutatedAuthData = Object.keys(mutatedAuthData).length !== 0;\n  return { hasMutatedAuthData, mutatedAuthData };\n};\n\nconst checkIfUserHasProvidedConfiguredProvidersForLogin = (\n  req = {},\n  authData = {},\n  userAuthData = {},\n  config\n) => {\n  const savedUserProviders = Object.keys(userAuthData).map(provider => ({\n    name: provider,\n    adapter: config.authDataManager.getValidatorForProvider(provider).adapter,\n  }));\n\n  const hasProvidedASoloProvider = savedUserProviders.some(\n    provider =>\n      provider && provider.adapter && provider.adapter.policy === 'solo' && authData[provider.name]\n  );\n\n  // Solo providers can be considered as safe, so we do not have to check if the user needs\n  // to provide an additional provider to login. An auth adapter with \"solo\" (like webauthn) means\n  // no \"additional\" auth needs to be provided to login (like OTP, MFA)\n  if (hasProvidedASoloProvider) {\n    return;\n  }\n\n  const additionProvidersNotFound = [];\n  const hasProvidedAtLeastOneAdditionalProvider = savedUserProviders.some(provider => {\n    let policy = provider.adapter.policy;\n    if (typeof policy === 'function') {\n      const requestObject = {\n        ip: req.config.ip,\n        user: req.auth.user,\n        master: req.auth.isMaster,\n      };\n      policy = policy.call(provider.adapter, requestObject, userAuthData[provider.name]);\n    }\n    if (policy === 'additional') {\n      if (authData[provider.name]) {\n        return true;\n      } else {\n        // Push missing provider for error message\n        additionProvidersNotFound.push(provider.name);\n      }\n    }\n  });\n  if (hasProvidedAtLeastOneAdditionalProvider || !additionProvidersNotFound.length) {\n    return;\n  }\n\n  throw new Parse.Error(\n    Parse.Error.OTHER_CAUSE,\n    `Missing additional authData ${additionProvidersNotFound.join(',')}`\n  );\n};\n\n// Validate each authData step-by-step and return the provider responses\nconst handleAuthDataValidation = async (authData, req, foundUser) => {\n  let user;\n  if (foundUser) {\n    user = Parse.User.fromJSON({ className: '_User', ...foundUser });\n    // Find user by session and current objectId; only pass user if it's the current user or master key is provided\n  } else if (\n    (req.auth &&\n      req.auth.user &&\n      typeof req.getUserId === 'function' &&\n      req.getUserId() === req.auth.user.id) ||\n    (req.auth && req.auth.isMaster && typeof req.getUserId === 'function' && req.getUserId())\n  ) {\n    user = new Parse.User();\n    user.id = req.auth.isMaster ? req.getUserId() : req.auth.user.id;\n    await user.fetch({ useMasterKey: true });\n  }\n\n  const { updatedObject } = req.buildParseObjects();\n  const requestObject = getRequestObject(undefined, req.auth, updatedObject, user, req.config);\n  // Perform validation as step-by-step pipeline for better error consistency\n  // and also to avoid to trigger a provider (like OTP SMS) if another one fails\n  const acc = { authData: {}, authDataResponse: {} };\n  const authKeys = Object.keys(authData).sort();\n  for (const provider of authKeys) {\n    let method = '';\n    try {\n      if (authData[provider] === null) {\n        acc.authData[provider] = null;\n        continue;\n      }\n      const { validator } = req.config.authDataManager.getValidatorForProvider(provider);\n      const authProvider = (req.config.auth || {})[provider] || {};\n      if (!validator || authProvider.enabled === false) {\n        throw new Parse.Error(\n          Parse.Error.UNSUPPORTED_SERVICE,\n          'This authentication method is unsupported.'\n        );\n      }\n      let validationResult = await validator(authData[provider], req, user, requestObject);\n      method = validationResult && validationResult.method;\n      requestObject.triggerName = method;\n      if (validationResult && validationResult.validator) {\n        validationResult = await validationResult.validator();\n      }\n      if (!validationResult) {\n        acc.authData[provider] = authData[provider];\n        continue;\n      }\n      if (!Object.keys(validationResult).length) {\n        acc.authData[provider] = authData[provider];\n        continue;\n      }\n\n      if (validationResult.response) {\n        acc.authDataResponse[provider] = validationResult.response;\n      }\n      // Some auth providers after initialization will avoid to replace authData already stored\n      if (!validationResult.doNotSave) {\n        acc.authData[provider] = validationResult.save || authData[provider];\n      }\n    } catch (err) {\n      const e = resolveError(err, {\n        code: Parse.Error.SCRIPT_FAILED,\n        message: 'Auth failed. Unknown error.',\n      });\n      const userString =\n        req.auth && req.auth.user ? req.auth.user.id : req.data.objectId || undefined;\n      logger.error(\n        `Failed running auth step ${method} for ${provider} for user ${userString} with Error: ` +\n          JSON.stringify(e),\n        {\n          authenticationStep: method,\n          error: e,\n          user: userString,\n          provider,\n        }\n      );\n      throw e;\n    }\n  }\n  return acc;\n};\n\nmodule.exports = {\n  Auth,\n  master,\n  maintenance,\n  nobody,\n  readOnly,\n  shouldUpdateSessionExpiry,\n  getAuthForSessionToken,\n  getAuthForLegacySessionToken,\n  findUsersWithAuthData,\n  hasMutatedAuthData,\n  checkIfUserHasProvidedConfiguredProvidersForLogin,\n  handleAuthDataValidation,\n};\n"],"mappings":"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","ignoreList":[]}