const path = require('path'); const find = require('findit'); const {Client,ClientOptions,GatewayIntentBits,Message, Partials, ActivityType,EmbedBuilder, ActionRowBuilder, ButtonBuilder, ButtonStyle} = require('discord.js') const roleRulesM = require('./models/autoRoleRule'); module.exports.GatewayIntentBits = GatewayIntentBits; module.exports.Partials = Partials; module.exports.ActivityType = ActivityType; module.exports.EmbedBuilder = EmbedBuilder; module.exports.ActionRowBuilder = ActionRowBuilder; module.exports.ButtonBuilder = ButtonBuilder; module.exports.ButtonStyle = ButtonStyle; module.exports.Message = Message; //@ts-check class CommandOptions { /** * * @param {String} name * @param {String[]} aliases * @param {String} description * @param {Boolean} needsAdmin * @param {Boolean} hidden */ constructor (name, aliases, description, needsAdmin, hidden) { this.name = name; this.aliases = aliases; this.description = description; this.needsAdmin = needsAdmin; this.hidden = hidden; } } class command { /** * @param {_Client} client extends discord.js.Client * @param {CommandOptions} options extends Lib.Options */ constructor(client, options) { this.name = options.name; this.aliases = options.aliases; this.description = options.description; this.needsAdmin =options.needsAdmin; this.hidden = options.hidden; this.client = client; } }module.exports.Command = command; // const {func}=require('lib.d.ts'); // func('string'); class _Client extends Client { /** * * @param {ClientOptions} options */ commands = new Map(); constructor(options) { super(options) let finder = find(path.resolve('commands')); finder.on('file', file=> { let command = require(file); if(typeof command === 'function') { let c = new command(this); this.commands.set(c.name,{ needsAdmin: c.needsAdmin?c.needsAdmin:false, command: command, }); if(c.aliases) { for(var i =0; i { this.enableCommands(); }) this.RoleSetter(); this.rustCommits = new rustCommits(this); this.freegames = new FreeGames(this); this.strikes = new Strikes(this); this.on('disconnect', (event) => { // Log a message to the console console.log(`Client disconnected: ${event.reason}`); }); // Register an event listener for the 'reconnecting' event this.on('reconnecting', () => { // Log a message to the console console.log('Client is reconnecting...'); }); } /** * Register a group of commands under the folder commands * @param {String} name - Name of the group. * @param {String} folderName - Name of the folder. */ async enableCommands() { //use needs admin here!!!! this.on("messageCreate", message=> { if(message.content.startsWith('!')) { let commandName=message.content.split('!')[1].split(' ')[0]; let args = message.content.split(`!${commandName}`)[1].split(' '); args = args.filter(n=>n); if(this.commands.get(commandName)) { let needsAdmin = this.commands.get(commandName).needsAdmin; let isAdmin = this.guilds.cache.get(message.guild.id).members.cache.get(message.author.id).permissions.has('ADMINISTRATOR'); let command = this.commands.get(commandName).command if((needsAdmin && isAdmin) || !needsAdmin) return new command(this).run(message, args); } new ErrorMessage(this).send(ErrorType.NotOnTheList,message) } }) } /** * * @param {String} name - Command Name * @returns {command} */ async RoleSetter () { this.on('messageReactionAdd', (reaction, user)=> { (async ()=> { const rule = await roleRulesM.findOne({mID:reaction.message.id,roleEmoji:reaction._emoji.id}).then(rule =>{return rule}).catch(err=>{return null}) if(rule) { console.log(`RoleAssignment: Guild:${this.guilds.cache.get(rule.gID)} Role ${rule.roleName} given to ${user.username}`) this.channels.cache.get(reaction.message.channelId).members.get(user.id).roles.add(rule.roleID) } })() }) this.on('messageReactionRemove', (reaction, user)=> { (async ()=> { const rule = await roleRulesM.findOne({mID:reaction.message.id,roleEmoji:reaction._emoji.id}).then(rule =>{return rule}).catch(err=>{return null}) if(rule) { console.log(`RoleAssignment: Guild:${this.guilds.cache.get(rule.gID)}: Removing, ${user.username}'s ${rule.roleName} role. `) this.channels.cache.get(reaction.message.channelId).members.get(user.id).roles.remove(rule.roleID) } })() }) } }module.exports.Client = _Client; const ErrorType = { Permissions: "Permissions", Arguments: "Arguments", NoArguments: "No Arguments Given", NotOnTheList: "Unknown Command", OldMessages: "Can't delete this messages." } module.exports.ErrorType = ErrorType; class Channel { /** * * @param {String} channelID */ constructor(channelID) { this.channel = channelID; } } module.exports.Channel = Channel; /** * * @param {Number} length * @returns {String} */ function Random(length) { length?length:length=5; var result = ''; var characters = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789'; var charactersLength = characters.length; for ( var i = 0; i < length; i++ ) { result += characters.charAt(Math.floor(Math.random() * charactersLength)); } return result; }module.exports.Random = Random; class ErrorMessage { /** * @param {_Client} client */ constructor(client) { this.client = client; } /** * * @param {ErrorType} errorType * @param {Message} message * @param {[... String]} extraMessages */ async send(errorType,message, extraMessages) { console.log('Sending Error Message') if (!message.channel.id) return; const embed = new EmbedBuilder() .setColor(0x4d0000) .setAuthor({name:"Rem-chan", iconURL:"https://i.imgur.com/g6FSNhL.png",url:'https://rem.wordfights.com/addtodiscord'}) .setTimestamp() .setFooter({text:'Rem-chan on ', iconURL:"https://i.imgur.com/g6FSNhL.png"}) .addFields({name: 'Error:', value: 'something'}) .addFields({name: 'Try:', value: 'something'}); switch (errorType) { case ErrorType.Arguments: embed.data.fields[0].value =ErrorType.Arguments; embed.data.fields[1].value = 'Verify the arguments provided.' break; case ErrorType.NoArguments: embed.data.fields[0].value =ErrorType.NoArguments; embed.data.fields[1].value = 'Provide the required arguments for this command.' break; case ErrorType.Permissions: embed.data.fields[0].value =ErrorType.Permissions; embed.data.fields[1].value = 'Ask this servers administrator to use the command.' break; case ErrorType.NotOnTheList: embed.data.fields[0].value =ErrorType.NotOnTheList; embed.data.fields[1].value = 'Use !help for the list of available.' break; case ErrorType.OldMessages: embed.data.fields[0].value =ErrorType.OldMessages; break; default: break; } if(extraMessages) { for(var i = 0; i i.customId === randomID; const collector = message.channel.createMessageComponentCollector({ filter, time: 60000 }); collector.on('collect', async m => { message.delete().then(()=>m.message.delete()) .catch(()=>m.message.delete()); }); } }module.exports.ErrorMessage = ErrorMessage; /** * AniList Client -> Search Functions */ class AnimeInfo { /** * * @param {String} title * @param {String} status * @param {String} episodes * @param {String} trailer * @param {String} description * @param {String} coverImage */ constructor(title, status, episodes, trailer, description, coverImage) { this.title = title; this.status = status; this.episodes=episodes; this.trailer = trailer; this.description = description; this.coverImage = coverImage; } } class CharInfo { /** * * @param {String} name * @param {String} gender * @param {String} image * @param {String} description */ constructor(name, gender, image, description) { this.name = name; this.gender = gender; this.image=image; this.description = description; } } class MangaInfo { /** * * @param {String} title * @param {String} status * @param {String} description * @param {String} coverImage */ constructor(title, status, description, coverImage) { this.title = title; this.status = status; this.description = description; this.coverImage = coverImage; } } class aniListCli { /** * * @param {String} ss - Search Query * @returns {AnimeInfo} */ async searchAnime(ss) { var variables = { ss }; var query = `query ($ss: String) { Media(search:$ss,type: ANIME) { id title { romaji english native } status episodes trailer { site id } description(asHtml:false) coverImage { large } } }`; var url = ``+'https://graphql.anilist.co', options = { method: 'POST', headers: { 'Content-Type': 'application/json', 'Accept': 'application/json', }, body: JSON.stringify({ query, variables }) }; return await fetch(url, options).then(handleResponse) .then(handleData) .catch(handleError); function handleResponse(response) { return response.json().then(function (json) { return response.ok ? json : Promise.reject(json); }); } function handleError(error) { return error; } function handleData(data) { return { title:data.data.Media.title, status:data.data.Media.status, episodes:data.data.Media.episodes, trailer:data.data.Media.trailer, description:data.data.Media.description, coverImage:data.data.Media.coverImage.large } } } /** * * @param {String} ss - Search Query * @returns {CharInfo} */ async searchChar(ss) { var variables = { ss }; var query = `query ($ss: String) { Character(search: $ss, sort:SEARCH_MATCH) { id name { full native } gender image { medium } description(asHtml:false) } } `; var url = 'https://graphql.anilist.co', options = { method: 'POST', headers: { 'Content-Type': 'application/json', 'Accept': 'application/json', }, body: JSON.stringify({ query, variables }) }; return await fetch(url, options).then(handleResponse) .then(handleData) .catch(handleError); function handleResponse(response) { return response.json().then(function (json) { return response.ok ? json : Promise.reject(json); }); } function handleError(error) { return error; } function handleData(data) { return { name:data.data.Character.name, gender:data.data.Character.gender, image:data.data.Character.image.medium, description:data.data.Character.description, } } } /** * * @param {String} ss - Search Query * @returns {MangaInfo} */ async searchManga(ss) { var variables = { ss }; var query = `query ($ss: String) { Media(search: $ss, type: MANGA) { title { romaji english native } status description(asHtml: false) coverImage { large } } } `; var url = 'https://graphql.anilist.co', options = { method: 'POST', headers: { 'Content-Type': 'application/json', 'Accept': 'application/json', }, body: JSON.stringify({ query, variables }) }; return await fetch(url, options).then(handleResponse) .then(handleData) .catch(handleError); function handleResponse(response) { return response.json().then(function (json) { return response.ok ? json : Promise.reject(json); }); } function handleError(error) { return error; } function handleData(data) { return { title:data.data.Media.title, status:data.data.Media.status, description:data.data.Media.description, coverImage:data.data.Media.coverImage.large } } } }module.exports.aniList = aniListCli; const channelM = require('./models/channels'); class rustCommits { /** * * @param {_Client} client */ constructor(client) { this.client = client; this.update(); setInterval(() => { this.update(); }, 3*60*1000); this.channels; } async update () { const channels = await channelM.find({for:'rust'}).then(channels=>{return channels}); this.channels = await this.resolveChannels(channels); const Res = await fetch("https://commits.facepunch.com/?format=json").then(handleResponse) .then(handleData) .catch(handleError); function handleResponse(response) { return response.json().then(function (json) { return response.ok ? json : Promise.reject(json); }); } function handleError(error) { return error; } function handleData(data) { data.results.splice(30, 20); return data.results.filter(x=>x.repo.search(/rust/i)!=-1) } for(var j = 0; j e.id).indexOf(this.channels[j].lastid); Res.splice(Pos, Res.length-Pos); for(var i = Res.length-1; i>0; i--) { if(Res[i].id > this.channels[j].lastid) { send( { Author:Res[i].user.name, Avatar:Res[i].user.avatar, Time:Res[i].created.split("T")[1]+ " of "+ Res[i].created.split("T")[0], Content:Res[i].message, ID:Res[i].id, Repo:Res[i].repo, Branch:Res[i].branch }, this.channels[j].cID, this.client); } } } /** * * @param {RustCommit} commit * @param {Channel} channel */ function send(commit, channel, client) { client.channels.cache.get(channel).send({embeds:[ new EmbedBuilder() .setColor(0xc23811) .setTitle(commit.Time) .setURL('https://commits.facepunch.com') .setAuthor({name:commit.Author, iconURL:commit.Avatar==''?"https://i.imgur.com/g6FSNhL.png":commit.Avatar,url:`https://commits.facepunch.com/${commit.Author.split(' ').join('')}`}) .setDescription(commit.Content) .addFields( { name:`${commit.Repo}`, value:`[${commit.Repo}/${commit.Branch}](https://commits.facepunch.com/r/${commit.Repo}/${commit.Branch}/ 'Branch Link')`}, { name: 'ID', value: commit.ID.toString() }, ) .setTimestamp() .setFooter({text:'Rem-chan on ', iconURL:"https://i.imgur.com/g6FSNhL.png"}) ]}); } } /** * * @param {channelM} channels */ async resolveChannels(channels) { for await(var channel of channels ) { if(this.client.channels.cache.get(channel.cID)) { const Channel = this.client.channels.cache.get(channel.cID) const lastid = await Channel.messages.fetch({limit:1}).then(message=> { if(!message) return 0 const [content] = message.values(); if(!content||!content.embeds||!content.embeds.length>0) return 0; return parseInt(content.embeds[0].fields[1].value) }) channel.lastid = lastid; } } return channels; } } class RustCommit { /** * * @param {String} Author * @param {String} Avatar - IMAGE URL * @param {String} Time * @param {String} Content * @param {Number} ID * @param {String} Repo * @param {String} Branch */ constructor(Author, Avatar, Time, Content, ID, Repo, Branch) { this.Author = Author this.Avatar = Avatar this.Time = Time this.Content = Content this.ID = ID this.Repo = Repo this.Branch = Branch } } class FreeGames { /** * * @param {_Client} client */ constructor(client) { this.client = client; this.update() setTimeout(() => { this.update(); }, 3*60*1000); } async update() { const channels = await channelM.find({for:'freegames'}).then(channels=>{return channels}); this.channels = await this.resolveChannels(channels); const Res = await fetch('https://www.gamerpower.com/api/giveaways').then(handleResponse) .then(handleData) .catch(handleError); function handleResponse(response) { return response.json().then(function (json) { return response.ok ? json : Promise.reject(json); }); } function handleError(error) { return error; } function handleData(data) { data.splice(10, data.length-10); return data } for(var j = 0; j e.id).indexOf(this.channels[j].lastid); Res.splice(Pos, Res.length-Pos) for(var i = Res.length-1; i>0; i--) { if(Res[i].id > this.channels[j].lastid) { send( { ID:Res[i].id, // Title:Res[i].title, // Type:Res[i].type, // Thumb:Res[i].thumbnail, // Image:Res[i].image, // Description:Res[i].description, // Instructions:Res[i].instructions,// URL:Res[i].open_giveaway_url,// Platform:Res[i].platforms,// EndDate:Res[i].end_date, }, this.channels[j].cID, this.client); } } } /** * * @param {FreeGameModel} game * @param {Channel} channel */ function send(game, channel, client) { client.channels.cache.get(channel).send({embeds:[ new EmbedBuilder() .setColor(0x2d9134) .setURL(game.URL) .setTitle(game.Title) .setAuthor({name:"Rem-chan", iconURL:"https://i.imgur.com/g6FSNhL.png",url:'https://rem.wordfights.com/addtodiscord'}) .setDescription(game.Description) .setImage(game.Image) .setThumbnail(game.Thumb) .addFields( { name: 'Platforms', value: game.Platform}, { name: 'Type of offer:', value: game.Type}, { name:'Instructions:', value:game.Instructions}, { name:`End date:`, value:`${game.EndDate}`}, ) .setTimestamp() .setFooter({text:`(${game.ID}) Rem-chan on `, iconURL:"https://i.imgur.com/g6FSNhL.png"}) ]}); } } /** * * @param {channelM} channels */ async resolveChannels(channels) { for await(var channel of channels ) { if(this.client.channels.cache.get(channel.cID)) { const Channel = this.client.channels.cache.get(channel.cID) const lastid = await Channel.messages.fetch({limit:1}).then(message=> { if(!message) return 0 const [content] = message.values(); if(!content||!content.embeds||!content.embeds.length>0) return 0; return parseInt(content.embeds[0].footer.text.split('(')[1].split(')')[0]) }) channel.lastid = lastid; } } return channels; } } class FreeGameModel { /** * @param {String} ID * @param {String} Title * @param {String} Thumb * @param {String} Type * @param {String} Image * @param {String} Description * @param {String} Instructions * @param {String} URL * @param {String} Platform * @param {Date} EndDate */ constructor(ID, Title, Thumb,Type, Image, Description, Instructions, URL, Platform, EndDate) { this.ID = ID; this.Title = Title; this.Type = Type; this.Thumb = Thumb; this.Image = Image; this.Description = Description; this.Instructions = Instructions; this.URL = URL; this.Platform = Platform; this.EndDate = EndDate; } } const strikesM = require('./models/strikes'); const guildsM = require('./models/guilds'); class Strikes { constructor(client) { this.client = client; this.check(); setInterval(() => { this.check(); }, 5*60*1000); //TODO: Handle old strikes //* Strikes life time should be difined by guild //* A default should be set //? 15 days } async check() { const Strikes = await strikesM.find().then(s=>{return s}); //console.log(Strikes); const guilds = await guildsM.find({strikes:true}).then(g=>{return g}); if(!guilds) return console.log('Striker: No Guilds') var guildIds = []; for(var i = 0; i { return guildIds.indexOf(e.guildID)>=0; } ) guildIds.forEach(guild=> { filtered.forEach(strike=> { (async (guild, strike)=> { const ammount = await strikesM.find({guildID:guild, strokedID:strike.strokedID}).then(res=>{return res.length}); if(ammount>2) { this.handleStroked(strike.strokedID, guild); } })(guild, strike) }) }) } handleStroked(user, guild) { strikesM.deleteMany({ guildID: guild, strokedID: user }, (err, res) => { if (!err) return console.log('Strikes Deleted') }); // If the user is found, kick them from the server const member = this.client.guilds.cache.get(guild).members.cache.get(user); if (!member) return console.log('Strikes: Handle Strikened: Member not valid') console.log(member.user.username) member.kick('Optional reason that will be displayed in the audit logs') .then(() => console.log(`Successfully kicked ${user.user.username}`)) .catch(err => console.log(`Unable to kick ${user.user.username}: ${err}`)); } }