mirror of
https://github.com/spacebarchat/server.git
synced 2026-04-26 21:45:20 +00:00
centerlize permision checking more
This commit is contained in:
@@ -429,56 +429,7 @@ export class Channel extends BaseClass {
|
||||
}
|
||||
|
||||
async getUserPermissions(opts: { user_id?: string; user?: User; member?: Member; guild?: Guild }): Promise<Permissions> {
|
||||
let guild = opts.guild;
|
||||
if (!guild) {
|
||||
if (this.guild) guild = this.guild;
|
||||
else if (this.guild_id) guild = await Guild.findOneOrFail({ where: { id: this.guild_id } });
|
||||
else {
|
||||
console.error("Channel.getUserPermissions: called without guild for non-DM channel.");
|
||||
return Permissions.NONE;
|
||||
}
|
||||
}
|
||||
|
||||
// check if we can resolve here to short-circuit possibly calling the database unnecessarily
|
||||
// TODO: do we want to have an instance-wide opt out of this behavior? It would just be an extra if statement here
|
||||
const ownerId = guild?.owner?.id ?? guild?.owner_id;
|
||||
if (!!opts.user_id && ownerId === opts.user_id) return Permissions.ALL;
|
||||
if (!!opts.user?.id && ownerId === opts.user?.id) return Permissions.ALL;
|
||||
if (!!opts.member?.id && ownerId === opts.member?.id) return Permissions.ALL;
|
||||
|
||||
let member = opts.member;
|
||||
if (!member) {
|
||||
if (opts.user) member = await Member.findOneOrFail({ where: { guild_id: guild.id, id: opts.user.id }, relations: ["roles"] });
|
||||
else if (opts.user_id) member = await Member.findOneOrFail({ where: { guild_id: guild.id, id: opts.user_id }, relations: ["roles"] });
|
||||
else {
|
||||
console.error("Channel.getUserPermissions: called without user or member for non-DM channel.");
|
||||
return Permissions.NONE;
|
||||
}
|
||||
}
|
||||
|
||||
const roles = (member.roles || (await Member.findOneOrFail({ where: { guild_id: guild.id, index: member.index }, relations: ["roles"] })).roles).sort(
|
||||
(a, b) => a.position - b.position,
|
||||
); // ascending by position
|
||||
|
||||
// calculate user's channel perms - should in theory match https://docs.discord.food/topics/permissions#permission-overwrites
|
||||
// start at role permissions
|
||||
let userPerms = new Permissions(new BitField(0).add(roles.map((r) => r.permissions)));
|
||||
|
||||
// TODO: do we want to have an instance-wide opt out of this behavior? It would just be an extra if statement here
|
||||
if (userPerms.has(Permissions.FLAGS.ADMINISTRATOR)) return userPerms;
|
||||
|
||||
// apply channel overrides
|
||||
if (this.permission_overwrites) {
|
||||
// role overwrites - TODO: this probably violates the geneva conventions - we should probably be ordering roles here
|
||||
for (const overwrite of this.permission_overwrites.filter((o) => o.type === ChannelPermissionOverwriteType.role && roles.map((r) => r.id).includes(o.id)))
|
||||
userPerms = new Permissions(userPerms.remove(overwrite.deny).add(overwrite.allow));
|
||||
|
||||
// member overwrite, throws if somehow we have multiple overwrites for the same member
|
||||
const memberOverwrite = this.permission_overwrites.single((o) => o.type === ChannelPermissionOverwriteType.member && o.id === member?.id);
|
||||
if (memberOverwrite) userPerms = new Permissions(userPerms.remove(memberOverwrite.deny).add(memberOverwrite.allow));
|
||||
}
|
||||
|
||||
return userPerms;
|
||||
return getPermission(opts.user_id || opts.user?.id, opts.guild, this);
|
||||
}
|
||||
|
||||
// TODO: should we throw for missing args?
|
||||
|
||||
@@ -88,26 +88,17 @@ export class Permissions extends BitField {
|
||||
// CUSTOM_PERMISSION: BigInt(1) << BigInt(0) + CUSTOM_PERMISSION_OFFSET
|
||||
};
|
||||
|
||||
static ALL_PERMISSIONS = Object.values(Permissions.FLAGS).reduce(
|
||||
(total, val) => total | val,
|
||||
BigInt(0),
|
||||
);
|
||||
static ALL_PERMISSIONS = Object.values(Permissions.FLAGS).reduce((total, val) => total | val, BigInt(0));
|
||||
|
||||
any(permission: PermissionResolvable, checkAdmin = true) {
|
||||
return (
|
||||
(checkAdmin && super.any(Permissions.FLAGS.ADMINISTRATOR)) ||
|
||||
super.any(permission)
|
||||
);
|
||||
return (checkAdmin && super.any(Permissions.FLAGS.ADMINISTRATOR)) || super.any(permission);
|
||||
}
|
||||
|
||||
/**
|
||||
* Checks whether the bitfield has a permission, or multiple permissions.
|
||||
*/
|
||||
has(permission: PermissionResolvable, checkAdmin = true) {
|
||||
return (
|
||||
(checkAdmin && super.has(Permissions.FLAGS.ADMINISTRATOR)) ||
|
||||
super.has(permission)
|
||||
);
|
||||
return (checkAdmin && super.has(Permissions.FLAGS.ADMINISTRATOR)) || super.has(permission);
|
||||
}
|
||||
|
||||
/**
|
||||
@@ -115,30 +106,21 @@ export class Permissions extends BitField {
|
||||
*/
|
||||
hasThrow(permission: PermissionResolvable) {
|
||||
if (this.has(permission) && this.has("VIEW_CHANNEL")) return true;
|
||||
throw new HTTPError(
|
||||
`You are missing the following permissions ${permission}`,
|
||||
403,
|
||||
);
|
||||
throw new HTTPError(`You are missing the following permissions ${permission}`, 403);
|
||||
}
|
||||
|
||||
overwriteChannel(overwrites: ChannelPermissionOverwrite[]) {
|
||||
if (!overwrites) return this;
|
||||
if (!this.cache) throw new Error("permission chache not available");
|
||||
overwrites = overwrites.filter((x) => {
|
||||
if (x.type === ChannelPermissionOverwriteType.role && this.cache.roles?.some((r) => r.id === x.id))
|
||||
return true;
|
||||
if (x.type === ChannelPermissionOverwriteType.role && this.cache.roles?.some((r) => r.id === x.id)) return true;
|
||||
if (x.type === ChannelPermissionOverwriteType.member && x.id == this.cache.user_id) return true;
|
||||
return false;
|
||||
});
|
||||
return new Permissions(
|
||||
Permissions.channelPermission(overwrites, this.bitfield),
|
||||
);
|
||||
return new Permissions(Permissions.channelPermission(overwrites, this.bitfield));
|
||||
}
|
||||
|
||||
static channelPermission(
|
||||
overwrites: ChannelPermissionOverwrite[],
|
||||
init?: bigint,
|
||||
) {
|
||||
static channelPermission(overwrites: ChannelPermissionOverwrite[], init?: bigint) {
|
||||
// TODO: do not deny any permissions if admin
|
||||
return overwrites.reduce(
|
||||
(permission, overwrite) => {
|
||||
@@ -146,10 +128,7 @@ export class Permissions extends BitField {
|
||||
// * permission: current calculated permission (e.g. 010)
|
||||
// * deny contains all denied permissions (e.g. 011)
|
||||
// * allow contains all explicitly allowed permisions (e.g. 100)
|
||||
return (
|
||||
(permission & ~BigInt(overwrite.deny)) |
|
||||
BigInt(overwrite.allow)
|
||||
);
|
||||
return (permission & ~BigInt(overwrite.deny)) | BigInt(overwrite.allow);
|
||||
// ~ operator inverts deny (e.g. 011 -> 100)
|
||||
// & operator only allows 1 for both ~deny and permission (e.g. 010 & 100 -> 000)
|
||||
// | operators adds both together (e.g. 000 + 100 -> 100)
|
||||
@@ -160,10 +139,7 @@ export class Permissions extends BitField {
|
||||
|
||||
static rolePermission(roles: Role[]) {
|
||||
// adds all permissions of all roles together (Bit OR)
|
||||
return roles.reduce(
|
||||
(permission, role) => permission | BigInt(role.permissions),
|
||||
BigInt(0),
|
||||
);
|
||||
return roles.reduce((permission, role) => permission | BigInt(role.permissions), BigInt(0));
|
||||
}
|
||||
|
||||
static finalPermission({
|
||||
@@ -194,8 +170,7 @@ export class Permissions extends BitField {
|
||||
}
|
||||
|
||||
if (channel?.recipient_ids) {
|
||||
if (channel?.owner_id === user.id)
|
||||
return new Permissions("ADMINISTRATOR");
|
||||
if (channel?.owner_id === user.id) return new Permissions("ADMINISTRATOR");
|
||||
if (channel.recipient_ids.includes(user.id)) {
|
||||
// Default dm permissions
|
||||
return new Permissions([
|
||||
@@ -221,10 +196,7 @@ export class Permissions extends BitField {
|
||||
}
|
||||
|
||||
static NONE: Permissions = new Permissions(0);
|
||||
static ALL: Permissions = new Permissions(Object.values(Permissions.FLAGS).reduce(
|
||||
(total, val) => total | val,
|
||||
BigInt(0),
|
||||
));
|
||||
static ALL: Permissions = new Permissions(Object.values(Permissions.FLAGS).reduce((total, val) => total | val, BigInt(0)));
|
||||
}
|
||||
|
||||
export type PermissionCache = {
|
||||
@@ -237,8 +209,8 @@ export type PermissionCache = {
|
||||
|
||||
export async function getPermission(
|
||||
user_id?: string,
|
||||
guild_id?: string,
|
||||
channel_id?: string,
|
||||
guild_id?: string | Guild,
|
||||
channel_id?: string | Channel,
|
||||
opts: {
|
||||
guild_select?: (keyof Guild)[];
|
||||
guild_relations?: string[];
|
||||
@@ -253,33 +225,31 @@ export async function getPermission(
|
||||
let member: Member | undefined;
|
||||
let guild: Guild | undefined;
|
||||
|
||||
if (channel_id) {
|
||||
if (typeof channel_id === "string") {
|
||||
channel = await Channel.findOneOrFail({
|
||||
where: { id: channel_id },
|
||||
relations: ["recipients", ...(opts.channel_relations || [])],
|
||||
select: [
|
||||
"id",
|
||||
"recipients",
|
||||
"permission_overwrites",
|
||||
"owner_id",
|
||||
"guild_id",
|
||||
...(opts.channel_select || []),
|
||||
],
|
||||
select: ["id", "recipients", "permission_overwrites", "owner_id", "guild_id", ...(opts.channel_select || [])],
|
||||
});
|
||||
if (channel.guild_id) guild_id = channel.guild_id; // derive guild_id from the channel
|
||||
} else if (channel_id) {
|
||||
channel = channel_id;
|
||||
}
|
||||
|
||||
if (guild_id) {
|
||||
guild = await Guild.findOneOrFail({
|
||||
where: { id: guild_id },
|
||||
select: ["id", "owner_id", ...(opts.guild_select || [])],
|
||||
relations: opts.guild_relations,
|
||||
});
|
||||
if (guild.owner_id === user_id)
|
||||
return new Permissions(Permissions.FLAGS.ADMINISTRATOR);
|
||||
if (typeof guild_id === "string") {
|
||||
guild = await Guild.findOneOrFail({
|
||||
where: { id: guild_id },
|
||||
select: ["id", "owner_id", ...(opts.guild_select || [])],
|
||||
relations: opts.guild_relations,
|
||||
});
|
||||
} else {
|
||||
guild = guild_id;
|
||||
}
|
||||
if (guild.owner_id === user_id) return new Permissions(Permissions.FLAGS.ADMINISTRATOR);
|
||||
|
||||
member = await Member.findOneOrFail({
|
||||
where: { guild_id, id: user_id },
|
||||
where: { guild_id: guild.id, id: user_id },
|
||||
relations: ["roles", ...(opts.member_relations || [])],
|
||||
// select: [
|
||||
// "id", // TODO: Bug in typeorm? adding these selects breaks the query.
|
||||
|
||||
Reference in New Issue
Block a user