/* * SPDX-FileCopyrightText: syuilo and other misskey contributors * SPDX-License-Identifier: AGPL-3.0-only */ import { Inject, Injectable } from '@nestjs/common'; import { Endpoint } from '@/server/api/endpoint-base.js'; import type { AppsRepository } from '@/models/index.js'; import { IdService } from '@/core/IdService.js'; import { unique } from '@/misc/prelude/array.js'; import { secureRndstr } from '@/misc/secure-rndstr.js'; import { AppEntityService } from '@/core/entities/AppEntityService.js'; import { DI } from '@/di-symbols.js'; export const meta = { tags: ['app'], requireCredential: false, res: { type: 'object', optional: false, nullable: false, ref: 'App', }, } as const; export const paramDef = { type: 'object', properties: { name: { type: 'string' }, description: { type: 'string' }, permission: { type: 'array', uniqueItems: true, items: { type: 'string', } }, callbackUrl: { type: 'string', nullable: true }, }, required: ['name', 'description', 'permission'], } as const; // eslint-disable-next-line import/no-default-export @Injectable() export default class extends Endpoint { constructor( @Inject(DI.appsRepository) private appsRepository: AppsRepository, private appEntityService: AppEntityService, private idService: IdService, ) { super(meta, paramDef, async (ps, me) => { // Generate secret const secret = secureRndstr(32); // for backward compatibility const permission = unique(ps.permission.map(v => v.replace(/^(.+)(\/|-)(read|write)$/, '$3:$1'))); // Create account const app = await this.appsRepository.insert({ id: this.idService.genId(), createdAt: new Date(), userId: me ? me.id : null, name: ps.name, description: ps.description, permission, callbackUrl: ps.callbackUrl, secret: secret, }).then(x => this.appsRepository.findOneByOrFail(x.identifiers[0])); return await this.appEntityService.pack(app, null, { detail: true, includeSecret: true, }); }); } }