BaseSubscriptionTokensController.ts 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111
  1. import { ControllerContainerInterface, SettingName } from '@standardnotes/domain-core'
  2. import { ErrorTag } from '@standardnotes/responses'
  3. import { Role, TokenEncoderInterface, CrossServiceTokenData } from '@standardnotes/security'
  4. import { BaseHttpController, results } from 'inversify-express-utils'
  5. import { Request, Response } from 'express'
  6. import { AuthenticateSubscriptionToken } from '../../../Domain/UseCase/AuthenticateSubscriptionToken/AuthenticateSubscriptionToken'
  7. import { CreateSubscriptionToken } from '../../../Domain/UseCase/CreateSubscriptionToken/CreateSubscriptionToken'
  8. import { ProjectorInterface } from '../../../Projection/ProjectorInterface'
  9. import { User } from '../../../Domain/User/User'
  10. import { GetSetting } from '../../../Domain/UseCase/GetSetting/GetSetting'
  11. import { ResponseLocals } from '../ResponseLocals'
  12. export class BaseSubscriptionTokensController extends BaseHttpController {
  13. constructor(
  14. protected createSubscriptionToken: CreateSubscriptionToken,
  15. protected authenticateToken: AuthenticateSubscriptionToken,
  16. protected getSetting: GetSetting,
  17. protected userProjector: ProjectorInterface<User>,
  18. protected roleProjector: ProjectorInterface<Role>,
  19. protected tokenEncoder: TokenEncoderInterface<CrossServiceTokenData>,
  20. protected jwtTTL: number,
  21. private controllerContainer?: ControllerContainerInterface,
  22. ) {
  23. super()
  24. if (this.controllerContainer !== undefined) {
  25. this.controllerContainer.register('auth.subscription-tokens.create', this.createToken.bind(this))
  26. }
  27. }
  28. async createToken(_request: Request, response: Response): Promise<results.JsonResult> {
  29. const locals = response.locals as ResponseLocals
  30. if (locals.readOnlyAccess) {
  31. return this.json(
  32. {
  33. error: {
  34. tag: ErrorTag.ReadOnlyAccess,
  35. message: 'Session has read-only access.',
  36. },
  37. },
  38. 401,
  39. )
  40. }
  41. const result = await this.createSubscriptionToken.execute({
  42. userUuid: locals.user.uuid,
  43. })
  44. return this.json({
  45. token: result.subscriptionToken.token,
  46. })
  47. }
  48. async validate(request: Request): Promise<results.JsonResult> {
  49. const authenticateTokenResponse = await this.authenticateToken.execute({
  50. token: request.params.token,
  51. })
  52. if (!authenticateTokenResponse.success) {
  53. return this.json(
  54. {
  55. error: {
  56. tag: 'invalid-auth',
  57. message: 'Invalid login credentials.',
  58. },
  59. },
  60. 401,
  61. )
  62. }
  63. const user = authenticateTokenResponse.user as User
  64. let extensionKey = undefined
  65. const extensionKeySettingOrError = await this.getSetting.execute({
  66. settingName: SettingName.NAMES.ExtensionKey,
  67. userUuid: user.uuid,
  68. allowSensitiveRetrieval: true,
  69. decrypted: true,
  70. })
  71. if (!extensionKeySettingOrError.isFailed()) {
  72. const extensionKeySetting = extensionKeySettingOrError.getValue()
  73. extensionKey = extensionKeySetting.decryptedValue as string
  74. }
  75. const roles = await user.roles
  76. const authTokenData: CrossServiceTokenData = {
  77. user: await this.projectUser(user),
  78. roles: await this.projectRoles(roles),
  79. extensionKey,
  80. }
  81. const authToken = this.tokenEncoder.encodeExpirableToken(authTokenData, this.jwtTTL)
  82. return this.json({ authToken })
  83. }
  84. private async projectUser(user: User): Promise<{ uuid: string; email: string }> {
  85. return <{ uuid: string; email: string }>await this.userProjector.projectSimple(user)
  86. }
  87. private async projectRoles(roles: Array<Role>): Promise<Array<{ uuid: string; name: string }>> {
  88. const roleProjections = []
  89. for (const role of roles) {
  90. roleProjections.push(<{ uuid: string; name: string }>await this.roleProjector.projectSimple(role))
  91. }
  92. return roleProjections
  93. }
  94. }