2022-03-19 10:37:45 +00:00
|
|
|
'use strict';
|
|
|
|
|
|
|
|
const { Buffer } = require('node:buffer');
|
2022-03-24 10:55:32 +00:00
|
|
|
const { setTimeout } = require('node:timers');
|
2022-03-19 10:37:45 +00:00
|
|
|
const { Collection } = require('@discordjs/collection');
|
2023-03-29 11:38:43 +00:00
|
|
|
require('lodash.permutations');
|
|
|
|
const _ = require('lodash');
|
2022-03-19 10:37:45 +00:00
|
|
|
const CachedManager = require('./CachedManager');
|
|
|
|
const { Error, TypeError, RangeError } = require('../errors');
|
|
|
|
const BaseGuildVoiceChannel = require('../structures/BaseGuildVoiceChannel');
|
|
|
|
const { GuildMember } = require('../structures/GuildMember');
|
|
|
|
const { Role } = require('../structures/Role');
|
2022-03-24 10:55:32 +00:00
|
|
|
const { Events, Opcodes } = require('../util/Constants');
|
2023-01-10 11:08:05 +00:00
|
|
|
const { PartialTypes } = require('../util/Constants');
|
2022-08-04 12:26:16 +00:00
|
|
|
const DataResolver = require('../util/DataResolver');
|
2023-02-19 05:07:03 +00:00
|
|
|
const GuildMemberFlags = require('../util/GuildMemberFlags');
|
2022-03-24 10:55:32 +00:00
|
|
|
const SnowflakeUtil = require('../util/SnowflakeUtil');
|
2022-03-19 10:37:45 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Manages API methods for GuildMembers and stores their cache.
|
|
|
|
* @extends {CachedManager}
|
|
|
|
*/
|
|
|
|
class GuildMemberManager extends CachedManager {
|
|
|
|
constructor(guild, iterable) {
|
|
|
|
super(guild.client, GuildMember, iterable);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The guild this manager belongs to
|
|
|
|
* @type {Guild}
|
|
|
|
*/
|
|
|
|
this.guild = guild;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The cache of this Manager
|
|
|
|
* @type {Collection<Snowflake, GuildMember>}
|
|
|
|
* @name GuildMemberManager#cache
|
|
|
|
*/
|
|
|
|
|
|
|
|
_add(data, cache = true) {
|
|
|
|
return super._add(data, cache, { id: data.user.id, extras: [this.guild] });
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Data that resolves to give a GuildMember object. This can be:
|
|
|
|
* * A GuildMember object
|
|
|
|
* * A User resolvable
|
|
|
|
* @typedef {GuildMember|UserResolvable} GuildMemberResolvable
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Resolves a {@link GuildMemberResolvable} to a {@link GuildMember} object.
|
|
|
|
* @param {GuildMemberResolvable} member The user that is part of the guild
|
|
|
|
* @returns {?GuildMember}
|
|
|
|
*/
|
|
|
|
resolve(member) {
|
|
|
|
const memberResolvable = super.resolve(member);
|
|
|
|
if (memberResolvable) return memberResolvable;
|
|
|
|
const userResolvable = this.client.users.resolveId(member);
|
|
|
|
if (userResolvable) return super.resolve(userResolvable);
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Resolves a {@link GuildMemberResolvable} to a member id.
|
|
|
|
* @param {GuildMemberResolvable} member The user that is part of the guild
|
|
|
|
* @returns {?Snowflake}
|
|
|
|
*/
|
|
|
|
resolveId(member) {
|
|
|
|
const memberResolvable = super.resolveId(member);
|
|
|
|
if (memberResolvable) return memberResolvable;
|
|
|
|
const userResolvable = this.client.users.resolveId(member);
|
|
|
|
return this.cache.has(userResolvable) ? userResolvable : null;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Options used to add a user to a guild using OAuth2.
|
|
|
|
* @typedef {Object} AddGuildMemberOptions
|
|
|
|
* @property {string} accessToken An OAuth2 access token for the user with the `guilds.join` scope granted to the
|
|
|
|
* bot's application
|
|
|
|
* @property {string} [nick] The nickname to give to the member (requires `MANAGE_NICKNAMES`)
|
|
|
|
* @property {Collection<Snowflake, Role>|RoleResolvable[]} [roles] The roles to add to the member
|
|
|
|
* (requires `MANAGE_ROLES`)
|
|
|
|
* @property {boolean} [mute] Whether the member should be muted (requires `MUTE_MEMBERS`)
|
|
|
|
* @property {boolean} [deaf] Whether the member should be deafened (requires `DEAFEN_MEMBERS`)
|
|
|
|
* @property {boolean} [force] Whether to skip the cache check and call the API directly
|
|
|
|
* @property {boolean} [fetchWhenExisting=true] Whether to fetch the user if not cached and already a member
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Adds a user to the guild using OAuth2. Requires the `CREATE_INSTANT_INVITE` permission.
|
|
|
|
* @param {UserResolvable} user The user to add to the guild
|
|
|
|
* @param {AddGuildMemberOptions} options Options for adding the user to the guild
|
|
|
|
* @returns {Promise<GuildMember|null>}
|
|
|
|
*/
|
|
|
|
async add(user, options) {
|
|
|
|
const userId = this.client.users.resolveId(user);
|
|
|
|
if (!userId) throw new TypeError('INVALID_TYPE', 'user', 'UserResolvable');
|
|
|
|
if (!options.force) {
|
|
|
|
const cachedUser = this.cache.get(userId);
|
|
|
|
if (cachedUser) return cachedUser;
|
|
|
|
}
|
|
|
|
const resolvedOptions = {
|
|
|
|
access_token: options.accessToken,
|
|
|
|
nick: options.nick,
|
|
|
|
mute: options.mute,
|
|
|
|
deaf: options.deaf,
|
|
|
|
};
|
|
|
|
if (options.roles) {
|
|
|
|
if (!Array.isArray(options.roles) && !(options.roles instanceof Collection)) {
|
|
|
|
throw new TypeError('INVALID_TYPE', 'options.roles', 'Array or Collection of Roles or Snowflakes', true);
|
|
|
|
}
|
|
|
|
const resolvedRoles = [];
|
|
|
|
for (const role of options.roles.values()) {
|
|
|
|
const resolvedRole = this.guild.roles.resolveId(role);
|
|
|
|
if (!resolvedRole) throw new TypeError('INVALID_ELEMENT', 'Array or Collection', 'options.roles', role);
|
|
|
|
resolvedRoles.push(resolvedRole);
|
|
|
|
}
|
|
|
|
resolvedOptions.roles = resolvedRoles;
|
|
|
|
}
|
2022-03-24 10:55:32 +00:00
|
|
|
const data = await this.client.api.guilds(this.guild.id).members(userId).put({ data: resolvedOptions });
|
2022-03-19 10:37:45 +00:00
|
|
|
// Data is an empty buffer if the member is already part of the guild.
|
|
|
|
return data instanceof Buffer ? (options.fetchWhenExisting === false ? null : this.fetch(userId)) : this._add(data);
|
|
|
|
}
|
|
|
|
|
2023-01-10 11:08:05 +00:00
|
|
|
/**
|
|
|
|
* The client user as a GuildMember of this guild
|
|
|
|
* @type {?GuildMember}
|
|
|
|
* @readonly
|
|
|
|
*/
|
|
|
|
get me() {
|
|
|
|
return (
|
|
|
|
this.resolve(this.client.user.id) ??
|
|
|
|
(this.client.options.partials.includes(PartialTypes.GUILD_MEMBER)
|
|
|
|
? this._add({ user: { id: this.client.user.id } }, true)
|
|
|
|
: null)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2022-03-19 10:37:45 +00:00
|
|
|
/**
|
|
|
|
* Options used to fetch a single member from a guild.
|
|
|
|
* @typedef {BaseFetchOptions} FetchMemberOptions
|
|
|
|
* @property {UserResolvable} user The user to fetch
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Options used to fetch multiple members from a guild.
|
|
|
|
* @typedef {Object} FetchMembersOptions
|
|
|
|
* @property {UserResolvable|UserResolvable[]} user The user(s) to fetch
|
|
|
|
* @property {?string} query Limit fetch to members with similar usernames
|
|
|
|
* @property {number} [limit=0] Maximum number of members to request
|
|
|
|
* @property {boolean} [withPresences=false] Whether or not to include the presences
|
|
|
|
* @property {number} [time=120e3] Timeout for receipt of members
|
|
|
|
* @property {?string} nonce Nonce for this request (32 characters max - default to base 16 now timestamp)
|
|
|
|
* @property {boolean} [force=false] Whether to skip the cache check and request the API
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Fetches member(s) from Discord, even if they're offline.
|
|
|
|
* @param {UserResolvable|FetchMemberOptions|FetchMembersOptions} [options] If a UserResolvable, the user to fetch.
|
|
|
|
* If undefined, fetches all members.
|
|
|
|
* If a query, it limits the results to users with similar usernames.
|
|
|
|
* @returns {Promise<GuildMember|Collection<Snowflake, GuildMember>>}
|
|
|
|
* @example
|
|
|
|
* // Fetch all members from a guild
|
|
|
|
* guild.members.fetch()
|
|
|
|
* .then(console.log)
|
|
|
|
* .catch(console.error);
|
|
|
|
* @example
|
|
|
|
* // Fetch a single member
|
|
|
|
* guild.members.fetch('66564597481480192')
|
|
|
|
* .then(console.log)
|
|
|
|
* .catch(console.error);
|
|
|
|
* @example
|
|
|
|
* // Fetch a single member without checking cache
|
|
|
|
* guild.members.fetch({ user, force: true })
|
|
|
|
* .then(console.log)
|
|
|
|
* .catch(console.error)
|
|
|
|
* @example
|
|
|
|
* // Fetch a single member without caching
|
|
|
|
* guild.members.fetch({ user, cache: false })
|
|
|
|
* .then(console.log)
|
|
|
|
* .catch(console.error);
|
|
|
|
* @example
|
|
|
|
* // Fetch by an array of users including their presences
|
|
|
|
* guild.members.fetch({ user: ['66564597481480192', '191615925336670208'], withPresences: true })
|
|
|
|
* .then(console.log)
|
|
|
|
* .catch(console.error);
|
|
|
|
* @example
|
|
|
|
* // Fetch by query
|
|
|
|
* guild.members.fetch({ query: 'hydra', limit: 1 })
|
|
|
|
* .then(console.log)
|
|
|
|
* .catch(console.error);
|
2022-11-18 12:23:12 +00:00
|
|
|
* @see {@link https://github.com/aiko-chan-ai/discord.js-selfbot-v13/blob/main/Document/FetchGuildMember.md}
|
2022-03-19 10:37:45 +00:00
|
|
|
*/
|
|
|
|
fetch(options) {
|
2023-02-23 05:02:36 +00:00
|
|
|
if (!options || (typeof options === 'object' && !('user' in options) && !('query' in options))) {
|
2022-10-23 11:51:55 +00:00
|
|
|
if (
|
2023-01-10 11:08:05 +00:00
|
|
|
this.guild.members.me.permissions.has('KICK_MEMBERS') ||
|
|
|
|
this.guild.members.me.permissions.has('BAN_MEMBERS') ||
|
|
|
|
this.guild.members.me.permissions.has('MANAGE_ROLES')
|
2022-10-23 11:51:55 +00:00
|
|
|
) {
|
|
|
|
return this._fetchMany();
|
2023-08-03 13:53:22 +00:00
|
|
|
} else if (this.guild.memberCount <= 10000) {
|
2023-07-17 14:05:58 +00:00
|
|
|
return this.fetchByMemberSafety();
|
2023-08-03 13:53:22 +00:00
|
|
|
} else {
|
|
|
|
// NOTE: This is a very slow method, and can take up to 999+ minutes to complete.
|
2023-09-12 06:38:12 +00:00
|
|
|
return this.fetchBruteforce({
|
2022-10-23 11:51:55 +00:00
|
|
|
delay: 50,
|
|
|
|
skipWarn: true,
|
2023-03-30 05:16:12 +00:00
|
|
|
depth: 1,
|
2022-10-23 11:51:55 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
2022-03-19 10:37:45 +00:00
|
|
|
const user = this.client.users.resolveId(options);
|
|
|
|
if (user) return this._fetchSingle({ user, cache: true });
|
|
|
|
if (options.user) {
|
|
|
|
if (Array.isArray(options.user)) {
|
|
|
|
options.user = options.user.map(u => this.client.users.resolveId(u));
|
|
|
|
return this._fetchMany(options);
|
|
|
|
} else {
|
|
|
|
options.user = this.client.users.resolveId(options.user);
|
|
|
|
}
|
|
|
|
if (!options.limit && !options.withPresences) return this._fetchSingle(options);
|
|
|
|
}
|
|
|
|
return this._fetchMany(options);
|
|
|
|
}
|
|
|
|
|
2023-01-10 11:08:05 +00:00
|
|
|
/**
|
|
|
|
* Fetches the client user as a GuildMember of the guild.
|
|
|
|
* @param {BaseFetchOptions} [options] The options for fetching the member
|
|
|
|
* @returns {Promise<GuildMember>}
|
|
|
|
*/
|
|
|
|
fetchMe(options) {
|
|
|
|
return this.fetch({ ...options, user: this.client.user.id });
|
|
|
|
}
|
|
|
|
|
2022-03-19 10:37:45 +00:00
|
|
|
/**
|
|
|
|
* Options used for searching guild members.
|
|
|
|
* @typedef {Object} GuildSearchMembersOptions
|
|
|
|
* @property {string} query Filter members whose username or nickname start with this query
|
2022-03-24 10:55:32 +00:00
|
|
|
* @property {number} [limit=1] Maximum number of members to search
|
2022-03-19 10:37:45 +00:00
|
|
|
* @property {boolean} [cache=true] Whether or not to cache the fetched member(s)
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Searches for members in the guild based on a query.
|
|
|
|
* @param {GuildSearchMembersOptions} options Options for searching members
|
|
|
|
* @returns {Promise<Collection<Snowflake, GuildMember>>}
|
|
|
|
*/
|
2022-03-24 10:55:32 +00:00
|
|
|
async search({ query, limit = 1, cache = true } = {}) {
|
|
|
|
const data = await this.client.api.guilds(this.guild.id).members.search.get({ query: { query, limit } });
|
2022-03-19 10:37:45 +00:00
|
|
|
return data.reduce((col, member) => col.set(member.user.id, this._add(member, cache)), new Collection());
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Options used for listing guild members.
|
|
|
|
* @typedef {Object} GuildListMembersOptions
|
|
|
|
* @property {Snowflake} [after] Limit fetching members to those with an id greater than the supplied id
|
2022-03-24 10:55:32 +00:00
|
|
|
* @property {number} [limit=1] Maximum number of members to list
|
2022-03-19 10:37:45 +00:00
|
|
|
* @property {boolean} [cache=true] Whether or not to cache the fetched member(s)
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Lists up to 1000 members of the guild.
|
|
|
|
* @param {GuildListMembersOptions} [options] Options for listing members
|
|
|
|
* @returns {Promise<Collection<Snowflake, GuildMember>>}
|
|
|
|
*/
|
2022-03-24 10:55:32 +00:00
|
|
|
async list({ after, limit = 1, cache = true } = {}) {
|
|
|
|
const data = await this.client.api.guilds(this.guild.id).members.get({ query: { after, limit } });
|
2022-03-19 10:37:45 +00:00
|
|
|
return data.reduce((col, member) => col.set(member.user.id, this._add(member, cache)), new Collection());
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The data for editing a guild member.
|
|
|
|
* @typedef {Object} GuildMemberEditData
|
|
|
|
* @property {?string} [nick] The nickname to set for the member
|
|
|
|
* @property {Collection<Snowflake, Role>|RoleResolvable[]} [roles] The roles or role ids to apply
|
|
|
|
* @property {boolean} [mute] Whether or not the member should be muted
|
|
|
|
* @property {boolean} [deaf] Whether or not the member should be deafened
|
|
|
|
* @property {GuildVoiceChannelResolvable|null} [channel] Channel to move the member to
|
|
|
|
* (if they are connected to voice), or `null` if you want to disconnect them from voice
|
|
|
|
* @property {DateResolvable|null} [communicationDisabledUntil] The date or timestamp
|
|
|
|
* for the member's communication to be disabled until. Provide `null` to enable communication again.
|
2023-02-19 05:07:03 +00:00
|
|
|
* @property {GuildMemberFlagsResolvable} [flags] The flags to set for the member
|
2022-08-04 12:26:16 +00:00
|
|
|
* @property {?(BufferResolvable|Base64Resolvable)} [avatar] The new guild avatar
|
|
|
|
* @property {?(BufferResolvable|Base64Resolvable)} [banner] The new guild banner
|
|
|
|
* @property {?string} [bio] The new guild about me
|
2022-03-19 10:37:45 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Edits a member of the guild.
|
|
|
|
* <info>The user must be a member of the guild</info>
|
|
|
|
* @param {UserResolvable} user The member to edit
|
|
|
|
* @param {GuildMemberEditData} data The data to edit the member with
|
|
|
|
* @param {string} [reason] Reason for editing this user
|
|
|
|
* @returns {Promise<GuildMember>}
|
|
|
|
*/
|
|
|
|
async edit(user, data, reason) {
|
|
|
|
const id = this.client.users.resolveId(user);
|
|
|
|
if (!id) throw new TypeError('INVALID_TYPE', 'user', 'UserResolvable');
|
|
|
|
|
|
|
|
// Clone the data object for immutability
|
|
|
|
const _data = { ...data };
|
|
|
|
if (_data.channel) {
|
|
|
|
_data.channel = this.guild.channels.resolve(_data.channel);
|
|
|
|
if (!(_data.channel instanceof BaseGuildVoiceChannel)) {
|
|
|
|
throw new Error('GUILD_VOICE_CHANNEL_RESOLVE');
|
|
|
|
}
|
|
|
|
_data.channel_id = _data.channel.id;
|
|
|
|
_data.channel = undefined;
|
|
|
|
} else if (_data.channel === null) {
|
|
|
|
_data.channel_id = null;
|
|
|
|
_data.channel = undefined;
|
|
|
|
}
|
|
|
|
_data.roles &&= _data.roles.map(role => (role instanceof Role ? role.id : role));
|
|
|
|
|
|
|
|
_data.communication_disabled_until =
|
2022-03-24 10:55:32 +00:00
|
|
|
_data.communicationDisabledUntil && new Date(_data.communicationDisabledUntil).toISOString();
|
2022-03-19 10:37:45 +00:00
|
|
|
|
2023-02-19 05:07:03 +00:00
|
|
|
_data.flags = _data.flags && GuildMemberFlags.resolve(_data.flags);
|
|
|
|
|
2022-08-04 12:26:16 +00:00
|
|
|
// Avatar, banner, bio
|
|
|
|
if (typeof _data.avatar !== 'undefined') {
|
|
|
|
_data.avatar = await DataResolver.resolveImage(_data.avatar);
|
|
|
|
}
|
|
|
|
if (typeof _data.banner !== 'undefined') {
|
|
|
|
_data.banner = await DataResolver.resolveImage(_data.banner);
|
|
|
|
}
|
|
|
|
|
2022-03-19 10:37:45 +00:00
|
|
|
let endpoint = this.client.api.guilds(this.guild.id);
|
|
|
|
if (id === this.client.user.id) {
|
|
|
|
const keys = Object.keys(data);
|
2022-08-04 12:26:16 +00:00
|
|
|
if (keys.length === 1 && ['nick', 'avatar', 'banner', 'bio'].includes(keys[0])) {
|
|
|
|
endpoint = endpoint.members('@me');
|
|
|
|
} else {
|
|
|
|
endpoint = endpoint.members(id);
|
|
|
|
}
|
2022-03-19 10:37:45 +00:00
|
|
|
} else {
|
|
|
|
endpoint = endpoint.members(id);
|
|
|
|
}
|
|
|
|
const d = await endpoint.patch({ data: _data, reason });
|
|
|
|
|
|
|
|
const clone = this.cache.get(id)?._clone();
|
|
|
|
clone?._patch(d);
|
|
|
|
return clone ?? this._add(d, false);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Options used for pruning guild members.
|
|
|
|
* <info>It's recommended to set {@link GuildPruneMembersOptions#count options.count}
|
|
|
|
* to `false` for large guilds.</info>
|
|
|
|
* @typedef {Object} GuildPruneMembersOptions
|
2022-03-24 10:55:32 +00:00
|
|
|
* @property {number} [days=7] Number of days of inactivity required to kick
|
2022-03-19 10:37:45 +00:00
|
|
|
* @property {boolean} [dry=false] Get the number of users that will be kicked, without actually kicking them
|
2022-03-24 10:55:32 +00:00
|
|
|
* @property {boolean} [count=true] Whether or not to return the number of users that have been kicked.
|
2022-03-19 10:37:45 +00:00
|
|
|
* @property {RoleResolvable[]} [roles] Array of roles to bypass the "...and no roles" constraint when pruning
|
|
|
|
* @property {string} [reason] Reason for this prune
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Prunes members from the guild based on how long they have been inactive.
|
|
|
|
* @param {GuildPruneMembersOptions} [options] Options for pruning
|
|
|
|
* @returns {Promise<number|null>} The number of members that were/will be kicked
|
|
|
|
* @example
|
|
|
|
* // See how many members will be pruned
|
|
|
|
* guild.members.prune({ dry: true })
|
|
|
|
* .then(pruned => console.log(`This will prune ${pruned} people!`))
|
|
|
|
* .catch(console.error);
|
|
|
|
* @example
|
|
|
|
* // Actually prune the members
|
|
|
|
* guild.members.prune({ days: 1, reason: 'too many people!' })
|
|
|
|
* .then(pruned => console.log(`I just pruned ${pruned} people!`))
|
|
|
|
* .catch(console.error);
|
|
|
|
* @example
|
|
|
|
* // Include members with a specified role
|
|
|
|
* guild.members.prune({ days: 7, roles: ['657259391652855808'] })
|
|
|
|
* .then(pruned => console.log(`I just pruned ${pruned} people!`))
|
|
|
|
* .catch(console.error);
|
|
|
|
*/
|
2022-03-24 10:55:32 +00:00
|
|
|
async prune({ days = 7, dry = false, count: compute_prune_count = true, roles = [], reason } = {}) {
|
2022-03-19 10:37:45 +00:00
|
|
|
if (typeof days !== 'number') throw new TypeError('PRUNE_DAYS_TYPE');
|
|
|
|
|
|
|
|
const query = { days };
|
|
|
|
const resolvedRoles = [];
|
|
|
|
|
|
|
|
for (const role of roles) {
|
|
|
|
const resolvedRole = this.guild.roles.resolveId(role);
|
|
|
|
if (!resolvedRole) {
|
|
|
|
throw new TypeError('INVALID_ELEMENT', 'Array', 'options.roles', role);
|
|
|
|
}
|
|
|
|
resolvedRoles.push(resolvedRole);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (resolvedRoles.length) {
|
|
|
|
query.include_roles = dry ? resolvedRoles.join(',') : resolvedRoles;
|
|
|
|
}
|
|
|
|
|
|
|
|
const endpoint = this.client.api.guilds(this.guild.id).prune;
|
|
|
|
|
|
|
|
const { pruned } = await (dry
|
2022-03-24 10:55:32 +00:00
|
|
|
? endpoint.get({ query, reason })
|
|
|
|
: endpoint.post({ data: { ...query, compute_prune_count }, reason }));
|
2022-03-19 10:37:45 +00:00
|
|
|
|
|
|
|
return pruned;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Kicks a user from the guild.
|
|
|
|
* <info>The user must be a member of the guild</info>
|
|
|
|
* @param {UserResolvable} user The member to kick
|
|
|
|
* @param {string} [reason] Reason for kicking
|
|
|
|
* @returns {Promise<GuildMember|User|Snowflake>} Result object will be resolved as specifically as possible.
|
|
|
|
* If the GuildMember cannot be resolved, the User will instead be attempted to be resolved. If that also cannot
|
|
|
|
* be resolved, the user's id will be the result.
|
|
|
|
* @example
|
|
|
|
* // Kick a user by id (or with a user/guild member object)
|
|
|
|
* guild.members.kick('84484653687267328')
|
2022-05-14 08:06:15 +00:00
|
|
|
* .then(kickInfo => console.log(`Kicked ${kickInfo.user?.tag ?? kickInfo.tag ?? kickInfo}`))
|
2022-03-19 10:37:45 +00:00
|
|
|
* .catch(console.error);
|
|
|
|
*/
|
|
|
|
async kick(user, reason) {
|
|
|
|
const id = this.client.users.resolveId(user);
|
|
|
|
if (!id) return Promise.reject(new TypeError('INVALID_TYPE', 'user', 'UserResolvable'));
|
|
|
|
|
2022-03-24 10:55:32 +00:00
|
|
|
await this.client.api.guilds(this.guild.id).members(id).delete({ reason });
|
2022-03-19 10:37:45 +00:00
|
|
|
|
|
|
|
return this.resolve(user) ?? this.client.users.resolve(user) ?? id;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Bans a user from the guild.
|
|
|
|
* @param {UserResolvable} user The user to ban
|
|
|
|
* @param {BanOptions} [options] Options for the ban
|
|
|
|
* @returns {Promise<GuildMember|User|Snowflake>} Result object will be resolved as specifically as possible.
|
|
|
|
* If the GuildMember cannot be resolved, the User will instead be attempted to be resolved. If that also cannot
|
|
|
|
* be resolved, the user id will be the result.
|
|
|
|
* Internally calls the GuildBanManager#create method.
|
|
|
|
* @example
|
|
|
|
* // Ban a user by id (or with a user/guild member object)
|
|
|
|
* guild.members.ban('84484653687267328')
|
2022-05-14 08:06:15 +00:00
|
|
|
* .then(banInfo => console.log(`Banned ${banInfo.user?.tag ?? banInfo.tag ?? banInfo}`))
|
2022-03-19 10:37:45 +00:00
|
|
|
* .catch(console.error);
|
|
|
|
*/
|
2022-09-04 11:31:39 +00:00
|
|
|
ban(user, options) {
|
2022-03-19 10:37:45 +00:00
|
|
|
return this.guild.bans.create(user, options);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Unbans a user from the guild. Internally calls the {@link GuildBanManager#remove} method.
|
|
|
|
* @param {UserResolvable} user The user to unban
|
|
|
|
* @param {string} [reason] Reason for unbanning user
|
2022-05-14 08:06:15 +00:00
|
|
|
* @returns {Promise<?User>} The user that was unbanned
|
2022-03-19 10:37:45 +00:00
|
|
|
* @example
|
|
|
|
* // Unban a user by id (or with a user/guild member object)
|
|
|
|
* guild.members.unban('84484653687267328')
|
|
|
|
* .then(user => console.log(`Unbanned ${user.username} from ${guild.name}`))
|
|
|
|
* .catch(console.error);
|
|
|
|
*/
|
|
|
|
unban(user, reason) {
|
|
|
|
return this.guild.bans.remove(user, reason);
|
|
|
|
}
|
|
|
|
|
|
|
|
async _fetchSingle({ user, cache, force = false }) {
|
|
|
|
if (!force) {
|
|
|
|
const existing = this.cache.get(user);
|
|
|
|
if (existing && !existing.partial) return existing;
|
|
|
|
}
|
|
|
|
|
|
|
|
const data = await this.client.api.guilds(this.guild.id).members(user).get();
|
|
|
|
return this._add(data, cache);
|
|
|
|
}
|
|
|
|
|
2022-08-31 04:04:42 +00:00
|
|
|
/**
|
|
|
|
* Options used to fetch multiple members from a guild.
|
|
|
|
* @typedef {Object} BruteforceOptions
|
2022-09-11 15:54:07 +00:00
|
|
|
* @property {number} [limit=100] Maximum number of members per request
|
2022-08-31 04:04:42 +00:00
|
|
|
* @property {number} [delay=500] Timeout for new requests in ms
|
2023-03-29 11:38:43 +00:00
|
|
|
* @property {number} [depth=1] Permutations length
|
2022-08-31 04:04:42 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Fetches multiple members from the guild.
|
|
|
|
* @param {BruteforceOptions} options Options for the bruteforce
|
|
|
|
* @returns {Collection<Snowflake, GuildMember>} (All) members in the guild
|
2022-08-31 04:17:22 +00:00
|
|
|
* @see https://github.com/aiko-chan-ai/discord.js-selfbot-v13/blob/main/Document/FetchGuildMember.md
|
2022-08-31 04:04:42 +00:00
|
|
|
* @example
|
2022-08-31 04:24:16 +00:00
|
|
|
* guild.members.fetchBruteforce()
|
2022-08-31 04:04:42 +00:00
|
|
|
* .then(members => console.log(`Fetched ${members.size} members`))
|
|
|
|
* .catch(console.error);
|
|
|
|
*/
|
2022-08-31 04:25:11 +00:00
|
|
|
fetchBruteforce(options = {}) {
|
2023-03-30 05:16:12 +00:00
|
|
|
const defaultQuery = 'abcdefghijklmnopqrstuvwxyz0123456789!"#$%&\'()*+,-./:;<=>?@[]^_`{|}~ ';
|
2023-03-29 11:38:43 +00:00
|
|
|
let dictionary;
|
2022-08-31 04:04:42 +00:00
|
|
|
let limit = 100;
|
|
|
|
let delay = 500;
|
2023-03-29 11:38:43 +00:00
|
|
|
let depth = 1;
|
2022-10-23 11:51:55 +00:00
|
|
|
if (options?.limit) limit = options?.limit;
|
|
|
|
if (options?.delay) delay = options?.delay;
|
2023-03-29 11:38:43 +00:00
|
|
|
if (options?.depth) depth = options?.depth;
|
2022-08-31 04:04:42 +00:00
|
|
|
if (typeof limit !== 'number') throw new TypeError('INVALID_TYPE', 'limit', 'Number');
|
|
|
|
if (limit < 1 || limit > 100) throw new RangeError('INVALID_RANGE_QUERY_MEMBER');
|
|
|
|
if (typeof delay !== 'number') throw new TypeError('INVALID_TYPE', 'delay', 'Number');
|
2023-03-29 11:38:43 +00:00
|
|
|
if (typeof depth !== 'number') throw new TypeError('INVALID_TYPE', 'depth', 'Number');
|
|
|
|
if (depth < 1) throw new RangeError('INVALID_RANGE_QUERY_MEMBER');
|
|
|
|
if (depth > 2) {
|
|
|
|
console.warn(`[WARNING] GuildMemberManager#fetchBruteforce: depth greater than 2, can lead to very slow speeds`);
|
|
|
|
}
|
2022-10-23 11:51:55 +00:00
|
|
|
if (delay < 500 && !options?.skipWarn) {
|
|
|
|
console.warn(
|
|
|
|
`[WARNING] GuildMemberManager#fetchBruteforce: delay is less than 500ms, this may cause rate limits.`,
|
|
|
|
);
|
|
|
|
}
|
2023-03-29 11:38:43 +00:00
|
|
|
let skipValues = [];
|
2022-08-31 04:04:42 +00:00
|
|
|
// eslint-disable-next-line no-async-promise-executor
|
|
|
|
return new Promise(async (resolve, reject) => {
|
2023-03-29 11:38:43 +00:00
|
|
|
for (let i = 1; i <= depth; i++) {
|
|
|
|
dictionary = _(defaultQuery)
|
|
|
|
.permutations(i)
|
|
|
|
.map(v => _.join(v, ''))
|
|
|
|
.value();
|
|
|
|
for (const query of dictionary) {
|
2023-03-29 17:13:40 +00:00
|
|
|
if (this.guild.members.cache.size >= this.guild.memberCount) break;
|
2023-03-29 11:38:43 +00:00
|
|
|
this.client.emit(
|
|
|
|
'debug',
|
|
|
|
`[INFO] GuildMemberManager#fetchBruteforce: Querying ${query}, Skip: [${skipValues.join(', ')}]`,
|
|
|
|
);
|
|
|
|
if (skipValues.some(v => query.startsWith(v))) continue;
|
|
|
|
await this._fetchMany({ query, limit })
|
|
|
|
.then(members => {
|
|
|
|
if (members.size === 0) skipValues.push(query);
|
|
|
|
})
|
|
|
|
.catch(reject);
|
|
|
|
await this.guild.client.sleep(delay);
|
|
|
|
}
|
2022-08-31 04:04:42 +00:00
|
|
|
}
|
|
|
|
resolve(this.guild.members.cache);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2023-07-17 14:05:58 +00:00
|
|
|
/**
|
2023-07-19 17:02:58 +00:00
|
|
|
* Experimental method to fetch members from the guild.
|
2023-07-17 14:24:45 +00:00
|
|
|
* <info>Lists up to 10000 members of the guild.</info>
|
2023-07-19 17:02:58 +00:00
|
|
|
* @param {number} [timeout=15_000] Timeout for receipt of members in ms
|
2023-07-17 14:05:58 +00:00
|
|
|
* @returns {Promise<Collection<Snowflake, GuildMember>>}
|
|
|
|
*/
|
2023-07-19 17:02:58 +00:00
|
|
|
fetchByMemberSafety(timeout = 15_000) {
|
2023-07-17 14:05:58 +00:00
|
|
|
return new Promise(resolve => {
|
2023-07-19 17:02:58 +00:00
|
|
|
const nonce = SnowflakeUtil.generate();
|
2023-07-17 14:05:58 +00:00
|
|
|
let timeout_ = setTimeout(() => {
|
|
|
|
this.client.removeListener(Events.GUILD_MEMBER_LIST_UPDATE, handler);
|
|
|
|
resolve(this.guild.members.cache);
|
|
|
|
}, timeout).unref();
|
|
|
|
const handler = (members, guild, raw) => {
|
2023-07-19 17:02:58 +00:00
|
|
|
if (guild.id == this.guild.id && raw.nonce == nonce) {
|
2023-07-17 14:05:58 +00:00
|
|
|
if (members.size > 0) {
|
|
|
|
this.client.ws.broadcast({
|
|
|
|
op: 35,
|
|
|
|
d: {
|
|
|
|
guild_id: this.guild.id,
|
|
|
|
query: '',
|
|
|
|
continuation_token: members.first()?.id,
|
2023-07-19 17:02:58 +00:00
|
|
|
nonce,
|
2023-07-17 14:05:58 +00:00
|
|
|
},
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
clearTimeout(timeout_);
|
|
|
|
this.client.removeListener(Events.GUILD_MEMBER_LIST_UPDATE, handler);
|
|
|
|
resolve(this.guild.members.cache);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
this.client.on('guildMembersChunk', handler);
|
|
|
|
this.client.ws.broadcast({
|
|
|
|
op: 35,
|
|
|
|
d: {
|
|
|
|
guild_id: this.guild.id,
|
|
|
|
query: '',
|
|
|
|
continuation_token: null,
|
2023-07-19 17:02:58 +00:00
|
|
|
nonce,
|
2023-07-17 14:05:58 +00:00
|
|
|
},
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2022-08-31 04:04:42 +00:00
|
|
|
/**
|
2022-11-23 16:50:09 +00:00
|
|
|
* Fetches multiple members from the guild in the channel.
|
2022-08-31 04:04:42 +00:00
|
|
|
* @param {GuildTextChannelResolvable} channel The channel to get members from (Members has VIEW_CHANNEL permission)
|
|
|
|
* @param {number} [offset=0] Start index of the members to get
|
2022-11-23 15:09:24 +00:00
|
|
|
* @param {boolean} [double=false] Whether to use double range
|
|
|
|
* @param {number} [retryMax=3] Number of retries
|
2022-08-31 04:04:42 +00:00
|
|
|
* @param {number} [time=10e3] Timeout for receipt of members
|
|
|
|
* @returns {Collection<Snowflake, GuildMember>} Members in the guild
|
2022-11-23 15:11:16 +00:00
|
|
|
* @see {@link https://github.com/aiko-chan-ai/discord.js-selfbot-v13/blob/main/Document/FetchGuildMember.md}
|
|
|
|
* @example
|
|
|
|
* const guild = client.guilds.cache.get('id');
|
|
|
|
* const channel = guild.channels.cache.get('id');
|
|
|
|
* // Overlap (slow)
|
|
|
|
* for (let index = 0; index <= guild.memberCount; index += 100) {
|
|
|
|
* await guild.members.fetchMemberList(channel, index, index !== 100).catch(() => {});
|
2022-11-23 16:50:09 +00:00
|
|
|
* await client.sleep(500);
|
2022-11-23 15:11:16 +00:00
|
|
|
* }
|
|
|
|
* // Non-overlap (fast)
|
|
|
|
* for (let index = 0; index <= guild.memberCount; index += 200) {
|
|
|
|
* await guild.members.fetchMemberList(channel, index == 0 ? 100 : index, index !== 100).catch(() => {});
|
2022-11-23 16:50:09 +00:00
|
|
|
* await client.sleep(500);
|
2022-11-23 15:11:16 +00:00
|
|
|
* }
|
|
|
|
* console.log(guild.members.cache.size); // will print the number of members in the guild
|
2022-08-31 04:04:42 +00:00
|
|
|
*/
|
2022-11-23 15:09:24 +00:00
|
|
|
fetchMemberList(channel, offset = 0, double = false, retryMax = 3, time = 10_000) {
|
2022-08-31 04:04:42 +00:00
|
|
|
const channel_ = this.guild.channels.resolve(channel);
|
|
|
|
if (!channel_?.isText()) throw new TypeError('INVALID_TYPE', 'channel', 'GuildTextChannelResolvable');
|
|
|
|
if (typeof offset !== 'number') throw new TypeError('INVALID_TYPE', 'offset', 'Number');
|
|
|
|
if (typeof time !== 'number') throw new TypeError('INVALID_TYPE', 'time', 'Number');
|
2022-11-23 15:09:24 +00:00
|
|
|
if (typeof retryMax !== 'number') throw new TypeError('INVALID_TYPE', 'retryMax', 'Number');
|
|
|
|
if (retryMax < 1) throw new RangeError('INVALID_RANGE_RETRY');
|
|
|
|
if (typeof double !== 'boolean') throw new TypeError('INVALID_TYPE', 'double', 'Boolean');
|
|
|
|
// TODO: if (this.guild.large) throw new Error('GUILD_IS_LARGE');
|
2022-08-31 04:04:42 +00:00
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
const default_ = [[0, 99]];
|
|
|
|
const fetchedMembers = new Collection();
|
2022-11-23 15:09:24 +00:00
|
|
|
if (offset > 99) {
|
|
|
|
// eslint-disable-next-line no-unused-expressions
|
|
|
|
double
|
|
|
|
? default_.push([offset, offset + 99], [offset + 100, offset + 199])
|
|
|
|
: default_.push([offset, offset + 99]);
|
2022-08-31 04:04:42 +00:00
|
|
|
}
|
2022-11-23 15:09:24 +00:00
|
|
|
let retry = 0;
|
2022-08-31 04:04:42 +00:00
|
|
|
const handler = (members, guild, type, raw) => {
|
|
|
|
timeout.refresh();
|
|
|
|
if (guild.id !== this.guild.id) return;
|
|
|
|
if (type == 'INVALIDATE' && offset > 100) {
|
2022-11-23 15:09:24 +00:00
|
|
|
if (retry < retryMax) {
|
|
|
|
this.guild.shard.send({
|
2023-02-16 04:55:35 +00:00
|
|
|
op: Opcodes.GUILD_SUBSCRIPTIONS,
|
2022-11-23 15:09:24 +00:00
|
|
|
d: {
|
|
|
|
guild_id: this.guild.id,
|
|
|
|
typing: true,
|
|
|
|
threads: true,
|
|
|
|
activities: true,
|
|
|
|
channels: {
|
|
|
|
[channel_.id]: default_,
|
|
|
|
},
|
|
|
|
thread_member_lists: [],
|
|
|
|
members: [],
|
|
|
|
},
|
|
|
|
});
|
|
|
|
retry++;
|
|
|
|
} else {
|
|
|
|
clearTimeout(timeout);
|
|
|
|
this.client.removeListener(Events.GUILD_MEMBER_LIST_UPDATE, handler);
|
|
|
|
this.client.decrementMaxListeners();
|
|
|
|
reject(new Error('INVALIDATE_MEMBER', raw.ops[0].range));
|
|
|
|
}
|
2022-08-31 04:04:42 +00:00
|
|
|
} else {
|
|
|
|
for (const member of members.values()) {
|
|
|
|
fetchedMembers.set(member.id, member);
|
|
|
|
}
|
|
|
|
clearTimeout(timeout);
|
|
|
|
this.client.removeListener(Events.GUILD_MEMBER_LIST_UPDATE, handler);
|
|
|
|
this.client.decrementMaxListeners();
|
|
|
|
resolve(fetchedMembers);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
const timeout = setTimeout(() => {
|
|
|
|
this.client.removeListener(Events.GUILD_MEMBER_LIST_UPDATE, handler);
|
|
|
|
this.client.decrementMaxListeners();
|
|
|
|
reject(new Error('GUILD_MEMBERS_TIMEOUT'));
|
|
|
|
}, time).unref();
|
|
|
|
this.client.incrementMaxListeners();
|
|
|
|
this.client.on(Events.GUILD_MEMBER_LIST_UPDATE, handler);
|
2022-11-23 15:09:24 +00:00
|
|
|
this.guild.shard.send({
|
2023-02-16 04:55:35 +00:00
|
|
|
op: Opcodes.GUILD_SUBSCRIPTIONS,
|
2022-11-23 15:09:24 +00:00
|
|
|
d: {
|
|
|
|
guild_id: this.guild.id,
|
|
|
|
typing: true,
|
|
|
|
threads: true,
|
|
|
|
activities: true,
|
|
|
|
channels: {
|
|
|
|
[channel_.id]: default_,
|
|
|
|
},
|
|
|
|
thread_member_lists: [],
|
|
|
|
members: [],
|
|
|
|
},
|
|
|
|
});
|
2022-08-31 04:04:42 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2023-02-22 05:01:29 +00:00
|
|
|
/**
|
|
|
|
* Adds a role to a member.
|
|
|
|
* @param {GuildMemberResolvable} user The user to add the role from
|
|
|
|
* @param {RoleResolvable} role The role to add
|
|
|
|
* @param {string} [reason] Reason for adding the role
|
|
|
|
* @returns {Promise<GuildMember|User|Snowflake>}
|
|
|
|
*/
|
|
|
|
async addRole(user, role, reason) {
|
|
|
|
const userId = this.guild.members.resolveId(user);
|
|
|
|
const roleId = this.guild.roles.resolveId(role);
|
|
|
|
|
|
|
|
await this.client.api.guilds(this.guild.id).members(userId).roles(roleId).put({ reason });
|
|
|
|
|
|
|
|
return this.resolve(user) ?? this.client.users.resolve(user) ?? userId;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Removes a role from a member.
|
|
|
|
* @param {UserResolvable} user The user to remove the role from
|
|
|
|
* @param {RoleResolvable} role The role to remove
|
|
|
|
* @param {string} [reason] Reason for removing the role
|
|
|
|
* @returns {Promise<GuildMember|User|Snowflake>}
|
|
|
|
*/
|
|
|
|
async removeRole(user, role, reason) {
|
|
|
|
const userId = this.guild.members.resolveId(user);
|
|
|
|
const roleId = this.guild.roles.resolveId(role);
|
|
|
|
|
|
|
|
await this.client.api.guilds(this.guild.id).members(userId).roles(roleId).delete({ reason });
|
|
|
|
|
|
|
|
return this.resolve(user) ?? this.client.users.resolve(user) ?? userId;
|
|
|
|
}
|
|
|
|
|
2022-03-19 10:37:45 +00:00
|
|
|
_fetchMany({
|
|
|
|
limit = 0,
|
2022-08-09 03:43:16 +00:00
|
|
|
withPresences: presences = true,
|
2022-03-19 10:37:45 +00:00
|
|
|
user: user_ids,
|
|
|
|
query,
|
|
|
|
time = 120e3,
|
2022-03-24 10:55:32 +00:00
|
|
|
nonce = SnowflakeUtil.generate(),
|
2022-03-19 10:37:45 +00:00
|
|
|
} = {}) {
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
if (!query && !user_ids) query = '';
|
|
|
|
if (nonce.length > 32) throw new RangeError('MEMBER_FETCH_NONCE_LENGTH');
|
2022-08-31 04:04:42 +00:00
|
|
|
this.guild.shard.send({
|
|
|
|
op: Opcodes.REQUEST_GUILD_MEMBERS,
|
|
|
|
d: {
|
|
|
|
guild_id: this.guild.id,
|
|
|
|
presences,
|
|
|
|
user_ids,
|
|
|
|
query,
|
|
|
|
nonce,
|
|
|
|
limit,
|
|
|
|
},
|
|
|
|
});
|
2022-03-19 10:37:45 +00:00
|
|
|
const fetchedMembers = new Collection();
|
|
|
|
let i = 0;
|
|
|
|
const handler = (members, _, chunk) => {
|
|
|
|
timeout.refresh();
|
2022-08-31 04:04:42 +00:00
|
|
|
if (chunk.nonce !== nonce) return;
|
2022-03-19 10:37:45 +00:00
|
|
|
i++;
|
|
|
|
for (const member of members.values()) {
|
|
|
|
fetchedMembers.set(member.id, member);
|
|
|
|
}
|
2022-08-31 04:04:42 +00:00
|
|
|
if (members.size < 1_000 || (limit && fetchedMembers.size >= limit) || i === chunk.count) {
|
2022-03-19 10:37:45 +00:00
|
|
|
clearTimeout(timeout);
|
2022-03-24 10:55:32 +00:00
|
|
|
this.client.removeListener(Events.GUILD_MEMBERS_CHUNK, handler);
|
2022-03-19 10:37:45 +00:00
|
|
|
this.client.decrementMaxListeners();
|
2022-08-31 04:04:42 +00:00
|
|
|
let fetched = fetchedMembers;
|
2022-03-19 10:37:45 +00:00
|
|
|
if (user_ids && !Array.isArray(user_ids) && fetched.size) fetched = fetched.first();
|
2022-08-09 03:43:16 +00:00
|
|
|
resolve(fetched);
|
2022-03-19 10:37:45 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
const timeout = setTimeout(() => {
|
2022-03-24 10:55:32 +00:00
|
|
|
this.client.removeListener(Events.GUILD_MEMBERS_CHUNK, handler);
|
2022-03-19 10:37:45 +00:00
|
|
|
this.client.decrementMaxListeners();
|
|
|
|
reject(new Error('GUILD_MEMBERS_TIMEOUT'));
|
|
|
|
}, time).unref();
|
|
|
|
this.client.incrementMaxListeners();
|
2022-03-24 10:55:32 +00:00
|
|
|
this.client.on(Events.GUILD_MEMBERS_CHUNK, handler);
|
2022-03-19 10:37:45 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
module.exports = GuildMemberManager;
|