import { Inject, Injectable } from '@nestjs/common'; import { Endpoint } from '@/server/api/endpoint-base.js'; import type { UsersRepository, AppsRepository, AccessTokensRepository, AuthSessionsRepository } from '@/models/index.js'; import { UserEntityService } from '@/core/entities/UserEntityService.js'; import { DI } from '@/di-symbols.js'; import { ApiError } from '../../../error.js'; export const meta = { tags: ['auth'], requireCredential: false, res: { type: 'object', optional: false, nullable: false, properties: { accessToken: { type: 'string', optional: false, nullable: false, }, user: { type: 'object', optional: false, nullable: false, ref: 'UserDetailedNotMe', }, }, }, errors: { noSuchApp: { message: 'No such app.', code: 'NO_SUCH_APP', id: 'fcab192a-2c5a-43b7-8ad8-9b7054d8d40d', }, noSuchSession: { message: 'No such session.', code: 'NO_SUCH_SESSION', id: '5b5a1503-8bc8-4bd0-8054-dc189e8cdcb3', }, pendingSession: { message: 'This session is not completed yet.', code: 'PENDING_SESSION', id: '8c8a4145-02cc-4cca-8e66-29ba60445a8e', }, }, } as const; export const paramDef = { type: 'object', properties: { appSecret: { type: 'string' }, token: { type: 'string' }, }, required: ['appSecret', 'token'], } as const; // eslint-disable-next-line import/no-default-export @Injectable() export default class extends Endpoint { constructor( @Inject(DI.usersRepository) private usersRepository: UsersRepository, @Inject(DI.appsRepository) private appsRepository: AppsRepository, @Inject(DI.authSessionsRepository) private authSessionsRepository: AuthSessionsRepository, @Inject(DI.accessTokensRepository) private accessTokensRepository: AccessTokensRepository, private userEntityService: UserEntityService, ) { super(meta, paramDef, async (ps, me) => { // Lookup app const app = await this.appsRepository.findOneBy({ secret: ps.appSecret, }); if (app == null) { throw new ApiError(meta.errors.noSuchApp); } // Fetch token const session = await this.authSessionsRepository.findOneBy({ token: ps.token, appId: app.id, }); if (session == null) { throw new ApiError(meta.errors.noSuchSession); } if (session.userId == null) { throw new ApiError(meta.errors.pendingSession); } // Lookup access token const accessToken = await this.accessTokensRepository.findOneByOrFail({ appId: app.id, userId: session.userId, }); // Delete session this.authSessionsRepository.delete(session.id); return { accessToken: accessToken.token, user: await this.userEntityService.pack(session.userId, me, { detail: true, }), }; }); } }