main.ts 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169
  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. let content = m.cleanContent.trim();
  61. if (await trigger(mCmd.ActionType.MESSAGE, m, content))
  62. return;
  63. if (m.mentions.users.size > 0 && m.mentions.users.has(client.user.id)) {
  64. if (m.content.trim().startsWith(client.user.toString())) {
  65. content = content.substring(`@${client.user.username}`.length).trim();
  66. let lowerCaseContent = content.toLowerCase();
  67. for (let c of botCommands) {
  68. if (typeof (c.pattern) == "string" && lowerCaseContent.startsWith(c.pattern)) {
  69. c.action(m, content);
  70. return;
  71. }
  72. else if (c.pattern instanceof RegExp) {
  73. let result = c.pattern.exec(content);
  74. if (result != null) {
  75. c.action(m, content, result);
  76. return;
  77. }
  78. }
  79. }
  80. if (await trigger(mCmd.ActionType.DIRECT_MENTION, m, lowerCaseContent))
  81. return;
  82. }
  83. if (await trigger(mCmd.ActionType.INDIRECT_MENTION, m))
  84. return;
  85. }
  86. await trigger(mCmd.ActionType.POST_MESSAGE);
  87. });
  88. client.on("messageReactionAdd", (r, u) => {
  89. if (Math.random() <= REACT_PROBABILITY && !u.bot) {
  90. console.log(`Reacting to message ${r.message.id} because user ${u.tag} reacted to it`);
  91. r.message.react(r.emoji);
  92. }
  93. });
  94. function loadCommand(mod: any) {
  95. for (let i in mod) {
  96. if (!mod.hasOwnProperty(i))
  97. continue;
  98. let commandClass = mod[i] as any;
  99. // Ensure this is indeed a command class
  100. if (!commandClass.prototype || commandClass.prototype.BOT_COMMAND !== BOT_COMMAND_DESCRIPTOR)
  101. continue;
  102. let cmd = new commandClass() as mCmd.ICommand;
  103. commandSets.push(cmd);
  104. if (cmd._botCommands)
  105. botCommands.push(...cmd._botCommands.map(c => ({ ...c, action: c.action.bind(cmd) })));
  106. if (cmd._botEvents)
  107. for (let [i, event] of Object.entries(cmd._botEvents)) {
  108. botEvents[+i as mCmd.ActionType].push((event as Function).bind(cmd));
  109. }
  110. if(cmd.onStart)
  111. startActions.push(cmd.onStart.bind(cmd));
  112. }
  113. }
  114. export function getDocumentation() {
  115. return botCommands.filter(m => m.documentation !== undefined).map(m => ({
  116. name: m.pattern.toString(),
  117. doc: m.documentation.description,
  118. example: m.documentation.example,
  119. auth: m.auth || false
  120. }));
  121. }
  122. async function main() {
  123. await createConnection({
  124. ...await getConnectionOptions(),
  125. entities: DB_ENTITIES
  126. });
  127. startRPCServer();
  128. let commandsPath = path.resolve(path.dirname(module.filename), "commands");
  129. let files = fs.readdirSync(commandsPath);
  130. for (const file of files) {
  131. let ext = path.extname(file);
  132. if (ext != ".js")
  133. continue;
  134. loadCommand(require(path.resolve(commandsPath, file)));
  135. }
  136. client.login(process.env.BOT_TOKEN);
  137. }
  138. main();