summaryrefslogtreecommitdiff
path: root/packages/backend/src/server/api/endpoints/admin/invite
diff options
context:
space:
mode:
authorsyuilo <Syuilotan@yahoo.co.jp>2023-07-21 20:36:07 +0900
committerGitHub <noreply@github.com>2023-07-21 20:36:07 +0900
commite64a81aa1d2801516e8eac8dc69aac540489f20b (patch)
tree56accbc0f5f71db864e1e975920135fb0a957291 /packages/backend/src/server/api/endpoints/admin/invite
parentMerge pull request #10990 from misskey-dev/develop (diff)
parentNew Crowdin updates (#11336) (diff)
downloadmisskey-e64a81aa1d2801516e8eac8dc69aac540489f20b.tar.gz
misskey-e64a81aa1d2801516e8eac8dc69aac540489f20b.tar.bz2
misskey-e64a81aa1d2801516e8eac8dc69aac540489f20b.zip
Merge pull request #11301 from misskey-dev/develop
Release: 13.14.0
Diffstat (limited to 'packages/backend/src/server/api/endpoints/admin/invite')
-rw-r--r--packages/backend/src/server/api/endpoints/admin/invite/create.ts80
-rw-r--r--packages/backend/src/server/api/endpoints/admin/invite/list.ts70
2 files changed, 150 insertions, 0 deletions
diff --git a/packages/backend/src/server/api/endpoints/admin/invite/create.ts b/packages/backend/src/server/api/endpoints/admin/invite/create.ts
new file mode 100644
index 0000000000..664b4d819f
--- /dev/null
+++ b/packages/backend/src/server/api/endpoints/admin/invite/create.ts
@@ -0,0 +1,80 @@
+import { Inject, Injectable } from '@nestjs/common';
+import { Endpoint } from '@/server/api/endpoint-base.js';
+import type { RegistrationTicketsRepository } from '@/models/index.js';
+import { InviteCodeEntityService } from '@/core/entities/InviteCodeEntityService.js';
+import { IdService } from '@/core/IdService.js';
+import { DI } from '@/di-symbols.js';
+import { generateInviteCode } from '@/misc/generate-invite-code.js';
+import { ApiError } from '../../../error.js';
+
+export const meta = {
+ tags: ['admin'],
+
+ requireCredential: true,
+ requireModerator: true,
+
+ errors: {
+ invalidDateTime: {
+ message: 'Invalid date-time format',
+ code: 'INVALID_DATE_TIME',
+ id: 'f1380b15-3760-4c6c-a1db-5c3aaf1cbd49',
+ },
+ },
+
+ res: {
+ type: 'array',
+ optional: false, nullable: false,
+ items: {
+ type: 'object',
+ optional: false, nullable: false,
+ properties: {
+ code: {
+ type: 'string',
+ optional: false, nullable: false,
+ example: 'GR6S02ERUA5VR',
+ },
+ },
+ },
+ },
+} as const;
+
+export const paramDef = {
+ type: 'object',
+ properties: {
+ count: { type: 'integer', minimum: 1, maximum: 100, default: 1 },
+ expiresAt: { type: 'string', nullable: true },
+ },
+ required: [],
+} as const;
+
+// eslint-disable-next-line import/no-default-export
+@Injectable()
+export default class extends Endpoint<typeof meta, typeof paramDef> {
+ constructor(
+ @Inject(DI.registrationTicketsRepository)
+ private registrationTicketsRepository: RegistrationTicketsRepository,
+
+ private inviteCodeEntityService: InviteCodeEntityService,
+ private idService: IdService,
+ ) {
+ super(meta, paramDef, async (ps, me) => {
+ if (ps.expiresAt && isNaN(Date.parse(ps.expiresAt))) {
+ throw new ApiError(meta.errors.invalidDateTime);
+ }
+
+ const ticketsPromises = [];
+
+ for (let i = 0; i < ps.count; i++) {
+ ticketsPromises.push(this.registrationTicketsRepository.insert({
+ id: this.idService.genId(),
+ createdAt: new Date(),
+ expiresAt: ps.expiresAt ? new Date(ps.expiresAt) : null,
+ code: generateInviteCode(),
+ }).then(x => this.registrationTicketsRepository.findOneByOrFail(x.identifiers[0])));
+ }
+
+ const tickets = await Promise.all(ticketsPromises);
+ return await this.inviteCodeEntityService.packMany(tickets, me);
+ });
+ }
+}
diff --git a/packages/backend/src/server/api/endpoints/admin/invite/list.ts b/packages/backend/src/server/api/endpoints/admin/invite/list.ts
new file mode 100644
index 0000000000..5d7a7f632c
--- /dev/null
+++ b/packages/backend/src/server/api/endpoints/admin/invite/list.ts
@@ -0,0 +1,70 @@
+import { Inject, Injectable } from '@nestjs/common';
+import { Endpoint } from '@/server/api/endpoint-base.js';
+import type { RegistrationTicketsRepository } from '@/models/index.js';
+import { InviteCodeEntityService } from '@/core/entities/InviteCodeEntityService.js';
+import { DI } from '@/di-symbols.js';
+
+export const meta = {
+ tags: ['admin'],
+
+ requireCredential: true,
+ requireModerator: true,
+
+ res: {
+ type: 'array',
+ optional: false, nullable: false,
+ items: {
+ type: 'object',
+ optional: false, nullable: false,
+ },
+ },
+} as const;
+
+export const paramDef = {
+ type: 'object',
+ properties: {
+ limit: { type: 'integer', minimum: 1, maximum: 100, default: 30 },
+ offset: { type: 'integer', default: 0 },
+ type: { type: 'string', enum: ['unused', 'used', 'expired', 'all'], default: 'all' },
+ sort: { type: 'string', enum: ['+createdAt', '-createdAt', '+usedAt', '-usedAt'] },
+ },
+ required: [],
+} as const;
+
+// eslint-disable-next-line import/no-default-export
+@Injectable()
+export default class extends Endpoint<typeof meta, typeof paramDef> {
+ constructor(
+ @Inject(DI.registrationTicketsRepository)
+ private registrationTicketsRepository: RegistrationTicketsRepository,
+
+ private inviteCodeEntityService: InviteCodeEntityService,
+ ) {
+ super(meta, paramDef, async (ps, me) => {
+ const query = this.registrationTicketsRepository.createQueryBuilder('ticket')
+ .leftJoinAndSelect('ticket.createdBy', 'createdBy')
+ .leftJoinAndSelect('ticket.usedBy', 'usedBy');
+
+ switch (ps.type) {
+ case 'unused': query.andWhere('ticket.usedBy IS NULL'); break;
+ case 'used': query.andWhere('ticket.usedBy IS NOT NULL'); break;
+ case 'expired': query.andWhere('ticket.expiresAt < :now', { now: new Date() }); break;
+ }
+
+ switch (ps.sort) {
+ case '+createdAt': query.orderBy('ticket.createdAt', 'DESC'); break;
+ case '-createdAt': query.orderBy('ticket.createdAt', 'ASC'); break;
+ case '+usedAt': query.orderBy('ticket.usedAt', 'DESC', 'NULLS LAST'); break;
+ case '-usedAt': query.orderBy('ticket.usedAt', 'ASC', 'NULLS FIRST'); break;
+ default: query.orderBy('ticket.id', 'DESC'); break;
+ }
+
+ query.limit(ps.limit);
+ query.skip(ps.offset);
+
+ const tickets = await query.getMany();
+
+ return await this.inviteCodeEntityService.packMany(tickets, me);
+ });
+ }
+}