api-token.ts 1.6 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253
  1. import type { IUser, IUserHasId } from '@growi/core/dist/interfaces';
  2. import { serializeUserSecurely } from '@growi/core/dist/models/serializers';
  3. import type { NextFunction, Response } from 'express';
  4. import type { HydratedDocument } from 'mongoose';
  5. import mongoose from 'mongoose';
  6. import loggerFactory from '~/utils/logger';
  7. import type { AccessTokenParserReq } from './interfaces';
  8. const logger = loggerFactory('growi:middleware:access-token-parser:api-token');
  9. const extractBearerToken = (authHeader: string | undefined): string | null => {
  10. if (authHeader == null) {
  11. return null;
  12. }
  13. if (!authHeader.startsWith('Bearer ')) {
  14. return null;
  15. }
  16. return authHeader.substring(7); // Remove 'Bearer ' prefix
  17. };
  18. export const parserForApiToken = async(req: AccessTokenParserReq, res: Response): Promise<void> => {
  19. // Extract token from Authorization header first
  20. const bearerToken = extractBearerToken(req.headers.authorization);
  21. // Try all possible token sources in order of priority
  22. const accessToken = bearerToken ?? req.query.access_token ?? req.body.access_token;
  23. if (accessToken == null || typeof accessToken !== 'string') {
  24. return;
  25. }
  26. logger.debug('accessToken is', accessToken);
  27. const User = mongoose.model<HydratedDocument<IUser>, { findUserByApiToken }>('User');
  28. const userByApiToken: IUserHasId = await User.findUserByApiToken(accessToken);
  29. if (userByApiToken == null) {
  30. return;
  31. }
  32. req.user = serializeUserSecurely(userByApiToken);
  33. if (req.user == null) {
  34. return;
  35. }
  36. logger.debug('Access token parsed.');
  37. return;
  38. };