api-token.ts 1.5 KB

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