api-token.ts 1.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051
  1. import type { IUser, IUserHasId } from '@growi/core/dist/interfaces';
  2. import type { AccessTokenParserReq } from '@growi/core/dist/interfaces/server';
  3. import { serializeUserSecurely } from '@growi/core/dist/models/serializers';
  4. import type { Response } from 'express';
  5. import type { HydratedDocument } from 'mongoose';
  6. import mongoose from 'mongoose';
  7. import loggerFactory from '~/utils/logger';
  8. import { extractBearerToken } from './extract-bearer-token';
  9. const logger = loggerFactory('growi:middleware:access-token-parser:api-token');
  10. export const parserForApiToken = async (
  11. req: AccessTokenParserReq,
  12. res: Response,
  13. ): Promise<void> => {
  14. // Extract token from Authorization header first
  15. // It is more efficient to call it only once in "AccessTokenParser," which is the caller of the method
  16. const bearerToken = extractBearerToken(req.headers.authorization);
  17. // Try all possible token sources in order of priority
  18. const accessToken =
  19. bearerToken ?? req.query.access_token ?? req.body.access_token;
  20. if (accessToken == null || typeof accessToken !== 'string') {
  21. return;
  22. }
  23. logger.debug(
  24. { accessToken: `${accessToken.slice(0, 4)}...${accessToken.slice(-4)}` },
  25. 'accessToken is',
  26. );
  27. const User = mongoose.model<HydratedDocument<IUser>, { findUserByApiToken }>(
  28. 'User',
  29. );
  30. const userByApiToken: IUserHasId = await User.findUserByApiToken(accessToken);
  31. if (userByApiToken == null) {
  32. return;
  33. }
  34. req.user = serializeUserSecurely(userByApiToken);
  35. if (req.user == null) {
  36. return;
  37. }
  38. logger.debug('Access token parsed.');
  39. return;
  40. };