diff --git a/bot/commands.js b/bot/commands.js new file mode 100644 index 0000000..5f448aa --- /dev/null +++ b/bot/commands.js @@ -0,0 +1,323 @@ +const Discord = require('discord.js'); +const fetch = require('node-fetch'); +const fs = require('fs'); +const child_process = require('child_process'); + +const helpMessage = "`do be helping`: display this help message\n`do be roundgen`: send a pdf round to the channel\n`do be roundgen dm`: dm a pdf round to you\n`do be scoring`: start a scoring session\n > `do be scoring (a/b)(4/10)`: add points to Team A or Team B\n > `do be scoring stop`: end scoring session and post final points\n`do be hits`: send the number of rounds generated\n`do be servers`: send the number of servers this bot is a part of\n`do be iss`: show the current location of the International Space Station\n`do be training`: send a quick practice problem (you **must** react to your answer, or the bot will yell at you)\n > subject options: astro, phys, chem, math, bio, ess, energy\n`do be top`: list cross-server top 10 players\nSource Code: https://github.com/ADawesomeguy/AwesomeSciBo (don't forget to star!)"; + +async function checkHits(message, hits) { + message.channel.send(hits); + fs.writeFile('numhits.txt', hits.toString(), (error) => { if (error) { console.log(error); } }); +} + +async function sendHelpMessage(message) { + message.channel.send(new Discord.MessageEmbed().setTitle("Help").setDescription(helpMessage)); +} + +async function generateRound(message, isDM) { + fs.writeFile('index.html', "

Here's your round!

", (error) => { if (error) { console.log(error); } }); + let i; + let generatingMsg = await message.channel.send("Generating..."); + for (i = 1; i < 26; i++) { + let tossup_question; + let question_category; + let tossup_format; + let tossup_answer; + let bonus_question; + let bonus_format; + let bonus_answer; + let htmlContent = ""; + await fetch('https://scibowldb.com/api/questions/random') + .then(response => response.json()) + .then(data => { + tossup_question = data.question.tossup_question; + tossup_answer = data.question.tossup_answer; + question_category = data.question.category; + tossup_format = data.question.tossup_format; + bonus_question = data.question.bonus_question; + bonus_answer = data.question.bonus_answer; + bonus_format = data.question.bonus_format; + htmlContent = `

${i}. Tossup\n

` + `${question_category}` + " " + `${tossup_format}` + " " + tossup_question + "

" + "ANSWER: " + tossup_answer + "

"; + htmlContent += "

Bonus\n

" + `${question_category}` + " " + `${bonus_format}` + " " + bonus_question + "

" + "ANSWER: " + bonus_answer + "

"; + htmlContent = htmlContent.replace(/\n/g, "
"); + fs.appendFile('index.html', htmlContent, (error) => { if (error) { console.log(error); } }); + }); + } + if (generatingMsg) { + generatingMsg.delete({ timeout: 100 }).catch(console.error); + } + child_process.execSync("curl --request POST --url https://localhost:3136/convert/html --header 'Content-Type: multipart/form-data' --form files=@index.html -o round.pdf", { encoding: 'utf-8' }); + if (isDM) { + client.users.cache.get(message.author.id).send(new Discord.MessageEmbed().setTitle("Here's your round!").attachFiles("round.pdf")).catch(() => message.reply("Unable to DM you! Make sure DMs from server members are allowed.")); + } else { + message.channel.send(new Discord.MessageEmbed().setTitle("Here's your round!").attachFiles("round.pdf")); + } + hits++; +} + +async function startScoring(message) { + let scoreA = 0; + let scoreB = 0; + const scoreboard = await message.channel.send(`Here's the score:\nTeam A: ${scoreA}\nTeam B: ${scoreB}`) + .then((scoreboard) => { + const filter = m => m.content.includes('do be'); + const collector = message.channel.createMessageCollector(filter, { time: 1500000 }); + collector.on('collect', m => { + if (m.content.toLowerCase() === "do be scoring a 4") { + m.delete({ timeout: 1000 }).catch(console.error); + scoreA += 4; + scoreboard.channel.send(`Here's the score:\nTeam A: ${scoreA}\nTeam B: ${scoreB}`); + } else if (m.content.toLowerCase() === "do be scoring a 10") { + m.delete({ timeout: 1000 }).catch(console.error); + scoreA += 10; + scoreboard.channel.send(`Here's the score:\nTeam A: ${scoreA}\nTeam B: ${scoreB}`); + } else if (m.content.toLowerCase() === "do be scoring b 4") { + m.delete({ timeout: 1000 }).catch(console.error); + scoreB += 4; + scoreboard.channel.send(`Here's the score:\nTeam A: ${scoreA}\nTeam B: ${scoreB}`); + } else if (m.content.toLowerCase() === "do be scoring b 10") { + m.delete({ timeout: 1000 }).catch(console.error); + scoreB += 10; + scoreboard.channel.send(`Here's the score:\nTeam A: ${scoreA}\nTeam B: ${scoreB}`); + } else if (m.content === "do be scoring stop") { + m.delete({ timeout: 1000 }).catch(console.error); + scoreboard.delete({ timeout: 1000 }); + m.channel.send(`**FINAL SCORE:**\nTeam A: ${scoreA}\nTeam B: ${scoreB}`); + collector.stop(); + } + }); + }) +} + +async function dontWorryBeHappy(message) { + message.channel.send(new Discord.MessageEmbed().setTitle(`Don't Worry Be Happy!`).setImage("https://media.giphy.com/media/7OKC8ZpTT0PVm/giphy.gif").setURL("https://youtu.be/d-diB65scQU")); +} + +async function showServerNumber(message) { + message.channel.send(client.guilds.cache.size); +} + +async function showIssLocation(message) { + await fetch("http://api.open-notify.org/iss-now.json") + .then(request => request.json()) + .then(data => { + message.channel.send(new Discord.MessageEmbed().setTitle("The current location of the ISS!").setImage(`https://api.mapbox.com/styles/v1/mapbox/light-v10/static/pin-s+000(${data.iss_position.longitude},${data.iss_position.latitude})/-87.0186,20,1/1000x1000?access_token=pk.eyJ1IjoiYWRhd2Vzb21lZ3V5IiwiYSI6ImNrbGpuaWdrYzJ0bGYydXBja2xsNmd2YTcifQ.Ude0UFOf9lFcQ-3BANWY5A`).setURL('https://spotthestation.nasa.gov/tracking_map.cfm')); + }); +} + +async function showLeaderboard(message) { + let messageContent = ''; + let scores = []; + + const directoryPath = path.join('userScore'); + fs.readdir(directoryPath, function (err, files) { + if (err) { + return console.log('Unable to scan directory: ' + err); + } + files.forEach(function (file) { + scores.push(`${fs.readFileSync('userScore/' + file, 'utf8')}|<@${file}>`) + }); + const scoresFormatted = scores.sort(function (a, b) { return b.split('|')[0] - a.split('|')[0] }); + if (scores.length < 10) { + message.channel.send("Not enough scores yet!"); + return; + } + for (let i = 0; i < 10; i++) { + const currentScore = scoresFormatted[i].split('|'); + messageContent += `${currentScore[1]}: ${currentScore[0]}\n\n`; + } + message.channel.send(new Discord.MessageEmbed().setTitle('Top Ten!').setDescription(messageContent)); + }); +} + +async function otherCommands(message) { + if (message.content.toLowerCase().startsWith("do be announcing") && message.author.id === process.argv[3]) { + const announcement = message.content.substring(17); + client.guilds.cache.forEach((guild) => { + const channel = guild.channels.cache.find(channel => channel.name === 'general'); + if (channel) { + if (channel.type === "text") { + channel.send(announcement).catch(console.error); + } + } + }); + } else if (message.content.toLowerCase().startsWith("do be training")) { + if (message.content === "do be training") { + const author = message.author; + fetch("https://scibowldb.com/api/questions/random") + .then(response => response.json()) + .then(data => { + let filter = m => m.author.id === message.author.id; + message.reply(data.question.tossup_question).then(() => { + message.channel.awaitMessages(filter, { + max: 1, + time: 30000, + errors: ['time'] + }) + .then(message => { + message = message.first(); + let score = 0; + if (fs.existsSync(`userScore/${message.author.id}`)) { + fs.readFile(`userScore/${message.author.id}`, 'utf8', function (err, data) { + score = Number(data); + if (err) { + console.log(err); + } + }); + } else { + fs.writeFile(`userScore/${message.author.id}`, score.toString(), (error) => { if (error) { console.log(error); } }); + } + let predicted = "unsure"; + if (data.question.tossup_format === "Multiple Choice") { + if (message.content.charAt(0).toLowerCase() === data.question.tossup_answer.charAt(0).toLowerCase()) { + predicted = "correct"; + } else { + predicted = "incorrect"; + } + } else { + if (message.content.toLowerCase() === data.question.tossup_answer.toLowerCase()) { + predicted = "correct"; + } else { + predicted = "incorrect"; + } + } + message.channel.send(`Correct answer: **${data.question.tossup_answer}**. Predicted: **${predicted}**. Please react to your answer!`); + message.react('✅'); + message.react('❌'); + const filter = (reaction, user) => { + return ['❌', '✅'].includes(reaction.emoji.name) && user.id === message.author.id; + }; + message.awaitReactions(filter, { max: 1, time: 600000, errors: ['time'] }) + .then(userReaction => { + const reaction = userReaction.first(); + if (reaction.emoji.name === "❌") { + fs.writeFile(`userScore/${message.author.id}`, score.toString(), (error) => { if (error) { console.log(error); } }); + message.reply(`nice try! Your score is now ${score}`); + } else { + score += 4; + fs.writeFile(`userScore/${message.author.id}`, score.toString(), (error) => { if (error) { console.log(error); } }); + message.reply(`nice job! Your score is now ${score}`); + fs.writeFile(`userScore/${message.author.id}`, score.toString(), (error) => { if (error) { console.log(error); } }); + } + }) + .catch(collected => { + }) + }) + .catch((collected) => { + message.reply('\n**ANSWER TIMEOUT**'); + }) + }) + }) + } else { + const subject = message.content.substring(15); + let subjectURL; + switch (subject) { + case 'astro': + case 'astronomy': + subjectURL = `https://moose.lcsrc.org/subjects/astronomy.json`; + break; + case 'bio': + case 'biology': + subjectURL = `https://moose.lcsrc.org/subjects/biology.json`; + break; + case 'ess': + case 'earth science': + case 'es': + subjectURL = `https://moose.lcsrc.org/subjects/ess.json`; + break; + case 'chem': + case 'chemistry': + subjectURL = `https://moose.lcsrc.org/subjects/chemistry.json`; + break; + case 'phys': + case 'physics': + subjectURL = `https://moose.lcsrc.org/subjects/physics.json`; + break; + case 'math': + subjectURL = `https://moose.lcsrc.org/subjects/math.json`; + break; + case 'energy': + subjectURL = `https://moose.lcsrc.org/subjects/energy.json`; + break; + default: + message.channel.send("Not a valid subject!"); + return; + break; + } + const author = message.author; + fetch(subjectURL) + .then(response => response.json()) + .then(data => { + const questionNum = Math.floor(Math.random() * data.length); + let filter = m => m.author.id === message.author.id; + message.reply(data[questionNum].tossup_question).then(() => { + message.channel.awaitMessages(filter, { + max: 1, + time: 30000, + errors: ['time'] + }) + .then(message => { + message = message.first(); + let score = 0; + if (fs.existsSync(`userScore/${message.author.id}`)) { + fs.readFile(`userScore/${message.author.id}`, 'utf8', function (err, data) { + score = Number(data); + if (err) { + console.log(err); + } + }); + } else { + fs.writeFile(`userScore/${message.author.id}`, score.toString(), (error) => { if (error) { console.log(error); } }); + } + let predicted = "unsure"; + if (data[questionNum].tossup_format === "Multiple Choice") { + if (message.content.charAt(0).toLowerCase() === data[questionNum].tossup_answer.charAt(0).toLowerCase()) { + predicted = "correct"; + } else { + predicted = "incorrect"; + } + } else { + if (message.content.toLowerCase() === data[questionNum].tossup_answer.toLowerCase()) { + predicted = "correct"; + } else { + predicted = "incorrect"; + } + } + message.channel.send(`Correct answer: **${data[questionNum].tossup_answer}**. Predicted: **${predicted}**. Please react to your answer!`); + message.react('✅'); + message.react('❌'); + const filter = (reaction, user) => { + return ['❌', '✅'].includes(reaction.emoji.name) && user.id === message.author.id; + }; + message.awaitReactions(filter, { max: 1, time: 600000, errors: ['time'] }) + .then(userReaction => { + const reaction = userReaction.first(); + if (reaction.emoji.name === "❌") { + fs.writeFile(`userScore/${message.author.id}`, score.toString(), (error) => { if (error) { console.log(error); } }); + message.reply(`nice try! Your score is now ${score}`); + } else { + score += 4; + fs.writeFile(`userScore/${message.author.id}`, score.toString(), (error) => { if (error) { console.log(error); } }); + message.reply(`nice job! Your score is now ${score}`); + fs.writeFile(`userScore/${message.author.id}`, score.toString(), (error) => { if (error) { console.log(error); } }); + } + }) + .catch(collected => { + }) + }) + .catch((collected, error) => { + message.reply('\n**ANSWER TIMEOUT**'); + }) + }) + }) + .catch(console.error); + } + } else { + if (formattedMessage.startsWith("dobescoring" || "dobetraining")) { + return; + } + message.channel.send("That didn't quite make sense! Please use `do be helping` to see the available commands.") + } +} + +module.exports = { checkHits, sendHelpMessage, generateRound, startScoring, dontWorryBeHappy, showServerNumber, showIssLocation, showLeaderboard, otherCommands } diff --git a/bot/package-lock.json b/bot/package-lock.json index 7f27465..8207f1a 100644 --- a/bot/package-lock.json +++ b/bot/package-lock.json @@ -391,6 +391,7 @@ "node": ">=0.10" } }, + "node_modules/discord.js": { "version": "12.5.1", "resolved": "https://registry.npmjs.org/discord.js/-/discord.js-12.5.1.tgz", @@ -2577,7 +2578,7 @@ "integrity": "sha512-NM8/P9n3XjXhIZn1lLhkFaACTOURQXjWhV4BA/RnOv8xvgqtqpAX9IO4mRQxSx1Rlo4tqzeqb0sOlruaOy3dug==" } } - }, + }, "text-table": { "version": "0.2.0", "resolved": "https://registry.npmjs.org/text-table/-/text-table-0.2.0.tgz", diff --git a/bot/userScore/745063586422063214 b/bot/userScore/745063586422063214 new file mode 100644 index 0000000..301160a --- /dev/null +++ b/bot/userScore/745063586422063214 @@ -0,0 +1 @@ +8 \ No newline at end of file