This repository has been archived by the owner on Dec 6, 2022. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathdiscordpurge.py
98 lines (82 loc) · 3.46 KB
/
discordpurge.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
"""discordpurge
Usage:
discordpurge.py [-q | --quiet] [--after=<utc_datetime>] [--before=<utc_datetime>] <target>
Options:
-q --quiet Disable interactive confirmation check
"""
import asyncio
import datetime
import sys
from pathlib import Path
import discord
import dateutil.parser as du_parser
from docopt import docopt
client = discord.Client()
@client.event
async def on_ready():
try:
counter = 0
print(f"Logged in as '{client.user}'")
# Get DM channels
print("Acquiring DM channels...")
dm_channels = (c for c in client.private_channels if isinstance(c, discord.DMChannel))
# Get specific DM channel by recipient name#number
print(f"Identifying DM channel with recipient '{name}#{number}'...")
try:
dm_channel = list(c for c in dm_channels
if c.recipient.name == name and c.recipient.discriminator == number)[0]
except IndexError:
raise Exception(f"No recipient '{name}#{number}' found in DMs")
# Acquire message history from after specified date
print(f"Deleting messages sent between '{after_dt} UTC' and '{before_dt} UTC'...")
async for msg in dm_channel.history(limit=None, after=after_dt, before=before_dt):
# If the message was written by me and is not a system message
if msg.author == client.user and msg.type == discord.MessageType.default:
counter += 1
print("*", end="", flush=True)
await msg.delete()
print(f"\n{counter} messages deleted.")
except asyncio.CancelledError:
print(f"\nPurge cancelled after {counter} messages deleted.")
except Exception as e:
print(f"Error: {str(e)}")
finally:
await client.close()
def _dt_parse(dts):
try:
return du_parser.parse(dts, ignoretz=True)
except du_parser._parser.ParserError:
print(f"Error: could not parse datetime '{dts}'")
sys.exit(1)
if __name__ == '__main__':
args = docopt(__doc__)
quiet = True if args["--quiet"] else False
# Load the authentication token from auth_token.txt
token_fp = Path(__file__).parent / Path("auth_token.txt")
if token_fp.exists():
print(f"Loading token from '{token_fp.resolve()}'...")
with token_fp.open("r", encoding="utf-8") as f:
data = f.read()
token = data.replace("\n", "")
else:
print(f"Error: no 'auth_token.txt' file found in '{token_fp.parent.resolve()}'")
sys.exit(1)
# Parse the target argument into name and number variables
try:
name, number = args["<target>"].split("#")
except ValueError:
print("Error: target must be supplied as 'name#number'")
sys.exit(1)
# Parse the optional before and after arguments if provided else use default
# after (discord release date) and/or before (now)
after_dt = _dt_parse(args["--after"]) if args["--after"] else datetime.datetime(2015, 5, 13)
before_dt = _dt_parse(args["--before"]) if args["--before"] else datetime.datetime.utcnow().replace(microsecond=0)
if not quiet:
i = input(f"Confirm (Y) purge against '{name}#{number}' between '{after_dt} UTC' and '{before_dt} UTC': ")
if i not in ["y", "Y"]:
print("Purge rejected.")
sys.exit(1)
try:
client.run(token, bot=False)
except discord.errors.LoginFailure:
print(f"Login failure: improper token in '{token_fp.resolve()}'")