-
Notifications
You must be signed in to change notification settings - Fork 11
/
main.py
executable file
·159 lines (124 loc) · 6.54 KB
/
main.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
from __future__ import annotations
import re
import asyncio
import traceback
from loguru import logger
import discord
from discord.ext import commands
from config import TOKEN, OWNER_ID, FICHUB_SITES
from utils.metadata import ao3_metadata, fichub_metadata
URL_VALIDATE = r"(?:(?:https?|ftp)://)(?:\S+(?::\S*)?@)?(?:(?:[1-9]\d?|1\d\d|2[01]\d|22[0-3])(?:\.(?:1?\d{1,2}|2[0-4]\d|25[0-5])){2}(?:\.(?:[1-9]\d?|1\d\d|2[0-4]\d|25[0-4]))|(?:(?:[a-z\u00a1-\uffff0-9]+-?)*[a-z\u00a1-\uffff0-9]+)(?:\.(?:[a-z\u00a1-\uffff0-9]+-?)*[a-z\u00a1-\uffff0-9]+)*(?:\.(?:[a-z\u00a1-\uffff]{2,})))(?::\d{2,5})?(?:/[^\s]*)?"
class FicFinder(commands.Bot):
def __init__(self) -> None:
# Use the minimal number of intents necessary to conserve memory.
super().__init__(
command_prefix=",",
help_command=None,
intents=discord.Intents(guilds=True, messages=True, reactions=True, message_content=True),
)
# Make sure all commands can only be invoked within guilds.
self.add_check(commands.guild_only().predicate)
async def setup_hook(self) -> None:
"""Load extensions after the bot is logged in, but before it connects to the Discord Gateway."""
await self.load_extension("cogs.settings")
await self.load_extension("cogs.help")
await self.load_extension("cogs.link")
async def on_raw_reaction_add(self, payload: discord.RawReactionActionEvent) -> None:
channel = self.get_channel(payload.channel_id)
message = await channel.fetch_message(payload.message_id)
# Check if the message was sent by the bot
if message.author == self.user:
if payload.emoji.name == "👎":
await message.delete()
else:
await message.remove_reaction(payload.emoji, payload.member)
async def on_message(self, message: discord.Message) -> None:
"""Command to search and find the fanfiction by scraping google"""
# TODO: Re-evaluate logging for messages containing "-log".
# To run client.commands & client.event simultaneously
await super().on_message(message)
try:
query = message.content.lower()
msg = list(message.content.lower())
if message.guild is None:
logger.info("Not allowed to reply to DMs.")
return # Do not reply to DMs
# Known since we must be in a guild.
assert isinstance(message.channel, discord.abc.GuildChannel)
# if in code blocks
if re.search(r"`(.*?)`", query) is not None:
logger.info("The backquote search was used. Searching Fichub")
str_found = re.findall(r"`(.*?)`", query, re.MULTILINE)
str_found = str_found[:1] # to limit the search query to 1 only
for i in str_found:
logger.info("Sleeping for 1s to avoid ratelimit")
await asyncio.sleep(1)
i = i.replace("-log", "")
await message.channel.typing()
embed_pg = fichub_metadata(i)
# if not found in FFN, search in AO3
if embed_pg is None or embed_pg.description.startswith(
"Fanfiction not found"
):
logger.info("Fanfiction not found on FFN. Trying to search AO3")
msg = i.replace("ao3", "")
msg = msg.replace("-log", "")
embed_pg = ao3_metadata(msg)
logger.info(
f"Sending embed to Channel-> {message.channel.guild}:{message.channel.name}"
)
try:
await message.reply(
embed=embed_pg, mention_author=False
)
except Exception as err:
logger.error(err)
await message.channel.send(embed=embed_pg)
elif (
url_found := re.search(URL_VALIDATE, query, re.MULTILINE)
) is not None:
logger.info("URL was passed. Verifying if URL is supported")
# to limit the url to 1 only
supported_url = (url_found.string,)
for url in supported_url:
logger.info("Sleeping for 2s to avoid ratelimit")
await asyncio.sleep(2)
if re.search(r"archiveofourown.org\b", url) is not None:
# ignore /users/ endpoint
if not re.search(r"/users/", url):
await message.channel.typing()
logger.info(
"archiveofourown.org URL was passed. Searching ao3"
)
embed_pg = ao3_metadata(url)
logger.info(
f"Sending embed to Channel-> {message.channel.guild}:{message.channel.name}"
)
try:
await message.reply(
embed=embed_pg, mention_author=False
)
except Exception as err:
logger.error(err)
await message.channel.send(embed=embed_pg)
elif not re.search(r"fanfiction.net/u/", url):
# Check if the URL is in the FICHUB_SITES list
if any(site.strip() in url.strip() for site in FICHUB_SITES):
await message.channel.typing()
logger.info("URL was passed. Searching Fichub")
embed_pg = fichub_metadata(url)
logger.info(
f"Sending embed to Channel-> {message.channel.guild}:{message.channel.name}"
)
try:
await message.reply(
embed=embed_pg, mention_author=False
)
except Exception as err:
logger.error(err)
await message.channel.send(embed=embed_pg)
except Exception:
logger.error(traceback.format_exc())
if __name__ == "__main__":
client = FicFinder()
client.run(TOKEN)