main.js 5.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170
  1. const Discord = require("discord.js");
  2. const TOKEN = require("./token.js");
  3. const db = require("./db.js");
  4. const RSSParser = require("rss-parser");
  5. const interval = require("interval-promise");
  6. const TurndownService = require("turndown");
  7. const turndown = new TurndownService();
  8. turndown.addRule("image", {
  9. filter: "img",
  10. replacement: () => ""
  11. });
  12. turndown.addRule("link", {
  13. filter: node => node.nodeName === "A" &&node.getAttribute("href"),
  14. replacement: (content, node) => node.getAttribute("href")
  15. });
  16. const client = new Discord.Client();
  17. const parser = new RSSParser();
  18. const RSS_UPDATE_INTERVAL_MIN = 5;
  19. async function checkFeeds() {
  20. let feeds = db.get("rssFeeds").value();
  21. let outlets = db.get("feedOutputs").value();
  22. for(let feedEntry of feeds) {
  23. let feed = await parser.parseURL(feedEntry.url);
  24. if(feed.items.length == 0)
  25. continue;
  26. feed.items.filter(i => i.isoDate > feedEntry.lastUpdate).forEach(item => {
  27. outlets.forEach(ch => {
  28. client.channels.get(ch).send(`**${item.title}**\nPosted by ${item.creator}\n${item.link}\n\n${turndown.turndown(item[feedEntry.contentElement]).replace(/( {2}\n|\n\n){2,}/gm, "\n").replace(item.link, "")}`);
  29. });
  30. });
  31. db.get("rssFeeds").find({ url: feedEntry.url}).assign({lastUpdate: feed.items[0].isoDate}).write();
  32. }
  33. }
  34. function isAuthorised(member) {
  35. if (db.get("editors.users").includes(member.id).value())
  36. return true;
  37. if (db.get("editors.roles").intersectionWith(member.roles.keyArray()).isEmpty().value())
  38. return false;
  39. return true;
  40. }
  41. const commands = {
  42. "make guide": msg => {
  43. if (!isAuthorised(msg.member)) return;
  44. let content = msg.content.substring(msg.content.indexOf("make guide") + "make guide ".length);
  45. let guideName = content.substring(0, content.indexOf("\n")).trim();
  46. let guideContent = content.substring(content.indexOf("\n")).trim();
  47. let guide = db.get("guides").find({ name: guideName });
  48. if (!guide.isUndefined().value()) {
  49. guide.assign({ content: guideContent }).write();
  50. } else {
  51. db.get("guides")
  52. .push({
  53. name: guideName,
  54. content: guideContent
  55. })
  56. .write();
  57. }
  58. msg.channel.send(
  59. `${msg.author.toString()} Added/updated "${guideName}"!`
  60. );
  61. },
  62. "delete guide": (msg, s) => {
  63. if (!isAuthorised(msg.member)) return;
  64. let guideName = s.substring("delete guide ".length).trim();
  65. let val = db.get("guides").find({ name: guideName });
  66. if (val.isUndefined().value()) {
  67. msg.channel.send(`${msg.author.toString()} No guide "${guideName}"!`);
  68. return;
  69. }
  70. db.get("guides")
  71. .remove({ name: guideName })
  72. .write();
  73. msg.channel.send(
  74. `${msg.author.toString()} Removed guide "${guideName}!"`
  75. );
  76. },
  77. help: msg => {
  78. let guides = db
  79. .get("guides")
  80. .map(g => g.name)
  81. .reduce((p, c) => `${p}\n${c}`, "")
  82. .value();
  83. msg.channel.send(
  84. `Hello! I am NoctBot! I have answers for the most common C(O)M-related questions.\nJust ping me with one of the following keywords:\n\`\`\`${guides}\`\`\``
  85. );
  86. }
  87. };
  88. client.on("ready", () => {
  89. console.log("Ready!");
  90. client.user.setActivity("@NoctBot help", { type: "PLAYING" });
  91. interval(checkFeeds, RSS_UPDATE_INTERVAL_MIN * 60 * 1000);
  92. });
  93. client.on("message", m => {
  94. if (m.author.id == client.user.id || m.mentions.users.size == 0) return;
  95. if (!db.get("reactableMentionedUsers").intersectionWith(m.mentions.users.map(u => u.id)).isEmpty().value()) {
  96. const emoteId = db
  97. .get("emotes")
  98. .get("angery")
  99. .randomElement()
  100. .value();
  101. m.react(client.emojis.find(e => e.id == emoteId));
  102. return;
  103. }
  104. if (m.mentions.users.first().id == client.user.id) {
  105. let content = m.cleanContent.trim();
  106. if(content.startsWith(`@${client.user.username}`)) {
  107. content = content
  108. .substring(`@${client.user.username} `.length);
  109. let lowerCaseContent = content.toLowerCase().trim();
  110. for (let c in commands) {
  111. if (lowerCaseContent.startsWith(c)) {
  112. commands[c](m, content);
  113. return;
  114. }
  115. }
  116. if (lowerCaseContent.length > 0) {
  117. let parts = lowerCaseContent.trim().split(" ");
  118. let guide = db
  119. .get("guides")
  120. .map(g => Object.assign({parts: g.name.toLowerCase().split(" ")}, g))
  121. .sortBy(g => g.parts.length)
  122. .maxBy(k => db._.intersection(parts, k.parts).length)
  123. .value();
  124. let hits =
  125. guide !== undefined &&
  126. db._.intersection(guide.name.toLowerCase().split(" "), parts).length > 0;
  127. if (hits) {
  128. m.channel.send(guide.content);
  129. return;
  130. }
  131. }
  132. }
  133. let emoteType = "angery";
  134. if(db.get("specialUsers").includes(m.author.id).value())
  135. emoteType = "hug";
  136. else if(db.get("bigUsers").includes(m.author.id).value())
  137. emoteType = "big";
  138. else if(db.get("dedUsers").includes(m.author.id).value())
  139. emoteType = "ded";
  140. const id = db
  141. .get("emotes")
  142. .get(emoteType)
  143. .randomElement()
  144. .value();
  145. m.channel.send(client.emojis.find(e => e.id == id).toString());
  146. } else if (m.content.includes("Noct")) {
  147. m.channel.send(
  148. client.emojis.find(e => e.name == "mukuNeighWaaaaaa").toString()
  149. );
  150. }
  151. });
  152. client.login(TOKEN);