-
Notifications
You must be signed in to change notification settings - Fork 23
/
gen.py
233 lines (191 loc) · 7.09 KB
/
gen.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
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
import argparse
import config
import ftplib
import logging
import os
import random
import sys
from arrays import FOLX, TREATS
from datetime import datetime, timezone
from enum import Enum
from mastodon import Mastodon
log = logging.getLogger(__name__)
# The chance for the treat to become a threat
THREAT_PROBABILITY = 1 / 100
class Visibility(Enum):
"""The possible visibilities for a post according to the mastodon client"""
private = "private"
direct = "direct"
unlisted = "unlisted"
public = "public"
def __str__(self: "Visibility") -> str:
return self.value
def get_log_level(no_log: bool, verbose: bool) -> int:
if no_log:
return logging.ERROR
elif verbose:
return logging.DEBUG
else:
return logging.INFO
def count_combinations() -> None:
"""Calculate the number of possible outputs"""
num_folx = len(FOLX)
num_treats = len(TREATS)
combinations = num_folx * num_treats
output = f"There are {num_folx} folx and {num_treats} treats, resulting in {combinations:,} possible combinations."
log.info(output)
print(output)
def update_bio(dry_run: bool = False) -> None:
"""Update the bot's bio with the number of possible combinations"""
num_folx = len(FOLX)
num_treats = len(TREATS)
combinations = num_folx * num_treats
last_update = datetime.now(timezone.utc).strftime("%Y-%m-%d %H:%M:%S")
bio = f"You can have another bot, as a treat.\n\nI can choose from {num_folx} folx and {num_treats} treats, so there are {combinations:,} possible combinations.\n\nI last updated this bio on {last_update} (UTC)."
if dry_run is False:
mastodon = Mastodon(
access_token=config.ACCESS_TOKEN, api_base_url=config.API_URL
)
mastodon.account_update_credentials(note=bio)
log.info("Updated bio to: %s", bio)
print(f"Updated bio to: {bio}")
else:
print(f"Dry run, would have updated bio to: {bio}")
log.info("Dry run, would have updated bio to: %s", bio)
def write_status(
status: str, dry_run: bool = False, visibility: Visibility = Visibility("unlisted")
) -> None:
"""Write a status to Mastodon"""
if dry_run is False:
# Post
mastodon = Mastodon(
access_token=config.ACCESS_TOKEN, api_base_url=config.API_URL
)
mastodon.status_post(status=status, visibility=str(visibility))
log.info('Posted "%s"', status)
print(f"Posted {status}")
else:
print(f"Dry run, would have posted {status}")
log.info('Dry run, would have posted "%s"', status)
def should_be_threat():
"""Use THREAT_PROBABILITY to determine if this treat should be a threat"""
range_max = int(1 / THREAT_PROBABILITY)
chosen_value = random.randint(1, range_max)
log.debug("Treat/Threat value %d (threat requires %d)", chosen_value, range_max)
is_threat = chosen_value == range_max
if is_threat:
log.debug("Post will be a threat")
else:
log.debug("Post will be a treat")
return is_threat
def get_used_filename(thing: str) -> str:
"""Get the filename for the file containing used _things_"""
return f"used_{thing}"
def save_used(thing: str, value: str) -> None:
"""Add an entry to the used _things_ list"""
filename = get_used_filename(thing)
with open(filename, "a") as f:
f.write(value + "\n")
def get_used(thing: str) -> list[str]:
"""Get the list of used _things_"""
filename = get_used_filename(thing)
if os.path.isfile(filename):
with open(filename, "r") as f:
return f.read().splitlines()
else:
return []
def clear_used(thing: str) -> None:
"""Clear the list of used _things_"""
filename = get_used_filename(thing)
with open(filename, "w") as f:
f.write("")
log.info("Cleared used %s list", thing)
def upload_logs(filename: str) -> None:
"""Upload a file to the FTP server"""
# Check if filename exists
if not os.path.isfile(filename):
log.error(f"File {filename} does not exist")
return
session = ftplib.FTP(config.FTP_HOST, config.FTP_USER, config.FTP_PASS)
ftplib.FTP.cwd(session, "as-a-treat")
file = open(filename, "rb")
session.storbinary(f"STOR {filename}", file)
file.close()
session.quit()
log.info(f"Uploaded {filename}")
if __name__ == "__main__":
parser = argparse.ArgumentParser(
description='Generate a string in the format "{folx} can have {treats}, as a treat" and post it to fedi'
)
parser.add_argument(
"-d",
"--dry-run",
action="store_true",
help="Generate output, but do not post it",
)
parser.add_argument(
"-c",
"--count",
action="store_true",
help="Count the number of possible outputs and exit",
)
parser.add_argument(
"-u",
"--update-bio",
action="store_true",
help="Update the bot's bio with the number of possible combinations",
)
parser.add_argument(
"--visibility",
type=Visibility,
choices=list(Visibility),
action="store",
default="unlisted",
)
parser.add_argument("--no-log", action="store_true", help="Disable logging")
parser.add_argument(
"-v", "--verbose", action="store_true", help="Enable verbose logging"
)
args = parser.parse_args()
log_level = get_log_level(args.no_log, args.verbose)
log_format = "%(asctime)s %(levelname)-5s %(message)s"
logging.basicConfig(filename="as-a-treat.log", level=log_level, format=log_format)
if args.count:
count_combinations()
sys.exit(0)
if args.update_bio:
update_bio(args.dry_run)
sys.exit(0)
used_folx = get_used("folx")
used_treats = get_used("treats")
# Remove previously used folx
available_folx = [item for item in FOLX if item not in used_folx]
log.debug("%d unused folx remaining", len(available_folx))
if len(available_folx) == 0:
available_folx = FOLX
clear_used("folx")
# Remove previously used treats
available_treats = [item for item in TREATS if item not in used_treats]
log.debug("%d unused treats remaining", len(available_treats))
if len(available_treats) == 0:
available_treats = TREATS
clear_used("treats")
# Choose a random folx and treat from the remaining available options
folx = random.choice(available_folx)
treat = random.choice(available_treats)
log.debug('Chose folx "%s" and treat "%s"', folx, treat)
# Save the chosen folx and treat so they can't be picked again
save_used("folx", folx)
save_used("treats", treat)
treat_or_threat = "threat" if should_be_threat() else "treat"
status = f"{folx} can have {treat}, as a {treat_or_threat}"
write_status(status, args.dry_run, args.visibility)
# Upload logs
if config.DONT_UPLOAD_LOGS:
print("Not uploading logs as DONT_UPLOAD_LOGS is True")
else:
log.info("Uploading logs...")
upload_logs("used_folx")
upload_logs("used_treats")
upload_logs("as-a-treat.log")
log.info("Finished uploading logs")