implement guild profiles and fix user profiles
This commit is contained in:
parent
5e289fea77
commit
b84aa73852
19895
assets/schemas.json
19895
assets/schemas.json
File diff suppressed because it is too large
Load Diff
@ -9,6 +9,7 @@ import {
|
|||||||
Sticker,
|
Sticker,
|
||||||
Emoji,
|
Emoji,
|
||||||
Guild,
|
Guild,
|
||||||
|
handleFile,
|
||||||
MemberChangeSchema,
|
MemberChangeSchema,
|
||||||
} from "@fosscord/util";
|
} from "@fosscord/util";
|
||||||
import { route } from "@fosscord/api";
|
import { route } from "@fosscord/api";
|
||||||
@ -26,39 +27,25 @@ router.get("/", route({}), async (req: Request, res: Response) => {
|
|||||||
return res.json(member);
|
return res.json(member);
|
||||||
});
|
});
|
||||||
|
|
||||||
router.patch(
|
router.patch("/", route({ body: "MemberChangeSchema" }), async (req: Request, res: Response) => {
|
||||||
"/",
|
|
||||||
route({ body: "MemberChangeSchema" }),
|
|
||||||
async (req: Request, res: Response) => {
|
|
||||||
let { guild_id, member_id } = req.params;
|
let { guild_id, member_id } = req.params;
|
||||||
if (member_id === "@me") member_id = req.user_id;
|
if (member_id === "@me") member_id = req.user_id;
|
||||||
const body = req.body as MemberChangeSchema;
|
const body = req.body as MemberChangeSchema;
|
||||||
|
|
||||||
const member = await Member.findOneOrFail({
|
let member = await Member.findOneOrFail({ where: { id: member_id, guild_id }, relations: ["roles", "user"] });
|
||||||
where: { id: member_id, guild_id },
|
|
||||||
relations: ["roles", "user"],
|
|
||||||
});
|
|
||||||
const permission = await getPermission(req.user_id, guild_id);
|
const permission = await getPermission(req.user_id, guild_id);
|
||||||
const everyone = await Role.findOneOrFail({
|
const everyone = await Role.findOneOrFail({ where: { guild_id: guild_id, name: "@everyone", position: 0 } });
|
||||||
where: { guild_id: guild_id, name: "@everyone", position: 0 },
|
|
||||||
});
|
|
||||||
|
|
||||||
if (body.roles) {
|
if (body.roles) {
|
||||||
permission.hasThrow("MANAGE_ROLES");
|
permission.hasThrow("MANAGE_ROLES");
|
||||||
|
|
||||||
if (body.roles.indexOf(everyone.id) === -1)
|
if (body.roles.indexOf(everyone.id) === -1) body.roles.push(everyone.id);
|
||||||
body.roles.push(everyone.id);
|
|
||||||
member.roles = body.roles.map((x) => Role.create({ id: x })); // foreign key constraint will fail if role doesn't exist
|
member.roles = body.roles.map((x) => Role.create({ id: x })); // foreign key constraint will fail if role doesn't exist
|
||||||
}
|
}
|
||||||
|
|
||||||
if ("nick" in body) {
|
if (body.avatar) body.avatar = await handleFile(`/guilds/${guild_id}/users/${member_id}/avatars`, body.avatar as string);
|
||||||
permission.hasThrow(
|
|
||||||
req.user_id == member.user.id
|
member.assign(body);
|
||||||
? "CHANGE_NICKNAME"
|
|
||||||
: "MANAGE_NICKNAMES",
|
|
||||||
);
|
|
||||||
member.nick = body.nick?.trim() || undefined;
|
|
||||||
}
|
|
||||||
|
|
||||||
await member.save();
|
await member.save();
|
||||||
|
|
||||||
@ -68,12 +55,11 @@ router.patch(
|
|||||||
await emitEvent({
|
await emitEvent({
|
||||||
event: "GUILD_MEMBER_UPDATE",
|
event: "GUILD_MEMBER_UPDATE",
|
||||||
guild_id,
|
guild_id,
|
||||||
data: { ...member, roles: member.roles.map((x) => x.id) },
|
data: { ...member, roles: member.roles.map((x) => x.id) }
|
||||||
} as GuildMemberUpdateEvent);
|
} as GuildMemberUpdateEvent);
|
||||||
|
|
||||||
res.json(member);
|
res.json(member);
|
||||||
},
|
});
|
||||||
);
|
|
||||||
|
|
||||||
router.put("/", route({}), async (req: Request, res: Response) => {
|
router.put("/", route({}), async (req: Request, res: Response) => {
|
||||||
// TODO: Lurker mode
|
// TODO: Lurker mode
|
||||||
|
30
src/api/routes/guilds/#guild_id/profile/index.ts
Normal file
30
src/api/routes/guilds/#guild_id/profile/index.ts
Normal file
@ -0,0 +1,30 @@
|
|||||||
|
import { route } from "@fosscord/api";
|
||||||
|
import { emitEvent, GuildMemberUpdateEvent, handleFile, Member, MemberChangeProfileSchema, OrmUtils } from "@fosscord/util";
|
||||||
|
import { Request, Response, Router } from "express";
|
||||||
|
|
||||||
|
const router = Router();
|
||||||
|
|
||||||
|
router.patch("/:member_id", route({ body: "MemberChangeProfileSchema" }), async (req: Request, res: Response) => {
|
||||||
|
let { guild_id, member_id } = req.params;
|
||||||
|
if (member_id === "@me") member_id = req.user_id;
|
||||||
|
const body = req.body as MemberChangeProfileSchema;
|
||||||
|
|
||||||
|
let member = await Member.findOneOrFail({ where: { id: req.user_id, guild_id }, relations: ["roles", "user"] });
|
||||||
|
|
||||||
|
if (body.banner) body.banner = await handleFile(`/guilds/${guild_id}/users/${req.user_id}/avatars`, body.banner as string);
|
||||||
|
|
||||||
|
member = await OrmUtils.mergeDeep(member, body);
|
||||||
|
|
||||||
|
await member.save();
|
||||||
|
|
||||||
|
// do not use promise.all as we have to first write to db before emitting the event to catch errors
|
||||||
|
await emitEvent({
|
||||||
|
event: "GUILD_MEMBER_UPDATE",
|
||||||
|
guild_id,
|
||||||
|
data: { ...member, roles: member.roles.map((x) => x.id) }
|
||||||
|
} as GuildMemberUpdateEvent);
|
||||||
|
|
||||||
|
res.json(member);
|
||||||
|
});
|
||||||
|
|
||||||
|
export default router;
|
@ -6,6 +6,11 @@ import {
|
|||||||
UserPublic,
|
UserPublic,
|
||||||
Member,
|
Member,
|
||||||
Guild,
|
Guild,
|
||||||
|
UserProfileModifySchema,
|
||||||
|
handleFile,
|
||||||
|
PrivateUserProjection,
|
||||||
|
emitEvent,
|
||||||
|
UserUpdateEvent,
|
||||||
} from "@fosscord/util";
|
} from "@fosscord/util";
|
||||||
import { route } from "@fosscord/api";
|
import { route } from "@fosscord/api";
|
||||||
|
|
||||||
@ -86,10 +91,10 @@ router.get(
|
|||||||
|
|
||||||
const guildMemberDto = guild_member
|
const guildMemberDto = guild_member
|
||||||
? {
|
? {
|
||||||
avatar: user.avatar, // TODO
|
avatar: guild_member.avatar,
|
||||||
banner: user.banner, // TODO
|
banner: guild_member.banner,
|
||||||
bio: req.user_bot ? null : user.bio, // TODO
|
bio: req.user_bot ? null : guild_member.bio,
|
||||||
communication_disabled_until: null, // TODO
|
communication_disabled_until: guild_member.communication_disabled_until,
|
||||||
deaf: guild_member.deaf,
|
deaf: guild_member.deaf,
|
||||||
flags: user.flags,
|
flags: user.flags,
|
||||||
is_pending: guild_member.pending,
|
is_pending: guild_member.pending,
|
||||||
@ -98,13 +103,17 @@ router.get(
|
|||||||
mute: guild_member.mute,
|
mute: guild_member.mute,
|
||||||
nick: guild_member.nick,
|
nick: guild_member.nick,
|
||||||
premium_since: guild_member.premium_since,
|
premium_since: guild_member.premium_since,
|
||||||
roles: guild_member.roles
|
roles: guild_member.roles.map((x) => x.id).filter((id) => id != guild_id),
|
||||||
.map((x) => x.id)
|
user: userDto
|
||||||
.filter((id) => id != guild_id),
|
|
||||||
user: userDto,
|
|
||||||
}
|
}
|
||||||
: undefined;
|
: undefined;
|
||||||
|
|
||||||
|
const guildMemberProfile = {
|
||||||
|
accent_color: null,
|
||||||
|
banner: guild_member?.banner || null,
|
||||||
|
bio: guild_member?.bio || "",
|
||||||
|
guild_id
|
||||||
|
};
|
||||||
res.json({
|
res.json({
|
||||||
connected_accounts: user.connected_accounts,
|
connected_accounts: user.connected_accounts,
|
||||||
premium_guild_since: premium_guild_since, // TODO
|
premium_guild_since: premium_guild_since, // TODO
|
||||||
@ -112,8 +121,34 @@ router.get(
|
|||||||
mutual_guilds: mutual_guilds, // TODO {id: "", nick: null} when ?with_mutual_guilds=true
|
mutual_guilds: mutual_guilds, // TODO {id: "", nick: null} when ?with_mutual_guilds=true
|
||||||
user: userDto,
|
user: userDto,
|
||||||
guild_member: guildMemberDto,
|
guild_member: guildMemberDto,
|
||||||
|
guild_member_profile: guildMemberProfile
|
||||||
});
|
});
|
||||||
},
|
});
|
||||||
);
|
|
||||||
|
router.patch("/", route({ body: "UserProfileModifySchema" }), async (req: Request, res: Response) => {
|
||||||
|
const body = req.body as UserProfileModifySchema;
|
||||||
|
|
||||||
|
if (body.banner) body.banner = await handleFile(`/banners/${req.user_id}`, body.banner as string);
|
||||||
|
let user = await User.findOneOrFail({ where: { id: req.user_id }, select: [...PrivateUserProjection, "data"] });
|
||||||
|
|
||||||
|
user.assign(body);
|
||||||
|
await user.save();
|
||||||
|
|
||||||
|
// @ts-ignore
|
||||||
|
delete user.data;
|
||||||
|
|
||||||
|
// TODO: send update member list event in gateway
|
||||||
|
await emitEvent({
|
||||||
|
event: "USER_UPDATE",
|
||||||
|
user_id: req.user_id,
|
||||||
|
data: user
|
||||||
|
} as UserUpdateEvent);
|
||||||
|
|
||||||
|
res.json({
|
||||||
|
accent_color: user.accent_color,
|
||||||
|
bio: user.bio,
|
||||||
|
banner: user.banner
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
export default router;
|
export default router;
|
||||||
|
@ -2,6 +2,7 @@ import { Server, ServerOptions } from "lambert-server";
|
|||||||
import { Config, initDatabase, registerRoutes } from "@fosscord/util";
|
import { Config, initDatabase, registerRoutes } from "@fosscord/util";
|
||||||
import path from "path";
|
import path from "path";
|
||||||
import avatarsRoute from "./routes/avatars";
|
import avatarsRoute from "./routes/avatars";
|
||||||
|
import guildProfilesRoute from "./routes/guild-profiles";
|
||||||
import iconsRoute from "./routes/role-icons";
|
import iconsRoute from "./routes/role-icons";
|
||||||
import bodyParser from "body-parser";
|
import bodyParser from "body-parser";
|
||||||
|
|
||||||
@ -74,6 +75,12 @@ export class CDNServer extends Server {
|
|||||||
this.app.use("/channel-icons/", avatarsRoute);
|
this.app.use("/channel-icons/", avatarsRoute);
|
||||||
this.log("verbose", "[Server] Route /channel-icons registered");
|
this.log("verbose", "[Server] Route /channel-icons registered");
|
||||||
|
|
||||||
|
this.app.use("/guilds/:guild_id/users/:user_id/avatars", guildProfilesRoute);
|
||||||
|
this.log("verbose", "[Server] Route /guilds/avatars registered");
|
||||||
|
|
||||||
|
this.app.use("/guilds/:guild_id/users/:user_id/banners", guildProfilesRoute);
|
||||||
|
this.log("verbose", "[Server] Route /guilds/banners registered");
|
||||||
|
|
||||||
return super.start();
|
return super.start();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
85
src/cdn/routes/guild-profiles.ts
Normal file
85
src/cdn/routes/guild-profiles.ts
Normal file
@ -0,0 +1,85 @@
|
|||||||
|
import { Config, Snowflake } from "@fosscord/util";
|
||||||
|
import crypto from "crypto";
|
||||||
|
import { Request, Response, Router } from "express";
|
||||||
|
import FileType from "file-type";
|
||||||
|
import { HTTPError } from "lambert-server";
|
||||||
|
import { multer } from "../util/multer";
|
||||||
|
import { storage } from "../util/Storage";
|
||||||
|
|
||||||
|
// TODO: check premium and animated pfp are allowed in the config
|
||||||
|
// TODO: generate different sizes of icon
|
||||||
|
// TODO: generate different image types of icon
|
||||||
|
// TODO: delete old icons
|
||||||
|
|
||||||
|
const ANIMATED_MIME_TYPES = ["image/apng", "image/gif", "image/gifv"];
|
||||||
|
const STATIC_MIME_TYPES = ["image/png", "image/jpeg", "image/webp", "image/svg+xml", "image/svg"];
|
||||||
|
const ALLOWED_MIME_TYPES = [...ANIMATED_MIME_TYPES, ...STATIC_MIME_TYPES];
|
||||||
|
|
||||||
|
const router = Router();
|
||||||
|
|
||||||
|
router.post("/", multer.single("file"), async (req: Request, res: Response) => {
|
||||||
|
if (req.headers.signature !== Config.get().security.requestSignature) throw new HTTPError("Invalid request signature");
|
||||||
|
if (!req.file) throw new HTTPError("Missing file");
|
||||||
|
const { buffer, mimetype, size, originalname, fieldname } = req.file;
|
||||||
|
const { guild_id, user_id } = req.params;
|
||||||
|
|
||||||
|
let hash = crypto.createHash("md5").update(Snowflake.generate()).digest("hex");
|
||||||
|
|
||||||
|
const type = await FileType.fromBuffer(buffer);
|
||||||
|
if (!type || !ALLOWED_MIME_TYPES.includes(type.mime)) throw new HTTPError("Invalid file type");
|
||||||
|
if (ANIMATED_MIME_TYPES.includes(type.mime)) hash = `a_${hash}`; // animated icons have a_ infront of the hash
|
||||||
|
|
||||||
|
const path = `guilds/${guild_id}/users/${user_id}/avatars/${hash}`;
|
||||||
|
const endpoint = Config.get().cdn.endpointPublic || "http://localhost:3003";
|
||||||
|
|
||||||
|
await storage.set(path, buffer);
|
||||||
|
|
||||||
|
return res.json({
|
||||||
|
id: hash,
|
||||||
|
content_type: type.mime,
|
||||||
|
size,
|
||||||
|
url: `${endpoint}${req.baseUrl}/${user_id}/${hash}`
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
router.get("/", async (req: Request, res: Response) => {
|
||||||
|
let { guild_id, user_id } = req.params;
|
||||||
|
user_id = user_id.split(".")[0]; // remove .file extension
|
||||||
|
const path = `guilds/${guild_id}/users/${user_id}/avatars`;
|
||||||
|
|
||||||
|
const file = await storage.get(path);
|
||||||
|
if (!file) throw new HTTPError("not found", 404);
|
||||||
|
const type = await FileType.fromBuffer(file);
|
||||||
|
|
||||||
|
res.set("Content-Type", type?.mime);
|
||||||
|
res.set("Cache-Control", "public, max-age=31536000");
|
||||||
|
|
||||||
|
return res.send(file);
|
||||||
|
});
|
||||||
|
|
||||||
|
router.get("/:hash", async (req: Request, res: Response) => {
|
||||||
|
let { guild_id, user_id, hash } = req.params;
|
||||||
|
hash = hash.split(".")[0]; // remove .file extension
|
||||||
|
const path = `guilds/${guild_id}/users/${user_id}/avatars/${hash}`;
|
||||||
|
|
||||||
|
const file = await storage.get(path);
|
||||||
|
if (!file) throw new HTTPError("not found", 404);
|
||||||
|
const type = await FileType.fromBuffer(file);
|
||||||
|
|
||||||
|
res.set("Content-Type", type?.mime);
|
||||||
|
res.set("Cache-Control", "public, max-age=31536000");
|
||||||
|
|
||||||
|
return res.send(file);
|
||||||
|
});
|
||||||
|
|
||||||
|
router.delete("/:id", async (req: Request, res: Response) => {
|
||||||
|
if (req.headers.signature !== Config.get().security.requestSignature) throw new HTTPError("Invalid request signature");
|
||||||
|
const { guild_id, user_id, id } = req.params;
|
||||||
|
const path = `guilds/${guild_id}/users/${user_id}/avatars/${id}`;
|
||||||
|
|
||||||
|
await storage.delete(path);
|
||||||
|
|
||||||
|
return res.send({ success: true });
|
||||||
|
});
|
||||||
|
|
||||||
|
export default router;
|
@ -114,7 +114,19 @@ export class Member extends BaseClassWithoutId {
|
|||||||
// do not auto-kick force-joined members just because their joiners left the server
|
// do not auto-kick force-joined members just because their joiners left the server
|
||||||
}) **/
|
}) **/
|
||||||
@Column({ nullable: true })
|
@Column({ nullable: true })
|
||||||
joined_by?: string;
|
joined_by: string;
|
||||||
|
|
||||||
|
@Column({ nullable: true })
|
||||||
|
avatar: string;
|
||||||
|
|
||||||
|
@Column({ nullable: true })
|
||||||
|
banner: string;
|
||||||
|
|
||||||
|
@Column()
|
||||||
|
bio: string;
|
||||||
|
|
||||||
|
@Column({ nullable: true })
|
||||||
|
communication_disabled_until: Date;
|
||||||
|
|
||||||
// TODO: add this when we have proper read receipts
|
// TODO: add this when we have proper read receipts
|
||||||
// @Column({ type: "simple-json" })
|
// @Column({ type: "simple-json" })
|
||||||
@ -313,6 +325,7 @@ export class Member extends BaseClassWithoutId {
|
|||||||
deaf: false,
|
deaf: false,
|
||||||
mute: false,
|
mute: false,
|
||||||
pending: false,
|
pending: false,
|
||||||
|
bio: "",
|
||||||
};
|
};
|
||||||
|
|
||||||
await Promise.all([
|
await Promise.all([
|
||||||
|
5
src/util/schemas/MemberChangeProfileSchema.ts
Normal file
5
src/util/schemas/MemberChangeProfileSchema.ts
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
export interface MemberChangeProfileSchema {
|
||||||
|
banner?: string | null;
|
||||||
|
nick?: string;
|
||||||
|
bio?: string;
|
||||||
|
}
|
@ -1,4 +1,6 @@
|
|||||||
export interface MemberChangeSchema {
|
export interface MemberChangeSchema {
|
||||||
roles?: string[];
|
roles?: string[];
|
||||||
nick?: string;
|
nick?: string;
|
||||||
|
avatar?: string | null;
|
||||||
|
bio?: string;
|
||||||
}
|
}
|
||||||
|
5
src/util/schemas/UserProfileModifySchema.ts
Normal file
5
src/util/schemas/UserProfileModifySchema.ts
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
export interface UserProfileModifySchema {
|
||||||
|
bio?: string;
|
||||||
|
accent_color?: number | null;
|
||||||
|
banner?: string | null;
|
||||||
|
}
|
@ -22,6 +22,11 @@ export * from "./TemplateModifySchema";
|
|||||||
export * from "./VanityUrlSchema";
|
export * from "./VanityUrlSchema";
|
||||||
export * from "./GuildUpdateWelcomeScreenSchema";
|
export * from "./GuildUpdateWelcomeScreenSchema";
|
||||||
export * from "./WidgetModifySchema";
|
export * from "./WidgetModifySchema";
|
||||||
|
export * from "./IdentifySchema";
|
||||||
|
export * from "./InviteCreateSchema";
|
||||||
|
export * from "./LazyRequestSchema";
|
||||||
|
export * from "./LoginSchema";
|
||||||
|
export * from "./MemberChangeProfileSchema";
|
||||||
export * from "./MemberChangeSchema";
|
export * from "./MemberChangeSchema";
|
||||||
export * from "./RoleModifySchema";
|
export * from "./RoleModifySchema";
|
||||||
export * from "./GuildTemplateCreateSchema";
|
export * from "./GuildTemplateCreateSchema";
|
||||||
@ -34,6 +39,11 @@ export * from "./MfaCodesSchema";
|
|||||||
export * from "./TotpDisableSchema";
|
export * from "./TotpDisableSchema";
|
||||||
export * from "./TotpEnableSchema";
|
export * from "./TotpEnableSchema";
|
||||||
export * from "./VoiceIdentifySchema";
|
export * from "./VoiceIdentifySchema";
|
||||||
|
export * from "./TotpSchema";
|
||||||
|
export * from "./UserModifySchema";
|
||||||
|
export * from "./UserProfileModifySchema";
|
||||||
|
export * from "./UserSettingsSchema";
|
||||||
|
export * from "./VanityUrlSchema";
|
||||||
export * from "./VoiceStateUpdateSchema";
|
export * from "./VoiceStateUpdateSchema";
|
||||||
export * from "./VoiceVideoSchema";
|
export * from "./VoiceVideoSchema";
|
||||||
export * from "./IdentifySchema";
|
export * from "./IdentifySchema";
|
||||||
|
Loading…
x
Reference in New Issue
Block a user