main.ts 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175
  1. // We need some kind of module resolver for @shared/db. We use module-alias.
  2. require("module-alias/register");
  3. import dotenv from "dotenv";
  4. if (process.env.NODE_ENV == "dev") {
  5. dotenv.config({
  6. path: "../.env"
  7. });
  8. dotenv.config({
  9. path: "../db.env"
  10. });
  11. process.env.TYPEORM_HOST = "localhost";
  12. process.env.TYPEORM_USERNAME = process.env.DB_USERNAME;
  13. process.env.TYPEORM_PASSWORD = process.env.DB_PASSWORD;
  14. process.env.TYPEORM_DATABASE = process.env.DB_NAME;
  15. }
  16. import * as fs from "fs";
  17. import * as path from "path";
  18. import { client } from "./client";
  19. import * as mCmd from "./model/command";
  20. import "reflect-metadata";
  21. import { createConnection, getConnectionOptions } from "typeorm";
  22. import { getNumberEnums } from "./util";
  23. import { DB_ENTITIES } from "@shared/db/entities";
  24. import { BOT_COMMAND_DESCRIPTOR } from "./model/command";
  25. import { startServer as startRPCServer } from "./rpc_service";
  26. const REACT_PROBABILITY = 0.3;
  27. async function trigger(type: mCmd.ActionType, ...params: any[]) {
  28. let actionDone = false;
  29. let actions = botEvents[type];
  30. for (let i = 0; i < actions.length; i++) {
  31. const action = actions[i] as (...args: any[]) => boolean | Promise<boolean>;
  32. let actionResult = action(actionDone, ...params);
  33. if (actionResult instanceof Promise)
  34. actionDone = (await actionResult) || actionDone;
  35. else
  36. actionDone = actionResult || actionDone;
  37. }
  38. return actionDone;
  39. }
  40. let commandSets: mCmd.ICommand[] = [];
  41. let botCommands: mCmd.IBotCommand[] = [];
  42. let botEvents: { [event in mCmd.ActionType]?: mCmd.BotAction[] } = getNumberEnums(mCmd.ActionType).reduce((p, c) => { p[c] = []; return p; }, {} as any);
  43. let startActions: Array<() => void | Promise<void>> = [];
  44. client.on("ready", async () => {
  45. console.log("Starting up NoctBot!");
  46. client.user.setActivity(process.env.NODE_ENV == "dev" ? "Maintenance" : "@NoctBot help", {
  47. type: "PLAYING"
  48. });
  49. for (let i = 0; i < startActions.length; i++) {
  50. const action = startActions[i];
  51. let val = action();
  52. if (val instanceof Promise)
  53. await val;
  54. }
  55. console.log("NoctBot is ready!");
  56. });
  57. client.on("message", async m => {
  58. if (m.author.id == client.user.id)
  59. return;
  60. if(process.env.FOOLS == "TRUE" && (m.channel.id == "297109482905796608" || m.channel.id == "429295461099110402") && Math.random() <= 0.01) {
  61. const neighs = ["*NEIGH*", "neeeeeigh!", "Gimme carrots!", "NEEEEIIIIGH", "**N E I G H**"];
  62. await m.channel.send(neighs[Math.floor(Math.random() * neighs.length)]);
  63. return;
  64. }
  65. let content = m.cleanContent.trim();
  66. if (await trigger(mCmd.ActionType.MESSAGE, m, content))
  67. return;
  68. if (m.mentions.users.size > 0 && m.mentions.users.has(client.user.id)) {
  69. if (m.content.trim().startsWith(`<@${client.user.id}>`) || m.content.trim().startsWith(`<@!${client.user.id}>`)) {
  70. content = content.substring(`@${client.user.username}`.length).trim();
  71. let lowerCaseContent = content.toLowerCase();
  72. for (let c of botCommands) {
  73. if (typeof (c.pattern) == "string" && lowerCaseContent.startsWith(c.pattern)) {
  74. c.action(m, content);
  75. return;
  76. }
  77. else if (c.pattern instanceof RegExp) {
  78. let result = c.pattern.exec(content);
  79. if (result != null) {
  80. c.action(m, content, result);
  81. return;
  82. }
  83. }
  84. }
  85. if (await trigger(mCmd.ActionType.DIRECT_MENTION, m, lowerCaseContent))
  86. return;
  87. }
  88. if (await trigger(mCmd.ActionType.INDIRECT_MENTION, m))
  89. return;
  90. }
  91. await trigger(mCmd.ActionType.POST_MESSAGE);
  92. });
  93. client.on("messageReactionAdd", (r, u) => {
  94. if (Math.random() <= REACT_PROBABILITY && !u.bot) {
  95. console.log(`Reacting to message ${r.message.id} because user ${u.tag} reacted to it`);
  96. r.message.react(r.emoji);
  97. }
  98. });
  99. function loadCommand(mod: any) {
  100. for (let i in mod) {
  101. if (!mod.hasOwnProperty(i))
  102. continue;
  103. let commandClass = mod[i] as any;
  104. // Ensure this is indeed a command class
  105. if (!commandClass.prototype || commandClass.prototype.BOT_COMMAND !== BOT_COMMAND_DESCRIPTOR)
  106. continue;
  107. let cmd = new commandClass() as mCmd.ICommand;
  108. commandSets.push(cmd);
  109. if (cmd._botCommands)
  110. botCommands.push(...cmd._botCommands.map(c => ({ ...c, action: c.action.bind(cmd) })));
  111. if (cmd._botEvents)
  112. for (let [i, event] of Object.entries(cmd._botEvents)) {
  113. botEvents[+i as mCmd.ActionType].push((event as Function).bind(cmd));
  114. }
  115. if(cmd.onStart)
  116. startActions.push(cmd.onStart.bind(cmd));
  117. }
  118. }
  119. export function getDocumentation() {
  120. return botCommands.filter(m => m.documentation !== undefined).map(m => ({
  121. name: m.pattern.toString(),
  122. doc: m.documentation.description,
  123. example: m.documentation.example,
  124. auth: m.auth || false
  125. }));
  126. }
  127. async function main() {
  128. await createConnection({
  129. ...await getConnectionOptions(),
  130. entities: DB_ENTITIES
  131. });
  132. startRPCServer();
  133. let commandsPath = path.resolve(path.dirname(module.filename), "commands");
  134. let files = fs.readdirSync(commandsPath);
  135. for (const file of files) {
  136. let ext = path.extname(file);
  137. if (ext != ".js")
  138. continue;
  139. loadCommand(require(path.resolve(commandsPath, file)));
  140. }
  141. client.login(process.env.BOT_TOKEN);
  142. }
  143. main();