-
Notifications
You must be signed in to change notification settings - Fork 9
/
Copy pathkarma.js
245 lines (214 loc) · 11.6 KB
/
karma.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
if (process.version.slice(1).split('.')[0] < 17) throw new Error('Node must be v17+ - please upgrade to the latest version of Node!')
const gist = require('snekgist')
const exec = require('child_process').exec
const os = require('os')
const moment = require('moment')
require('moment-duration-format')
const c = require('ansi-colors')
const config = require('./config.json')
const Ratelimiter = require('./Ratelimiter.js')
const rl = new Ratelimiter()
const randomColor = require('randomcolor')
const Enmap = require('enmap')
require('log-timestamp')('karmabot-git-4.0.0')
const { Client, GatewayIntentBits, EmbedBuilder } = require('discord.js')
const client = new Client({
intents: [
GatewayIntentBits.Guilds,
GatewayIntentBits.GuildMessages,
GatewayIntentBits.GuildMessageReactions,
GatewayIntentBits.MessageContent
],
})
client.karmaStore = new Enmap({ name: 'karmaStore', autoFetch: true, fetchAll: true, dataDir: './karmaStore', cloneLevel: 'deep' })
;(async () => {
await client.karmaStore.defer
console.log(c.red.bold('Enmap Init: ' + client.karmaStore.size + ' keys loaded'))
})().catch(err => {
console.error(c.bgRed.underline(err))
})
client.on('messageCreate', async (message) => {
if (message.author.bot) return
const check = await rl.check(message)
if (check === true) {
if (message.cleanContent.startsWith(config.prefix)) {
if (message.channel.type === 'dm') return
const keyword = message.cleanContent.replace(config.prefix, '').trim()
if (!client.karmaStore.has(keyword)) {
client.karmaStore.set(keyword, {
numKarma: 0
})
}
message.channel.send(`${keyword} has **${client.karmaStore.getProp(keyword, 'numKarma') || 0}** Karma!`)
} else if ((message.cleanContent.endsWith('--')) || message.cleanContent.endsWith('++')) {
if (message.channel.type === 'dm') return
let type
if (message.cleanContent.endsWith('--')) {
type = 'minus'
} else if (message.cleanContent.endsWith('++')) {
type = 'plus'
} else {
return
}
const keyword = message.cleanContent.replace(/([+-]{2,})$/m, '').trim()
if (!client.karmaStore.has(keyword)) {
client.karmaStore.set(keyword, {
numKarma: 0
})
}
if (keyword === '') return
let currentKarma = client.karmaStore.getProp(keyword, 'numKarma')
if (type === 'minus') client.karmaStore.setProp(keyword, 'numKarma', --currentKarma)
else if (type === 'plus') client.karmaStore.setProp(keyword, 'numKarma', ++currentKarma)
console.log(`[KARMAOPS]: [USER ${c.green.bold(message.author.id)}] karma-op success: ${c.cyan.bold(keyword)} ${c.red.bold.underline(type)}`)
message.channel.send(`[KARMA] **${keyword}** has **${client.karmaStore.getProp(keyword, 'numKarma') || 0}** Karma. To lookup later use **${config.prefix}** and type **${config.prefix} ${keyword}**`)
}
}
if (message.content.match(new RegExp(`^<@!?${client.user.id}>( |)$`))) {
message.reply('Hi there! Please type `@KarmaBot help` for help using this bot or `@KarmaBot stats` to get bot statistics.')
return message.react('\u2705')
}
if ((message.content.startsWith(`<@!${client.user.id}>` + ' help')) || message.content.startsWith(`<@${client.user.id}>` + ' help')) {
if (message.channel.type === 'dm') return
try {
const embed = new EmbedBuilder()
.setTitle('KarmaBot Help & Information')
.setThumbnail(message.guild.iconURL())
.setURL('https://discord.gg/dUkPxzv6em')
.setColor(randomColor())
.setDescription('**KarmaBot Help and Information (basic usage, invite URL, support)**')
.addFields(
{ name:'**❯❯ Add Karma (++):**', value: 'To **add or increase** karma, type *any* keyword (can be a username, emoji, or any string of text) followed by two plus symbols **++** For example, typing **keyword++** will increase the karma of keyword by one.', inline: true },
{ name:'**❯❯ Subtract Karma (--):**', value: 'To **subtract or decrease** karma, type *any* keyword (can be a username, emoji, or any string of text) followed by two minus symbols **--** For example, typing **keyword--** will decrease the karma of keyword by one.', inline: true },
{ name:'**❯❯ Lookup Karma (>k):**', value: 'To **lookup** karma, type **>k** followed by the keyword to lookup. For example, typing **>k keyword** will return the karma of keyword. This is shared across all guilds using KarmaBot.', inline: true },
{ name:'**❯❯ Blacklist (Per Guild):**', value: 'To **blacklist** a user from being able to add or subtract Karma in a guild, create the role **NoKarma** and assign it to the users you wish to blacklist. Users can still lookup Karma, so this can act as a way for admins/mods to, for example, award points to users without the users all being able to add/remove Karma. By default this bot will take commands from any user, but messages [are internally rate-limited for spam protection](https://cdn.rawgit.com/shikhir-arora/karma-simple/3848016d/Ratelimiter.js).', inline: true },
{ name:'**❯❯ Stats:**', value: 'For **KarmaBot Stats,** type `@KarmaBot stats` - fun stuff!', inline: true },
{ name:'**❯❯ Support:**', value: '**For support, visit:** [our Discord server](https://discord.gg/dUkPxzv6em) or [GitHub](https://github.com/shikhir-arora/karma-simple/issues).', inline: true },
)
.setTimestamp()
.setFooter({ text: 'Project by vlexar#0001 | KarmaBot Help' })
await message.reply({ embeds: [embed] })
} catch (e) {
console.error(e)
}
}
if ((message.content.startsWith(`<@!${client.user.id}>` + ' stats')) || message.content.startsWith(`<@${client.user.id}>` + ' stats')) {
try {
const embed = new EmbedBuilder()
.setTitle('KarmaBot Stats')
.setThumbnail(client.user.displayAvatarURL())
.setURL('https://discord.gg/dUkPxzv6em')
.setColor(randomColor())
.setDescription('**KarmaBot Stats/Info**')
.addFields(
{ name: '**❯❯ Users:**', value: `${client.users.cache.size}`},
{ name: '**❯❯ Channels:**', value: `${client.channels.cache.size}`},
{ name: '**❯❯ Shards:**', value: `${client.ws.shards.size}` },
{ name: '**❯❯ Uptime:**', value: moment.duration(process.uptime(), 'seconds').format('dd:hh:mm:ss')},
{ name: '**❯❯ CPU:**', value: `${os.cpus().length}x ${os.cpus()[0].model}`},
{ name: '**❯❯ Gateway Ping:**', value: `${client.ws.ping.toFixed(5)} ms`},
{ name: '**❯❯ Load Average:**', value: `${os.loadavg()[1]}`},
{ name: '**❯❯ Memory Usage:**', value: `${(process.memoryUsage().rss / 1048576).toFixed(2)}MB / ${(os.totalmem() / 1073741824).toFixed(2)}GB`},
{ name: '**❯❯ System:**', value: `${os.type()} - ${os.arch()} ${os.release()}`},
{ name: '**❯❯ Node Version:**', value: process.version},
{ name: '**❯❯ Bot Version:**', value: '4.0.0'},
{ name: '**❯❯ Discord.js:**', value: `v${require('discord.js').version}`},
{ name: '**❯❯ GitHub:**', value: '[GitHub Repo](https://github.com/shikhir-arora/karma-simple).', inline: true },
)
.setTimestamp()
.setFooter({ text: 'Project by vlexar#0001 | KarmaBot Stats'})
await message.reply({ embeds: [embed] })
} catch (e) {
console.error(e)
}
}
const clean = (text) => {
if (typeof (text) === 'string') { return text.replace(/`/g, '`' + String.fromCharCode(8203)).replace(/@/g, '@' + String.fromCharCode(8203)) } else { return text }
}
const args = message.content.split(' ').slice(1)
if (message.content.startsWith(config.adminprefix + 'eval')) {
if (message.author.id !== config.ownerID) return
try {
const code = args.join(' ')
let evaled = await eval(code) // eslint-disable-line no-eval
if (typeof evaled !== 'string') { evaled = require('util').inspect(evaled, { depth: 0 }) }
if (evaled.includes(client.token || config.token)) {
evaled = evaled.replace(client.token, 'REDACTED!')
}
if (clean(evaled).length > 1800) {
await gist(clean(evaled))
.then(res => {
const embed = new EmbedBuilder()
.setTitle('Eval output exceeds 2000 characters. View on Gist.')
.setURL(`${res.html_url}`)
.setColor(randomColor())
.setDescription(`Eval output exceeds 2000 characters. View Gist [here](${res.html_url}).`)
.setTimestamp()
.setFooter({ text: 'Eval Output'})
message.channel.send({ embeds: [embed] }).catch((e) => message.channel.send(e.message))
})
} else {
message.channel.send(clean(evaled), {
code: 'fix'
})
}
} catch (err) {
console.log(err)
err = err.toString() // eslint-disable-line no-ex-assign
if (err.includes(client.token || config.token)) {
err = err.replace(client.token, 'REDACTED!') // eslint-disable-line no-ex-assign
}
message.channel.send(`\`ERROR\` \`\`\`fix\n${clean(err)}\n\`\`\``)
}
}
if (message.content.startsWith(config.adminprefix + 'exec')) {
if (message.author.id !== config.ownerID) return
exec(args.join(' '), async (e, stdout, stderr) => {
if (stdout.length > 1800 || stderr.length > 1800) {
await gist(`${stdout}\n\n${stderr}`)
.then(res => {
const embed = new EmbedBuilder()
.setTitle('Console output exceeds 2000 characters. View on Gist.')
.setURL(`${res.html_url}`)
.setColor(randomColor())
.setDescription(`Console output exceeds 2000 characters. View Gist [here](${res.html_url}).`)
.setTimestamp()
.setFooter({ text: 'Exec Output'})
message.channel.send({ embeds: [embed] }).catch((e) => message.channel.send(e.message))
})
} else {
stdout && message.channel.send(`\`INFO:\`\n\n\`\`\`fix\n${stdout}\`\`\``)
stderr && message.channel.send(`\`ERRORS:\`\n\n\`\`\`fix\n${stderr}\`\`\``)
if (!stderr && !stdout) { message.react('\u2705') }
}
})
}
})
client.on('ready', () => {
console.log(`[READY] Connected as ${c.red.bold.underline(client.user.username)}#${c.cyan.bold(client.user.discriminator)} ${c.green.bold(client.user.id)}`)
setInterval(() => client.user.setActivity('@KarmaBot help', { type: ActivityType.Watching }), 90000)
})
client.on('guildCreate', (guild) => {
console.log(`New guild joined: ${c.blue.bold.underline(guild.name)} (id: ${c.yellow.italic(guild.id)}). This guild has ${c.green.underline(guild.memberCount)} members!`)
})
client.on('guildDelete', (guild) => {
console.log(`I have been removed from: ${c.red.bold.underline(guild.name)} (id: ${c.yellow.bold(guild.id)})`)
})
client.on('disconnect', (event) => {
setTimeout(() => client.destroy().then(() => client.login(config.token)), 10000)
console.log(c.bgRed.underline(`[DISCONNECT] Notice: Disconnected from gateway with code ${event.code} - Attempting reconnect.`))
})
client.on('reconnecting', () => {
console.log(c.bgYellow.italic('[NOTICE] ReconnectAction: Reconnecting to Discord...'))
})
client.on('rateLimit', console.log)
client.on('error', console.error)
client.on('warn', console.warn)
process.on('unhandledRejection', (error) => {
console.error(c.bgRed(`Uncaught Promise Error: \n${error.stack}`))
})
process.on('uncaughtException', (err) => {
const errmsg = (err ? err.stack || err : '').toString().replace(new RegExp(`${__dirname}/`, 'g'), './')
console.error(c.red.bold(errmsg))
})
client.login(config.token)