refactor progress
This commit is contained in:
151
packages/api/src/constants/Channel.tsx
Normal file
151
packages/api/src/constants/Channel.tsx
Normal file
@@ -0,0 +1,151 @@
|
||||
/*
|
||||
* Copyright (C) 2026 Fluxer Contributors
|
||||
*
|
||||
* This file is part of Fluxer.
|
||||
*
|
||||
* Fluxer 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.
|
||||
*
|
||||
* Fluxer 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 Fluxer. If not, see <https://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
import type {ValueOf} from '@fluxer/constants/src/ValueOf';
|
||||
|
||||
export const ChannelTypes = {
|
||||
GUILD_TEXT: 0,
|
||||
DM: 1,
|
||||
GUILD_VOICE: 2,
|
||||
GROUP_DM: 3,
|
||||
GUILD_CATEGORY: 4,
|
||||
GUILD_LINK: 998,
|
||||
DM_PERSONAL_NOTES: 999,
|
||||
} as const;
|
||||
|
||||
export const TEXT_BASED_CHANNEL_TYPES = new Set<number>([
|
||||
ChannelTypes.GUILD_TEXT,
|
||||
ChannelTypes.DM,
|
||||
ChannelTypes.DM_PERSONAL_NOTES,
|
||||
ChannelTypes.GROUP_DM,
|
||||
]);
|
||||
|
||||
export const InviteTypes = {
|
||||
GUILD: 0,
|
||||
GROUP_DM: 1,
|
||||
EMOJI_PACK: 2,
|
||||
STICKER_PACK: 3,
|
||||
} as const;
|
||||
|
||||
export const MessageTypes = {
|
||||
DEFAULT: 0,
|
||||
RECIPIENT_ADD: 1,
|
||||
RECIPIENT_REMOVE: 2,
|
||||
CALL: 3,
|
||||
CHANNEL_NAME_CHANGE: 4,
|
||||
CHANNEL_ICON_CHANGE: 5,
|
||||
CHANNEL_PINNED_MESSAGE: 6,
|
||||
USER_JOIN: 7,
|
||||
REPLY: 19,
|
||||
} as const;
|
||||
type MessageTypeValue = ValueOf<typeof MessageTypes>;
|
||||
|
||||
const MESSAGE_TYPE_DELETABLE = {
|
||||
[MessageTypes.DEFAULT]: true,
|
||||
[MessageTypes.REPLY]: true,
|
||||
[MessageTypes.CHANNEL_PINNED_MESSAGE]: true,
|
||||
[MessageTypes.USER_JOIN]: true,
|
||||
[MessageTypes.RECIPIENT_ADD]: false,
|
||||
[MessageTypes.RECIPIENT_REMOVE]: false,
|
||||
[MessageTypes.CALL]: false,
|
||||
[MessageTypes.CHANNEL_NAME_CHANGE]: false,
|
||||
[MessageTypes.CHANNEL_ICON_CHANGE]: false,
|
||||
} as const satisfies Record<MessageTypeValue, boolean>;
|
||||
|
||||
export function isMessageTypeDeletable(type: number): boolean {
|
||||
return type in MESSAGE_TYPE_DELETABLE ? MESSAGE_TYPE_DELETABLE[type as MessageTypeValue] : false;
|
||||
}
|
||||
|
||||
export const MessageReferenceTypes = {
|
||||
DEFAULT: 0,
|
||||
FORWARD: 1,
|
||||
} as const;
|
||||
|
||||
export const MessageFlags = {
|
||||
SUPPRESS_EMBEDS: 1 << 2,
|
||||
SUPPRESS_NOTIFICATIONS: 1 << 12,
|
||||
COMPACT_ATTACHMENTS: 1 << 17,
|
||||
} as const;
|
||||
|
||||
export const SENDABLE_MESSAGE_FLAGS =
|
||||
MessageFlags.SUPPRESS_EMBEDS | MessageFlags.SUPPRESS_NOTIFICATIONS | MessageFlags.COMPACT_ATTACHMENTS;
|
||||
|
||||
export const MessageAttachmentFlags = {
|
||||
IS_SPOILER: 1 << 3,
|
||||
CONTAINS_EXPLICIT_MEDIA: 1 << 4,
|
||||
IS_ANIMATED: 1 << 5,
|
||||
} as const;
|
||||
|
||||
export const Permissions = {
|
||||
CREATE_INSTANT_INVITE: 1n << 0n,
|
||||
KICK_MEMBERS: 1n << 1n,
|
||||
BAN_MEMBERS: 1n << 2n,
|
||||
ADMINISTRATOR: 1n << 3n,
|
||||
MANAGE_CHANNELS: 1n << 4n,
|
||||
MANAGE_GUILD: 1n << 5n,
|
||||
ADD_REACTIONS: 1n << 6n,
|
||||
VIEW_AUDIT_LOG: 1n << 7n,
|
||||
PRIORITY_SPEAKER: 1n << 8n,
|
||||
STREAM: 1n << 9n,
|
||||
VIEW_CHANNEL: 1n << 10n,
|
||||
SEND_MESSAGES: 1n << 11n,
|
||||
SEND_TTS_MESSAGES: 1n << 12n,
|
||||
MANAGE_MESSAGES: 1n << 13n,
|
||||
EMBED_LINKS: 1n << 14n,
|
||||
ATTACH_FILES: 1n << 15n,
|
||||
READ_MESSAGE_HISTORY: 1n << 16n,
|
||||
MENTION_EVERYONE: 1n << 17n,
|
||||
USE_EXTERNAL_EMOJIS: 1n << 18n,
|
||||
CONNECT: 1n << 20n,
|
||||
SPEAK: 1n << 21n,
|
||||
MUTE_MEMBERS: 1n << 22n,
|
||||
DEAFEN_MEMBERS: 1n << 23n,
|
||||
MOVE_MEMBERS: 1n << 24n,
|
||||
USE_VAD: 1n << 25n,
|
||||
CHANGE_NICKNAME: 1n << 26n,
|
||||
MANAGE_NICKNAMES: 1n << 27n,
|
||||
MANAGE_ROLES: 1n << 28n,
|
||||
MANAGE_WEBHOOKS: 1n << 29n,
|
||||
MANAGE_EXPRESSIONS: 1n << 30n,
|
||||
USE_EXTERNAL_STICKERS: 1n << 37n,
|
||||
MODERATE_MEMBERS: 1n << 40n,
|
||||
CREATE_EXPRESSIONS: 1n << 43n,
|
||||
PIN_MESSAGES: 1n << 51n,
|
||||
BYPASS_SLOWMODE: 1n << 52n,
|
||||
UPDATE_RTC_REGION: 1n << 53n,
|
||||
} as const;
|
||||
|
||||
export const ALL_PERMISSIONS = Object.values(Permissions).reduce((acc, p) => acc | p, 0n);
|
||||
|
||||
export const DEFAULT_PERMISSIONS =
|
||||
Permissions.CREATE_INSTANT_INVITE |
|
||||
Permissions.ADD_REACTIONS |
|
||||
Permissions.STREAM |
|
||||
Permissions.VIEW_CHANNEL |
|
||||
Permissions.SEND_MESSAGES |
|
||||
Permissions.EMBED_LINKS |
|
||||
Permissions.ATTACH_FILES |
|
||||
Permissions.READ_MESSAGE_HISTORY |
|
||||
Permissions.USE_EXTERNAL_EMOJIS |
|
||||
Permissions.CONNECT |
|
||||
Permissions.SPEAK |
|
||||
Permissions.USE_VAD |
|
||||
Permissions.CHANGE_NICKNAME |
|
||||
Permissions.USE_EXTERNAL_STICKERS |
|
||||
Permissions.CREATE_EXPRESSIONS;
|
||||
22
packages/api/src/constants/Core.tsx
Normal file
22
packages/api/src/constants/Core.tsx
Normal file
@@ -0,0 +1,22 @@
|
||||
/*
|
||||
* Copyright (C) 2026 Fluxer Contributors
|
||||
*
|
||||
* This file is part of Fluxer.
|
||||
*
|
||||
* Fluxer 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.
|
||||
*
|
||||
* Fluxer 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 Fluxer. If not, see <https://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
import {createUserID} from '@fluxer/api/src/BrandedTypes';
|
||||
|
||||
export const SYSTEM_USER_ID = createUserID(0n);
|
||||
78
packages/api/src/constants/Gateway.tsx
Normal file
78
packages/api/src/constants/Gateway.tsx
Normal file
@@ -0,0 +1,78 @@
|
||||
/*
|
||||
* Copyright (C) 2026 Fluxer Contributors
|
||||
*
|
||||
* This file is part of Fluxer.
|
||||
*
|
||||
* Fluxer 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.
|
||||
*
|
||||
* Fluxer 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 Fluxer. If not, see <https://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
export type GatewayDispatchEvent =
|
||||
| 'READY'
|
||||
| 'RESUMED'
|
||||
| 'SESSIONS_REPLACE'
|
||||
| 'USER_UPDATE'
|
||||
| 'USER_PINNED_DMS_UPDATE'
|
||||
| 'USER_SETTINGS_UPDATE'
|
||||
| 'USER_GUILD_SETTINGS_UPDATE'
|
||||
| 'USER_NOTE_UPDATE'
|
||||
| 'RECENT_MENTION_DELETE'
|
||||
| 'SAVED_MESSAGE_CREATE'
|
||||
| 'SAVED_MESSAGE_DELETE'
|
||||
| 'FAVORITE_MEME_CREATE'
|
||||
| 'FAVORITE_MEME_UPDATE'
|
||||
| 'FAVORITE_MEME_DELETE'
|
||||
| 'AUTH_SESSION_CHANGE'
|
||||
| 'PRESENCE_UPDATE'
|
||||
| 'GUILD_CREATE'
|
||||
| 'GUILD_UPDATE'
|
||||
| 'GUILD_DELETE'
|
||||
| 'GUILD_MEMBER_ADD'
|
||||
| 'GUILD_MEMBER_UPDATE'
|
||||
| 'GUILD_MEMBER_REMOVE'
|
||||
| 'GUILD_ROLE_CREATE'
|
||||
| 'GUILD_ROLE_UPDATE'
|
||||
| 'GUILD_ROLE_UPDATE_BULK'
|
||||
| 'GUILD_ROLE_DELETE'
|
||||
| 'GUILD_EMOJIS_UPDATE'
|
||||
| 'GUILD_STICKERS_UPDATE'
|
||||
| 'GUILD_BAN_ADD'
|
||||
| 'GUILD_BAN_REMOVE'
|
||||
| 'CHANNEL_CREATE'
|
||||
| 'CHANNEL_UPDATE'
|
||||
| 'CHANNEL_UPDATE_BULK'
|
||||
| 'CHANNEL_DELETE'
|
||||
| 'CHANNEL_PINS_UPDATE'
|
||||
| 'CHANNEL_PINS_ACK'
|
||||
| 'CHANNEL_RECIPIENT_ADD'
|
||||
| 'CHANNEL_RECIPIENT_REMOVE'
|
||||
| 'MESSAGE_CREATE'
|
||||
| 'MESSAGE_UPDATE'
|
||||
| 'MESSAGE_DELETE'
|
||||
| 'MESSAGE_DELETE_BULK'
|
||||
| 'MESSAGE_REACTION_ADD'
|
||||
| 'MESSAGE_REACTION_ADD_MANY'
|
||||
| 'MESSAGE_REACTION_REMOVE'
|
||||
| 'MESSAGE_REACTION_REMOVE_ALL'
|
||||
| 'MESSAGE_REACTION_REMOVE_EMOJI'
|
||||
| 'MESSAGE_ACK'
|
||||
| 'TYPING_START'
|
||||
| 'WEBHOOKS_UPDATE'
|
||||
| 'INVITE_CREATE'
|
||||
| 'INVITE_DELETE'
|
||||
| 'RELATIONSHIP_ADD'
|
||||
| 'RELATIONSHIP_UPDATE'
|
||||
| 'RELATIONSHIP_REMOVE'
|
||||
| 'USER_CONNECTIONS_UPDATE'
|
||||
| 'VOICE_STATE_UPDATE'
|
||||
| 'VOICE_SERVER_UPDATE';
|
||||
22
packages/api/src/constants/InstanceConfig.tsx
Normal file
22
packages/api/src/constants/InstanceConfig.tsx
Normal file
@@ -0,0 +1,22 @@
|
||||
/*
|
||||
* Copyright (C) 2026 Fluxer Contributors
|
||||
*
|
||||
* This file is part of Fluxer.
|
||||
*
|
||||
* Fluxer 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.
|
||||
*
|
||||
* Fluxer 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 Fluxer. If not, see <https://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
export const SNOWFLAKE_RESERVATION_KEY_PREFIX = 'snowflake_reservation:';
|
||||
export const SNOWFLAKE_RESERVATION_REFRESH_CHANNEL = 'snowflake_reservation:refresh';
|
||||
export const FIRST_ADMIN_ACL_CONFIG_KEY = 'first_admin_acl_assigned';
|
||||
20
packages/api/src/constants/IpBan.tsx
Normal file
20
packages/api/src/constants/IpBan.tsx
Normal file
@@ -0,0 +1,20 @@
|
||||
/*
|
||||
* Copyright (C) 2026 Fluxer Contributors
|
||||
*
|
||||
* This file is part of Fluxer.
|
||||
*
|
||||
* Fluxer 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.
|
||||
*
|
||||
* Fluxer 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 Fluxer. If not, see <https://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
export const IP_BAN_REFRESH_CHANNEL = 'ip_bans:refresh';
|
||||
158
packages/api/src/constants/LimitConfig.tsx
Normal file
158
packages/api/src/constants/LimitConfig.tsx
Normal file
@@ -0,0 +1,158 @@
|
||||
/*
|
||||
* Copyright (C) 2026 Fluxer Contributors
|
||||
*
|
||||
* This file is part of Fluxer.
|
||||
*
|
||||
* Fluxer 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.
|
||||
*
|
||||
* Fluxer 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 Fluxer. If not, see <https://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
import type {LimitKey} from '@fluxer/constants/src/LimitConfigMetadata';
|
||||
import {LIMIT_KEYS} from '@fluxer/constants/src/LimitConfigMetadata';
|
||||
import {DEFAULT_FREE_LIMITS, DEFAULT_PREMIUM_LIMITS} from '@fluxer/limits/src/LimitDefaults';
|
||||
import type {LimitConfigSnapshot, LimitRule} from '@fluxer/limits/src/LimitTypes';
|
||||
|
||||
const LIMIT_RULE_IDS = {
|
||||
DEFAULT: 'default',
|
||||
PREMIUM: 'premium',
|
||||
} as const;
|
||||
|
||||
export interface CachedLimitConfig {
|
||||
config: LimitConfigSnapshot;
|
||||
defaultsHash: string;
|
||||
}
|
||||
|
||||
export function getLimitConfigKvKey(selfHosted: boolean): string {
|
||||
return `limit_config:${selfHosted ? 'self_hosted' : 'saas'}`;
|
||||
}
|
||||
|
||||
export const LIMIT_CONFIG_REFRESH_CHANNEL = 'limit-config-refresh';
|
||||
export const LIMIT_CONFIG_REFRESH_LOCK_KEY = 'limit-config-refresh-lock';
|
||||
|
||||
function cloneLimitConfigSnapshot(config: LimitConfigSnapshot): LimitConfigSnapshot {
|
||||
return structuredClone(config);
|
||||
}
|
||||
|
||||
export function sanitizeLimitConfigForInstance(
|
||||
config: LimitConfigSnapshot,
|
||||
options?: {selfHosted?: boolean},
|
||||
): LimitConfigSnapshot {
|
||||
const selfHosted = options?.selfHosted ?? false;
|
||||
const normalized: LimitConfigSnapshot = {
|
||||
traitDefinitions: Array.isArray(config.traitDefinitions) ? config.traitDefinitions : [],
|
||||
rules: Array.isArray(config.rules) ? config.rules : [],
|
||||
};
|
||||
|
||||
if (!selfHosted) {
|
||||
return normalized;
|
||||
}
|
||||
|
||||
const traitDefinitions = normalized.traitDefinitions.filter((t) => t !== 'premium');
|
||||
|
||||
const rules = normalized.rules.filter((rule) => {
|
||||
const traits = rule.filters?.traits ?? [];
|
||||
return !traits.includes('premium');
|
||||
});
|
||||
|
||||
return {
|
||||
traitDefinitions,
|
||||
rules,
|
||||
};
|
||||
}
|
||||
|
||||
export function createDefaultLimitConfig(options?: {selfHosted?: boolean}): LimitConfigSnapshot {
|
||||
const selfHosted = options?.selfHosted ?? false;
|
||||
|
||||
const hostedDefault: LimitConfigSnapshot = {
|
||||
traitDefinitions: selfHosted ? [] : ['premium'],
|
||||
rules: selfHosted
|
||||
? [
|
||||
{
|
||||
id: LIMIT_RULE_IDS.DEFAULT,
|
||||
limits: {...DEFAULT_PREMIUM_LIMITS},
|
||||
},
|
||||
]
|
||||
: [
|
||||
{
|
||||
id: LIMIT_RULE_IDS.PREMIUM,
|
||||
filters: {traits: ['premium']},
|
||||
limits: {...DEFAULT_PREMIUM_LIMITS},
|
||||
},
|
||||
{
|
||||
id: LIMIT_RULE_IDS.DEFAULT,
|
||||
limits: {...DEFAULT_FREE_LIMITS},
|
||||
},
|
||||
],
|
||||
};
|
||||
|
||||
return sanitizeLimitConfigForInstance(cloneLimitConfigSnapshot(hostedDefault), {selfHosted});
|
||||
}
|
||||
|
||||
export function mergeWithCurrentDefaults(
|
||||
stored: LimitConfigSnapshot,
|
||||
options?: {selfHosted?: boolean},
|
||||
): LimitConfigSnapshot {
|
||||
const selfHosted = options?.selfHosted ?? false;
|
||||
const newDefaults = createDefaultLimitConfig({selfHosted});
|
||||
const mergedRules: Array<LimitRule> = [];
|
||||
|
||||
const existingRulesMap = new Map<string, LimitRule>();
|
||||
for (const rule of stored.rules) {
|
||||
existingRulesMap.set(rule.id, rule);
|
||||
}
|
||||
|
||||
for (const defaultRule of newDefaults.rules) {
|
||||
const existingRule = existingRulesMap.get(defaultRule.id);
|
||||
|
||||
if (!existingRule) {
|
||||
mergedRules.push({...defaultRule});
|
||||
continue;
|
||||
}
|
||||
|
||||
const mergedLimits: Partial<Record<LimitKey, number>> = {...defaultRule.limits};
|
||||
const modifiedFields: Array<LimitKey> = [];
|
||||
|
||||
for (const key of LIMIT_KEYS) {
|
||||
const existingValue = existingRule.limits[key];
|
||||
const defaultValue = defaultRule.limits[key];
|
||||
|
||||
if (existingValue !== undefined && existingValue !== defaultValue) {
|
||||
mergedLimits[key] = existingValue;
|
||||
modifiedFields.push(key);
|
||||
}
|
||||
}
|
||||
|
||||
mergedRules.push({
|
||||
id: existingRule.id,
|
||||
filters: existingRule.filters ?? defaultRule.filters,
|
||||
limits: mergedLimits,
|
||||
modifiedFields: modifiedFields.length > 0 ? modifiedFields : undefined,
|
||||
});
|
||||
|
||||
existingRulesMap.delete(defaultRule.id);
|
||||
}
|
||||
|
||||
for (const [, rule] of existingRulesMap) {
|
||||
mergedRules.push({
|
||||
id: rule.id,
|
||||
filters: rule.filters,
|
||||
limits: rule.limits,
|
||||
modifiedFields: rule.modifiedFields ?? (Object.keys(rule.limits) as Array<LimitKey>),
|
||||
});
|
||||
}
|
||||
|
||||
return {
|
||||
traitDefinitions: stored.traitDefinitions,
|
||||
rules: mergedRules,
|
||||
};
|
||||
}
|
||||
Reference in New Issue
Block a user