123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111 |
- import { ControllerContainerInterface, SettingName } from '@standardnotes/domain-core'
- import { ErrorTag } from '@standardnotes/responses'
- import { Role, TokenEncoderInterface, CrossServiceTokenData } from '@standardnotes/security'
- import { BaseHttpController, results } from 'inversify-express-utils'
- import { Request, Response } from 'express'
- import { AuthenticateSubscriptionToken } from '../../../Domain/UseCase/AuthenticateSubscriptionToken/AuthenticateSubscriptionToken'
- import { CreateSubscriptionToken } from '../../../Domain/UseCase/CreateSubscriptionToken/CreateSubscriptionToken'
- import { ProjectorInterface } from '../../../Projection/ProjectorInterface'
- import { User } from '../../../Domain/User/User'
- import { GetSetting } from '../../../Domain/UseCase/GetSetting/GetSetting'
- import { ResponseLocals } from '../ResponseLocals'
- export class BaseSubscriptionTokensController extends BaseHttpController {
- constructor(
- protected createSubscriptionToken: CreateSubscriptionToken,
- protected authenticateToken: AuthenticateSubscriptionToken,
- protected getSetting: GetSetting,
- protected userProjector: ProjectorInterface<User>,
- protected roleProjector: ProjectorInterface<Role>,
- protected tokenEncoder: TokenEncoderInterface<CrossServiceTokenData>,
- protected jwtTTL: number,
- private controllerContainer?: ControllerContainerInterface,
- ) {
- super()
- if (this.controllerContainer !== undefined) {
- this.controllerContainer.register('auth.subscription-tokens.create', this.createToken.bind(this))
- }
- }
- async createToken(_request: Request, response: Response): Promise<results.JsonResult> {
- const locals = response.locals as ResponseLocals
- if (locals.readOnlyAccess) {
- return this.json(
- {
- error: {
- tag: ErrorTag.ReadOnlyAccess,
- message: 'Session has read-only access.',
- },
- },
- 401,
- )
- }
- const result = await this.createSubscriptionToken.execute({
- userUuid: locals.user.uuid,
- })
- return this.json({
- token: result.subscriptionToken.token,
- })
- }
- async validate(request: Request): Promise<results.JsonResult> {
- const authenticateTokenResponse = await this.authenticateToken.execute({
- token: request.params.token,
- })
- if (!authenticateTokenResponse.success) {
- return this.json(
- {
- error: {
- tag: 'invalid-auth',
- message: 'Invalid login credentials.',
- },
- },
- 401,
- )
- }
- const user = authenticateTokenResponse.user as User
- let extensionKey = undefined
- const extensionKeySettingOrError = await this.getSetting.execute({
- settingName: SettingName.NAMES.ExtensionKey,
- userUuid: user.uuid,
- allowSensitiveRetrieval: true,
- decrypted: true,
- })
- if (!extensionKeySettingOrError.isFailed()) {
- const extensionKeySetting = extensionKeySettingOrError.getValue()
- extensionKey = extensionKeySetting.decryptedValue as string
- }
- const roles = await user.roles
- const authTokenData: CrossServiceTokenData = {
- user: await this.projectUser(user),
- roles: await this.projectRoles(roles),
- extensionKey,
- }
- const authToken = this.tokenEncoder.encodeExpirableToken(authTokenData, this.jwtTTL)
- return this.json({ authToken })
- }
- private async projectUser(user: User): Promise<{ uuid: string; email: string }> {
- return <{ uuid: string; email: string }>await this.userProjector.projectSimple(user)
- }
- private async projectRoles(roles: Array<Role>): Promise<Array<{ uuid: string; name: string }>> {
- const roleProjections = []
- for (const role of roles) {
- roleProjections.push(<{ uuid: string; name: string }>await this.roleProjector.projectSimple(role))
- }
- return roleProjections
- }
- }
|