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