'use strict'; const JSONBig = require('json-bigint'); const Intents = require('./Intents'); const { randomUA } = require('../util/Constants'); /** * Rate limit data * @typedef {Object} RateLimitData * @property {number} timeout Time until this rate limit ends, in milliseconds * @property {number} limit The maximum amount of requests of this endpoint * @property {string} method The HTTP method of this request * @property {string} path The path of the request relative to the HTTP endpoint * @property {string} route The route of the request relative to the HTTP endpoint * @property {boolean} global Whether this is a global rate limit */ /** * Whether this rate limit should throw an Error * @typedef {Function} RateLimitQueueFilter * @param {RateLimitData} rateLimitData The data of this rate limit * @returns {boolean|Promise} */ /** * @typedef {Function} CacheFactory * @param {Function} manager The manager class the cache is being requested from. * @param {Function} holds The class that the cache will hold. * @returns {Collection} A Collection used to store the cache of the manager. */ /** * Options for a client. * @typedef {Object} ClientOptions * @property {number|number[]|string} [shards] The shard's id to run, or an array of shard ids. If not specified, * the client will spawn {@link ClientOptions#shardCount} shards. If set to `auto`, it will fetch the * recommended amount of shards from Discord and spawn that amount * @property {number} [closeTimeout=5000] The amount of time in milliseconds to wait for the close frame to be received * from the WebSocket. Don't have this too high/low. Its best to have it between 2_000-6_000 ms. * @property {boolean} [checkUpdate=true] Display module update information on the screen * @property {boolean} [readyStatus=true] Sync state with Discord Client * @property {boolean} [patchVoice=false] Automatically patch @discordjs/voice module (support for call) * @property {string} [captchaService=null] Captcha service to use for solving captcha {@link captchaServices} * @property {string} [captchaKey=null] Captcha service key * @property {string} [password=null] Your Discord account password * @property {boolean} [usingNewAttachmentAPI=true] Use new attachment API * @property {string} [interactionTimeout=15000] The amount of time in milliseconds to wait for an interaction response, before rejecting * @property {boolean} [autoRedeemNitro=false] Automaticlly redeems nitro codes * @property {string} [proxy] Proxy to use for the WebSocket + REST connection (proxy-agent uri type) {@link https://www.npmjs.com/package/proxy-agent}. * @property {boolean} [DMSync=false] Automatically synchronize call status (DM and group) at startup (event synchronization) [Warning: May cause rate limit to gateway) * @property {number} [shardCount=1] The total amount of shards used by all processes of this bot * (e.g. recommended shard count, shard count of the ShardingManager) * @property {CacheFactory} [makeCache] Function to create a cache. * You can use your own function, or the {@link Options} class to customize the Collection used for the cache. * Overriding the cache used in `GuildManager`, `ChannelManager`, `GuildChannelManager`, `RoleManager`, * and `PermissionOverwriteManager` is unsupported and **will** break functionality * @property {number} [messageCacheLifetime=0] DEPRECATED: Pass `lifetime` to `sweepers.messages` instead. * How long a message should stay in the cache until it is considered sweepable (in seconds, 0 for forever) * @property {number} [messageSweepInterval=0] DEPRECATED: Pass `interval` to `sweepers.messages` instead. * How frequently to remove messages from the cache that are older than the message cache lifetime * (in seconds, 0 for never) * @property {MessageMentionOptions} [allowedMentions] Default value for {@link MessageOptions#allowedMentions} * @property {number} [invalidRequestWarningInterval=0] The number of invalid REST requests (those that return * 401, 403, or 429) in a 10 minute window between emitted warnings (0 for no warnings). That is, if set to 500, * warnings will be emitted at invalid request number 500, 1000, 1500, and so on. * @property {PartialType[]} [partials=['USER', 'CHANNEL', 'GUILD_MEMBER', 'MESSAGE', 'REACTION', 'GUILD_SCHEDULED_EVENT']] Structures allowed to be partial. This means events can be emitted even when * they're missing all the data for a particular structure. See the "Partial Structures" topic on the * [guide](https://discordjs.guide/popular-topics/partials.html) for some * important usage information, as partials require you to put checks in place when handling data. * @property {number} [restWsBridgeTimeout=5000] Maximum time permitted between REST responses and their * corresponding WebSocket events * @property {number} [restTimeOffset=500] Extra time in milliseconds to wait before continuing to make REST * requests (higher values will reduce rate-limiting errors on bad connections) * @property {number} [restRequestTimeout=15000] Time to wait before cancelling a REST request, in milliseconds * @property {number} [restSweepInterval=60] How frequently to delete inactive request buckets, in seconds * (or 0 for never) * @property {number} [restGlobalRateLimit=0] How many requests to allow sending per second (0 for unlimited, 50 for * the standard global limit used by Discord) * @property {string[]|RateLimitQueueFilter} [rejectOnRateLimit] Decides how rate limits and pre-emptive throttles * should be handled. If this option is an array containing the prefix of the request route (e.g. /channels to match any * route starting with /channels, such as /channels/222197033908436994/messages) or a function returning true, a * {@link RateLimitError} will be thrown. Otherwise the request will be queued for later * @property {number} [retryLimit=1] How many times to retry on 5XX errors * (Infinity for an indefinite amount of retries) * @property {boolean} [failIfNotExists=true] Default value for {@link ReplyMessageOptions#failIfNotExists} * @property {string[]} [userAgentSuffix] An array of additional bot info to be appended to the end of the required * [User Agent](https://discord.com/developers/docs/reference#user-agent) header * @property {PresenceData} [presence={}] Presence data to use upon login * @property {IntentsResolvable} [intents=131071] Intents to enable for this connection (but not using) * @property {number} [waitGuildTimeout=15000] Time in milliseconds that Clients with the GUILDS intent should wait for * @property {number} [messageCreateEventGuildTimeout=100] Time in milliseconds that Clients to register for messages with each guild * missing guilds to be received before starting the bot. If not specified, the default is 15 seconds. * @property {SweeperOptions} [sweepers={}] Options for cache sweeping * @property {WebsocketOptions} [ws] Options for the WebSocket * @property {HTTPOptions} [http] HTTP options * @property {CustomCaptchaSolver} [captchaSolver] Function to solve a captcha (custom) */ /** * Function to solve a captcha * @typedef {function} CustomCaptchaSolver * @param {Captcha} captcha The captcha to solve * @param {string} userAgent The user agent to use for the request * @returns {Promise} hcaptcha token */ /** * Options for {@link Sweepers} defining the behavior of cache sweeping * @typedef {Object} SweeperOptions */ /** * Options for sweeping a single type of item from cache * @typedef {Object} SweepOptions * @property {number} interval The interval (in seconds) at which to perform sweeping of the item * @property {number} [lifetime] How long an item should stay in cache until it is considered sweepable. * This property is only valid for the `invites`, `messages`, and `threads` keys. The `filter` property * is mutually exclusive to this property and takes priority * @property {GlobalSweepFilter} filter The function used to determine the function passed to the sweep method * This property is optional when the key is `invites`, `messages`, or `threads` and `lifetime` is set */ /** * WebSocket options (these are left as snake_case to match the API) * @typedef {Object} WebsocketOptions * @property {boolean} [compress=false] Whether to compress data sent on the connection * @property {WebSocketProperties} [properties] Properties to identify the client with */ /** * HTTPS Agent options. * @typedef {Object} AgentOptions * @see {@link https://nodejs.org/api/https.html#https_class_https_agent} * @see {@link https://nodejs.org/api/http.html#http_new_agent_options} */ /** * HTTP options * @typedef {Object} HTTPOptions * @property {number} [version=9] API version to use * @property {AgentOptions} [agent={}] HTTPS Agent options * @property {string} [api='https://discord.com/api'] Base URL of the API * @property {string} [cdn='https://cdn.discordapp.com'] Base URL of the CDN * @property {string} [invite='https://discord.gg'] Base URL of invites * @property {string} [template='https://discord.new'] Base URL of templates * @property {Object} [headers] Additional headers to send for all API requests * @property {string} [scheduledEvent='https://discord.com/events'] Base URL of guild scheduled events */ /** * Contains various utilities for client options. */ class Options extends null { /** * The default client options. * @returns {ClientOptions} */ static createDefault() { return { jsonTransformer: object => JSONBig.stringify(object), captchaSolver: captcha => Promise.reject(new Error('CAPTCHA_SOLVER_NOT_IMPLEMENTED', captcha)), closeTimeout: 5_000, checkUpdate: true, readyStatus: true, autoRedeemNitro: false, captchaService: '', captchaKey: null, DMSync: false, patchVoice: false, password: null, usingNewAttachmentAPI: true, interactionTimeout: 15_000, waitGuildTimeout: 15_000, messageCreateEventGuildTimeout: 100, shardCount: 1, makeCache: this.cacheWithLimits(this.defaultMakeCacheSettings), messageCacheLifetime: 0, messageSweepInterval: 0, invalidRequestWarningInterval: 0, intents: Intents.ALL, partials: ['USER', 'CHANNEL', 'GUILD_MEMBER', 'MESSAGE', 'REACTION', 'GUILD_SCHEDULED_EVENT'], // Enable the partials restWsBridgeTimeout: 5_000, restRequestTimeout: 15_000, restGlobalRateLimit: 0, retryLimit: 1, restTimeOffset: 500, restSweepInterval: 60, failIfNotExists: false, userAgentSuffix: [], presence: { status: 'online', since: 0, activities: [], afk: false }, sweepers: {}, proxy: '', ws: { compress: false, // https://discord-user-api.cf/api/v1/properties/web properties: { os: 'Windows', browser: 'Chrome', device: '', system_locale: 'en-US', browser_version: '108.0.0.0', os_version: '10', referrer: '', referring_domain: '', referrer_current: '', referring_domain_current: '', release_channel: 'stable', client_build_number: 165485, client_event_source: null, }, // ! capabilities: 4093, version: 9, client_state: { api_code_version: 0, guild_versions: {}, highest_last_message_id: '0', read_state_version: 0, user_guild_settings_version: -1, user_settings_version: -1, private_channels_version: '0', }, }, http: { agent: {}, headers: { 'User-Agent': randomUA(), }, version: 9, api: 'https://discord.com/api', cdn: 'https://cdn.discordapp.com', invite: 'https://discord.gg', template: 'https://discord.new', scheduledEvent: 'https://discord.com/events', }, }; } /** * Create a cache factory using predefined settings to sweep or limit. * @param {Object} [settings={}] Settings passed to the relevant constructor. * If no setting is provided for a manager, it uses Collection. * If a number is provided for a manager, it uses that number as the max size for a LimitedCollection. * If LimitedCollectionOptions are provided for a manager, it uses those settings to form a LimitedCollection. * @returns {CacheFactory} * @example * // Store up to 200 messages per channel and discard archived threads if they were archived more than 4 hours ago. * // Note archived threads will remain in the guild and client caches with these settings * Options.cacheWithLimits({ * MessageManager: 200, * ThreadManager: { * sweepInterval: 3600, * sweepFilter: LimitedCollection.filterByLifetime({ * getComparisonTimestamp: e => e.archiveTimestamp, * excludeFromSweep: e => !e.archived, * }), * }, * }); * @example * // Sweep messages every 5 minutes, removing messages that have not been edited or created in the last 30 minutes * Options.cacheWithLimits({ * // Keep default thread sweeping behavior * ...Options.defaultMakeCacheSettings, * // Override MessageManager * MessageManager: { * sweepInterval: 300, * sweepFilter: LimitedCollection.filterByLifetime({ * lifetime: 1800, * getComparisonTimestamp: e => e.editedTimestamp ?? e.createdTimestamp, * }) * } * }); */ static cacheWithLimits(settings = {}) { const { Collection } = require('@discordjs/collection'); const LimitedCollection = require('./LimitedCollection'); return manager => { const setting = settings[manager.name]; /* eslint-disable-next-line eqeqeq */ if (setting == null) { return new Collection(); } if (typeof setting === 'number') { if (setting === Infinity) { return new Collection(); } return new LimitedCollection({ maxSize: setting }); } /* eslint-disable eqeqeq */ const noSweeping = setting.sweepFilter == null || setting.sweepInterval == null || setting.sweepInterval <= 0 || setting.sweepInterval === Infinity; const noLimit = setting.maxSize == null || setting.maxSize === Infinity; /* eslint-enable eqeqeq */ if (noSweeping && noLimit) { return new Collection(); } return new LimitedCollection(setting); }; } /** * Create a cache factory that always caches everything. * @returns {CacheFactory} */ static cacheEverything() { const { Collection } = require('@discordjs/collection'); return () => new Collection(); } /** * The default settings passed to {@link Options.cacheWithLimits}. * The caches that this changes are: * * `MessageManager` - Limit to 200 messages * * `ChannelManager` - Sweep archived threads * * `GuildChannelManager` - Sweep archived threads * * `ThreadManager` - Sweep archived threads * If you want to keep default behavior and add on top of it you can use this object and add on to it, e.g. * `makeCache: Options.cacheWithLimits({ ...Options.defaultMakeCacheSettings, ReactionManager: 0 })` * @type {Object} */ static get defaultMakeCacheSettings() { return { MessageManager: 200, /* ChannelManager: { sweepInterval: 3600, sweepFilter: require('./Util').archivedThreadSweepFilter(), }, GuildChannelManager: { sweepInterval: 3600, sweepFilter: require('./Util').archivedThreadSweepFilter(), }, ThreadManager: { sweepInterval: 3600, sweepFilter: require('./Util').archivedThreadSweepFilter(), }, */ }; } } /** * The default settings passed to {@link Options.sweepers} (for v14). * The sweepers that this changes are: * * `threads` - Sweep archived threads every hour, removing those archived more than 4 hours ago * If you want to keep default behavior and add on top of it you can use this object and add on to it, e.g. * `sweepers: { ...Options.defaultSweeperSettings, messages: { interval: 300, lifetime: 600 } })` * @type {SweeperOptions} */ Options.defaultSweeperSettings = { threads: { interval: 3600, lifetime: 14400, }, }; module.exports = Options;