This repository has been archived on 2025-01-25. You can view files and clone it, but cannot push or open issues or pull requests.
lset/twitch-bot/bot.js
2022-03-13 13:56:32 -06:00

256 lines
8.8 KiB
JavaScript

/*
* Created by Bryson Steck (@brysonsteck on GitHub)
* Feel free to make changes and send a PR!
* Open source under the MIT License:
*
* Copyright (c) 2022 Bryson Steck
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in all
* copies or substantial portions of the Software.
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
* SOFTWARE.
*
*/
const tmi = require('tmi.js');
const fs = require('fs');
// read settings and reactions json files
try {
var settings = JSON.parse(fs.readFileSync('./settings.json', 'utf8'));
var reacts = JSON.parse(fs.readFileSync('./reacts.json', 'utf8'));
var chat_commands = JSON.parse(fs.readFileSync('./commands.json', 'utf8'));
var mod_commands = JSON.parse(fs.readFileSync('./mod_commands.json', 'utf8'));
} catch (err) {
console.error("An error occured trying to read the files for the Twitch bot: " + err);
}
const opts = {
identity: {
username: settings.bot_username,
password: settings.bot_token
},
channels: settings.channels
};
const linkProtection = settings.link_protection;
const bannedUrlEndings = settings.banned_endings;
var modString = "";
var mods;
try {
modString = fs.readFileSync('mods.txt', 'utf8');
trustedString = fs.readFileSync('trusted_users.txt', 'utf8');
modString = modString.replace(/(\r\n|\n|\r)/gm, "");
trustedString = trustedString.replace(/(\r\n|\n|\r)/gm, "");
console.log('* Loaded mods and trusted users from file.');
mods = modString.split(",");
trustedUsers = trustedString.split(",");
} catch (err) {
console.error(err)
}
var modCommand = false;
var message, fullMessage;
function onMessageHandler (target, context, msg, self) {
if (self) { return; } // Ignore messages from the bot
const user = context.username;
// save full message if adding/editing command
fullMessage = msg.trim();
msg = msg.toLowerCase();
message = msg.trim();
// link protection stuff, only enables when true in settings.json
var findUrlEndings = false;
if (linkProtection) {
for (var i=urlEndings.length; i--;) {
if (message.includes(urlEndings[i])) findUrlEndings = true;
else if (message.includes(urlEndings[i].replace('.', '*'))) findUrlEndings = true;
}
}
if (message.charAt(0) === settings.command_char) {
message = message.substr(1);
valid = commands(target, message, user, mods);
} else {
if (linkProtection && findUrlEndings)
if (!linkProtect())
reactions(target, message, user);
else
reactions(target, message, user);
}
}
function commands (target, commandName, user, mods) {
modCommand = false;
var valid = false;
var isMod = false;
if (mods.indexOf(user) >= 0) isMod = true;
if (commandName === "help") {
var finalString = settings.command_char + "help " + settings.command_char + "about ";
chat_commands.forEach(command => {
finalString = finalString + settings.command_char + command.command + " ";
});
client.say(target, `Here is the list of commands: ${finalString}`);
valid = true;
} else if (commandName === "about") {
client.say(target, `This bot was stolen from https://github.com/brysonsteck/lset under the MIT License and configured by your streamer, ${settings.your_username}!`);
valid = true;
} else {
chat_commands.forEach(command => {
if (commandName.search(command.command) !== -1) {
client.say(target, `${command.reply}`);
valid = true;
return true;
}
});
}
if (valid === false) {
if (isMod) {
valid = modCommands(target, commandName, isMod);
}
}
if (valid) {
console.log(`* ${user} executed !${commandName}`);
} else {
if (modCommand === false) {
client.say(target, `@${user} Unknown command "${settings.command_char}${commandName}". Type ${settings.command_char}help for all commands.`);
}
console.warn(`! ${user} tried to execute unknown/banned command !${commandName}`);
}
}
function modCommands(target, commandName, message, isMod) {
var valid = false;
test_mod_commands = ["addcommand", "editcommand"];
test_mod_commands.forEach(command => {
if (commandName.search(command) !== -1) {
if (!isMod) {
client.say(target, `Only moderators can run this command...`);
return true;
} else {
if (command === "addcommand") {
createCommand(target);
} else if (command === "editcommand") {
editCommand(target);
}
client.say(target, `${command.reply}`);
valid = true;
return true;
}
}
});
return valid;
}
function createCommand(target) {
chat_commands.forEach(command => {
if (fullMessage[1].search(command.command) !== -1) {
client.say(target, `The command ${settings.command_char}${fullMessage[1]} already exists. Use !editcommand to change it's contents.`);
return false;
}
});
var command_reply = "";
for (int i = 2; i < fullMessage.length; i++) {
command_reply = command_reply + fullMessage[i] + " ";
}
chat_commands.push(['command': fullMessage[1], 'reply': command_reply]);
try {
const data = JSON.stringify(chat_commands, null, 4);
fs.writeFileSync('commands.json', data);
console.log('* Added command ' + settings.command_char + fullMessage[1]);
client.say(target, `Successfully added command: ${settings.command_char}${fullMessage[1]}`);
} catch (err) {
console.error('An error occured trying to run !addcommand: ' + err);
client.say(target, `Something went wrong adding this command, please try again later.`);
}
}
function editCommand(target) {
chat_commands.forEach(command => {
if (fullMessage[1].search(command.command) !== -1) {
var command_reply = "";
for (int i = 2; i < fullMessage.length; i++) {
command_reply = command_reply + fullMessage[i] + " ";
}
chat_commands[fullMessage[1]] = command_reply;
try {
const data = JSON.stringify(chat_commands, null, 4);
fs.writeFileSync('commands.json', data);
console.log('* Edited command ' + settings.command_char + fullMessage[1]);
client.say(target, `Successfully edited existing command: ${settings.command_char}${fullMessage[1]}`);
} catch (err) {
console.error('An error occured trying to run !editcommand: ' + err);
client.say(target, `Something went wrong editing this command, please try again later.`);
}
return true;
}
});
return false;
}
function reactions (target, message, user) {
reacts.forEach(react => {
if (message.search(react.trigger) !== -1) {
client.say(target, `${react.reply}`);
}
});
}
function linkProtect() {
if (trustedUsers.indexOf(user) === -1) {
if (message.search("http") !== -1 || message.search("www.") !== -1 || findUrlEndings) {
urlAttempt = message;
urlAttemptUser = user;
console.log(`! ${user} tried to post a URL. Warned and timedout for 10 seconds. Message: "${urlAttempt}"`);
client.say(target, `/timeout ${user} 10 Links from untrusted users are deleted as a protection against chat bots.`);
client.say(target, `@${user} Your link was deleted because you haven't been in my chat before. If you have a legitimate link and aren't a bot, let me know!`);
// is a bot, don't go farther
return true;
} else {
trustedString = trustedString + `,${user}`;
trustedString = trustedString.replace(/(\r\n|\n|\r)/gm, "");
trustedUsers.push(user);
fs.writeFile('trusted_users.txt', `${trustedString}`, function (err) {
if (err) return console.log(err);
console.log(`* ${user} is now a trusted chatter.`);
});
// not a bot, move on to reactions
return false;
}
}
}
// Twitch bot initialization
const client = new tmi.client(opts);
// Register our event handlers (defined below)
client.on('message', onMessageHandler);
client.on('connected', onConnectedHandler);
// Connect to Twitch:
client.connect();
// Notify when connected
function onConnectedHandler (addr, port) {
console.log(`* Main Bot successfully connected to ${addr}:${port}`);
}