2b8056a852
* fix(backend): use insertOne insteadof insert/findOneOrFail combination * fix: typo * fix(backend): inherit mainAlias? * refactor(backend): use extend * fix(backend): invalid entityTarget * fix(backend): fake where * chore: debug * chore: debug * test: log * fix(backend): column names * fix(backend): remove dummy from * revert: log * fix(backend): position * fix(backend): automatic aliasing * chore(backend): alias * chore(backend): remove from * fix(backend): type * fix(backend): avoid pure name * test(backend): fix type * chore(backend): use cte * fix(backend): avoid useless alias * fix(backend): fix typo * fix(backend): __disambiguation__ * fix(backend): quote * chore(backend): t * chore(backend): accessible * chore(backend): concrete returning * fix(backend): quote * chore: log more * chore: log metadata * chore(backend): use raw * fix(backend): returning column name * fix(backend): transform * build(backend): wanna logging * build(backend): transform empty * build(backend): build alias * build(backend): restore name * chore: return entity * fix: test case * test(backend): 204 * chore(backend): log sql * chore(backend): assert user joined * fix(backend): typo * chore(backend): log long sql * chore(backend): log join * chore(backend): log join depth null * chore(backend): joinAttributes * chore(backend): override createJoinExpression * chore: join log * fix(backend): escape * test(backend): log log * chore(backend): join gonna success? * chore(backend): relations * chore(backend): undefined * chore(backend): target * chore(backend): remove log * chore(backend): log chart update * chore(backend): log columns * chore(backend): check hasMetadata * chore(backend): unshift id when not included * chore(backend): missing select * chore(backend): remove debug code
138 lines
3.8 KiB
TypeScript
138 lines
3.8 KiB
TypeScript
/*
|
|
* SPDX-FileCopyrightText: syuilo and misskey-project
|
|
* SPDX-License-Identifier: AGPL-3.0-only
|
|
*/
|
|
|
|
import { Inject, Injectable } from '@nestjs/common';
|
|
import { Endpoint } from '@/server/api/endpoint-base.js';
|
|
import { IdService } from '@/core/IdService.js';
|
|
import type { UserListsRepository, AntennasRepository } from '@/models/_.js';
|
|
import { GlobalEventService } from '@/core/GlobalEventService.js';
|
|
import { AntennaEntityService } from '@/core/entities/AntennaEntityService.js';
|
|
import { DI } from '@/di-symbols.js';
|
|
import { RoleService } from '@/core/RoleService.js';
|
|
import { ApiError } from '../../error.js';
|
|
|
|
export const meta = {
|
|
tags: ['antennas'],
|
|
|
|
requireCredential: true,
|
|
|
|
prohibitMoved: true,
|
|
|
|
kind: 'write:account',
|
|
|
|
errors: {
|
|
noSuchUserList: {
|
|
message: 'No such user list.',
|
|
code: 'NO_SUCH_USER_LIST',
|
|
id: '95063e93-a283-4b8b-9aa5-bcdb8df69a7f',
|
|
},
|
|
|
|
tooManyAntennas: {
|
|
message: 'You cannot create antenna any more.',
|
|
code: 'TOO_MANY_ANTENNAS',
|
|
id: 'faf47050-e8b5-438c-913c-db2b1576fde4',
|
|
},
|
|
},
|
|
|
|
res: {
|
|
type: 'object',
|
|
optional: false, nullable: false,
|
|
ref: 'Antenna',
|
|
},
|
|
} as const;
|
|
|
|
export const paramDef = {
|
|
type: 'object',
|
|
properties: {
|
|
name: { type: 'string', minLength: 1, maxLength: 100 },
|
|
src: { type: 'string', enum: ['home', 'all', 'users', 'list', 'users_blacklist'] },
|
|
userListId: { type: 'string', format: 'misskey:id', nullable: true },
|
|
keywords: { type: 'array', items: {
|
|
type: 'array', items: {
|
|
type: 'string',
|
|
},
|
|
} },
|
|
excludeKeywords: { type: 'array', items: {
|
|
type: 'array', items: {
|
|
type: 'string',
|
|
},
|
|
} },
|
|
users: { type: 'array', items: {
|
|
type: 'string',
|
|
} },
|
|
caseSensitive: { type: 'boolean' },
|
|
localOnly: { type: 'boolean' },
|
|
excludeBots: { type: 'boolean' },
|
|
withReplies: { type: 'boolean' },
|
|
withFile: { type: 'boolean' },
|
|
},
|
|
required: ['name', 'src', 'keywords', 'excludeKeywords', 'users', 'caseSensitive', 'withReplies', 'withFile'],
|
|
} as const;
|
|
|
|
@Injectable()
|
|
export default class extends Endpoint<typeof meta, typeof paramDef> { // eslint-disable-line import/no-default-export
|
|
constructor(
|
|
@Inject(DI.antennasRepository)
|
|
private antennasRepository: AntennasRepository,
|
|
|
|
@Inject(DI.userListsRepository)
|
|
private userListsRepository: UserListsRepository,
|
|
|
|
private antennaEntityService: AntennaEntityService,
|
|
private roleService: RoleService,
|
|
private idService: IdService,
|
|
private globalEventService: GlobalEventService,
|
|
) {
|
|
super(meta, paramDef, async (ps, me) => {
|
|
if (ps.keywords.flat().every(x => x === '') && ps.excludeKeywords.flat().every(x => x === '')) {
|
|
throw new Error('either keywords or excludeKeywords is required.');
|
|
}
|
|
|
|
const currentAntennasCount = await this.antennasRepository.countBy({
|
|
userId: me.id,
|
|
});
|
|
if (currentAntennasCount > (await this.roleService.getUserPolicies(me.id)).antennaLimit) {
|
|
throw new ApiError(meta.errors.tooManyAntennas);
|
|
}
|
|
|
|
let userList;
|
|
|
|
if (ps.src === 'list' && ps.userListId) {
|
|
userList = await this.userListsRepository.findOneBy({
|
|
id: ps.userListId,
|
|
userId: me.id,
|
|
});
|
|
|
|
if (userList == null) {
|
|
throw new ApiError(meta.errors.noSuchUserList);
|
|
}
|
|
}
|
|
|
|
const now = new Date();
|
|
|
|
const antenna = await this.antennasRepository.insertOne({
|
|
id: this.idService.gen(now.getTime()),
|
|
lastUsedAt: now,
|
|
userId: me.id,
|
|
name: ps.name,
|
|
src: ps.src,
|
|
userListId: userList ? userList.id : null,
|
|
keywords: ps.keywords,
|
|
excludeKeywords: ps.excludeKeywords,
|
|
users: ps.users,
|
|
caseSensitive: ps.caseSensitive,
|
|
localOnly: ps.localOnly,
|
|
excludeBots: ps.excludeBots,
|
|
withReplies: ps.withReplies,
|
|
withFile: ps.withFile,
|
|
});
|
|
|
|
this.globalEventService.publishInternalEvent('antennaCreated', antenna);
|
|
|
|
return await this.antennaEntityService.pack(antenna);
|
|
});
|
|
}
|
|
}
|