create tags

This commit is contained in:
MathMan05
2026-02-05 10:50:48 -06:00
parent 9e5976a4d5
commit c26b0f3a4c
7 changed files with 112 additions and 2 deletions

View File

@@ -0,0 +1,66 @@
/*
Spacebar: A FOSS re-implementation and extension of the Discord.com backend.
Copyright (C) 2023 Spacebar and Spacebar Contributors
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU Affero General Public License as published
by the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU Affero General Public License for more details.
You should have received a copy of the GNU Affero General Public License
along with this program. If not, see <https://www.gnu.org/licenses/>.
*/
import { route } from "@spacebar/api";
import { Channel, ChannelUpdateEvent, emitEvent, Tag } from "@spacebar/util";
import { Request, Response, Router } from "express";
import { TagCreateSchema } from "@spacebar/schemas";
const router: Router = Router({ mergeParams: true });
router.post(
"/",
route({
requestBody: "TagCreateSchema",
permission: "MANAGE_CHANNELS",
responses: {
201: {},
404: {},
},
}),
async (req: Request, res: Response) => {
const body = req.body as TagCreateSchema;
const { channel_id } = req.params as Record<string, string>;
const channel = await Channel.findOneOrFail({
where: { id: channel_id },
relations: ["available_tags"],
});
if (!channel.isForum()) throw new Error("is not thread only channel");
const tag = Tag.create({
channel,
...body,
});
channel.available_tags?.push(tag);
await Promise.all([
tag.save(),
emitEvent({
event: "CHANNEL_UPDATE",
data: channel.toJSON(),
channel_id,
} as ChannelUpdateEvent),
]);
res.json(channel.toJSON());
},
);
export default router;

View File

@@ -304,6 +304,7 @@ export async function onIdentify(this: WebSocket, data: Payload) {
type: Not(In([ChannelType.GUILD_PUBLIC_THREAD, ChannelType.GUILD_PRIVATE_THREAD, ChannelType.GUILD_NEWS_THREAD])),
},
order: { guild_id: "ASC" },
relations: ["available_tags"],
}),
),
timePromise(() =>

View File

@@ -0,0 +1,24 @@
/*
Spacebar: A FOSS re-implementation and extension of the Discord.com backend.
Copyright (C) 2023 Spacebar and Spacebar Contributors
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU Affero General Public License as published
by the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU Affero General Public License for more details.
You should have received a copy of the GNU Affero General Public License
along with this program. If not, see <https://www.gnu.org/licenses/>.
*/
export interface TagCreateSchema {
name: string;
moderated?: boolean;
emoji_id?: string;
emoji_name?: string;
}

View File

@@ -95,3 +95,4 @@ export * from "./MessageThreadCreationSchema";
export * from "./ThreadCreationSchema";
export * from "./MessageActivity";
export * from "./PostDataSchema";
export * from "./TagCreateSchema";

View File

@@ -593,6 +593,9 @@ export class Channel extends BaseClass {
isThread() {
return this.type === ChannelType.GUILD_NEWS_THREAD || this.type === ChannelType.GUILD_PUBLIC_THREAD || this.type === ChannelType.GUILD_PRIVATE_THREAD;
}
isForum() {
return this.type === ChannelType.GUILD_FORUM || this.type === ChannelType.GUILD_MEDIA;
}
isPrivateThread() {
return this.type === ChannelType.GUILD_PRIVATE_THREAD;

View File

@@ -39,10 +39,10 @@ export class Tag extends BaseClass {
@Column()
moderated: boolean = false;
@Column()
@Column({ nullable: true })
emoji_id?: string;
@Column()
@Column({ nullable: true })
emoji_name?: string;
toJSON() {

View File

@@ -0,0 +1,15 @@
import { MigrationInterface, QueryRunner } from "typeorm";
export class NullableTags1770310028511 implements MigrationInterface {
name = "NullableTags1770310028511";
public async up(queryRunner: QueryRunner): Promise<void> {
await queryRunner.query(`ALTER TABLE "tags" ALTER COLUMN "emoji_id" DROP NOT NULL`);
await queryRunner.query(`ALTER TABLE "tags" ALTER COLUMN "emoji_name" DROP NOT NULL`);
}
public async down(queryRunner: QueryRunner): Promise<void> {
await queryRunner.query(`ALTER TABLE "tags" ALTER COLUMN "emoji_name" SET NOT NULL`);
await queryRunner.query(`ALTER TABLE "tags" ALTER COLUMN "emoji_id" SET NOT NULL`);
}
}