forked from flathunters/flathunter
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathconfig_wizard.py
315 lines (281 loc) · 11.8 KB
/
config_wizard.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
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
"""
Create or edit a basic flathunter configuration
"""
import sys
import re
import os
from typing import List, Any, Dict, Optional
from enum import Enum
from functools import reduce
from ruamel.yaml import YAML
from prompt_toolkit.shortcuts import message_dialog, radiolist_dialog, clear, button_dialog
from prompt_toolkit import prompt
from prompt_toolkit.document import Document
from prompt_toolkit.validation import Validator, ValidationError
from flathunter.config import YamlConfig
from flathunter.crawler import immobilienscout
class ConfigurationAborted(Exception):
"""Exception to indicate the user has aborted the configuration"""
def __str__(self):
return "Configuration Aborted"
class ConfigurationError(Exception):
"""Exception to indicate that the configuration failed (programming error)"""
class Notifier(Enum):
"""Class for the different types of notifier"""
TELEGRAM = "telegram"
MATTERMOST = "mattermost"
APPRISE = "apprise"
SLACK = "slack"
def welcome():
"""Display the welcome dialog"""
message_dialog(
title="Flathunter Configuration Wizard",
text="Welcome to the Flathunter Configuration Wizard\n\n"
"This Wizard will take you through the configuration of your Flathunter\ninstallation. "
"The configuration generated will be saved to `config.yaml`\n"
"in your project directory\n\n"
"Press ENTER to continue.",
).run()
class UrlsValidator(Validator):
"""Validate that URLs entered in the URL entry screen are crawled by one of the
configured crawlers"""
def __init__(self, urls, config):
self.urls = urls
self.config = config
def validate(self, document: Document):
"""Runs when Enter is pressed at the URL Entry prompt"""
if len(document.text) == 0:
if len(self.urls) == 0:
raise ValidationError(cursor_position=0, message="Supply at least one URL")
return
for searcher in self.config.searchers():
if re.search(searcher.URL_PATTERN, document.text):
return
raise ValidationError(cursor_position=len(document.text),
message="URL did not match any configured scraper")
def gather_urls(config: YamlConfig) -> List[str]:
"""Get a list of URLs from the user for crawling"""
urls = config.target_urls()
result = ""
first_run = True
while first_run or len(urls) == 0 or len(result) > 0:
clear()
print("Enter URLs for Scraping\n")
print("Flathunter scrapes property portals by fetching content from search URLs\n"
"on the websites. Visit ImmoScout, ImmoWelt, Kleinanzeigen or WG-Gesucht,\n"
"make a search for the flat that you are looking for (e.g. pick a city), and\n"
"copy the URL here. You can add as many URLs as you like.\n\n")
if len(urls) > 0:
print("\n".join(urls))
print("")
result = prompt("Enter a target URL (or hit enter to continue): ",
validator=UrlsValidator(urls, config), validate_while_typing=False)
if len(result) > 0:
urls.append(result)
if len(result) == 0 and len(urls) == 0:
raise ConfigurationAborted()
first_run = False
return urls
def select_notifier(config: YamlConfig) -> str:
"""Select which notifier to use"""
if len(config.notifiers()) > 0:
default = config.notifiers()[0]
else:
default = Notifier.TELEGRAM.value
return radiolist_dialog(
values=[
(Notifier.TELEGRAM.value, "Telegram"),
(Notifier.MATTERMOST.value, "Mattermost"),
(Notifier.APPRISE.value, "Apprise"),
(Notifier.SLACK.value, "Slack"),
],
title="Configure notifications",
text="Choose a notification platform.",
default=default
).run()
def prompt_with_default(prompt_string: str, default_value: Optional[str]) -> str:
"""Prompt the user for input, providing an optional default value"""
if default_value is None:
return prompt(prompt_string)
return prompt(prompt_string, default = default_value)
def get_bot_token(config: YamlConfig) -> str:
"""Ask the user for the Telegram Bot token"""
clear()
print("Telegram Bot Token\n")
print("To send Telegram messages, we need a Telegram Bot Token. You can follow\n"
"the instructions here to track down 'The BotFather' and generate your token:\n"
"https://medium.com/geekculture/generate-telegram-token-for-bot-api-d26faf9bf064\n")
result = prompt_with_default("Enter Bot Token: ", config.telegram_bot_token())
if result is None or len(result) == 0:
raise ConfigurationAborted()
return result
def get_receiver_id(config: YamlConfig) -> str:
"""Ask the user for the target Telegram User ID for the Telegram notifications"""
clear()
print("Telegram Receiver ID\n")
print("Your Telegram Bot needs to know which user to send the notifications to.\n"
"This will normally be the User ID associated with your Telegram Account.\n"
"To work out your User ID, start a chat with the @userinfobot:\n"
"https://telegram.me/userinfobot\n")
current_receiver_id = None
if len(config.telegram_receiver_ids()) > 0:
current_receiver_id = str(config.telegram_receiver_ids()[0])
result = prompt_with_default("Enter Receiver ID: ", current_receiver_id)
if len(result) == 0:
raise ConfigurationAborted()
return result
def configure_telegram(config: YamlConfig) -> Dict[str, Any]:
"""Ask the user for details required for the Telegram configuration"""
bot_token = get_bot_token(config)
receiver_id = get_receiver_id(config)
return {
Notifier.TELEGRAM.value: {
"bot_token": bot_token,
"receiver_ids": [ receiver_id ]
}
}
def configure_mattermost(config: YamlConfig) -> Dict[str, Any]:
"""Ask the user for the mattermost webhook URL"""
clear()
print("Mattermost Webhook URL\n")
print("To receive messages over Mattermost, Flathunter will need the Webhook URL\n"
"of your Mattermost server.\n")
webhook_url = prompt_with_default("Enter Webhook URL: ", config.mattermost_webhook_url())
if len(webhook_url) == 0:
raise ConfigurationAborted()
return {
Notifier.MATTERMOST.value: {
"webhook_url": webhook_url
}
}
def configure_apprise(config: YamlConfig) -> Dict[str, Any]:
"""Ask the user for the apprise notification URL"""
clear()
print("Apprise notification URL\n")
print("To receive messages using Apprise, you need to supply a notification URL in the\n"
"apprise format, e.g. 'gotifys://...' or 'mailto://...'\n")
if len(config.apprise_urls()) > 0:
apprise_url = prompt("Enter Apprise notification URL: ", default=config.apprise_urls()[0])
else:
apprise_url = prompt("Enter Apprise notification URL: ")
if len(apprise_url) == 0:
raise ConfigurationAborted()
return {
Notifier.APPRISE.value: [ apprise_url ]
}
def configure_slack(config: YamlConfig) -> Dict[str, Any]:
"""Ask the user for the Slack webhook URL"""
clear()
print("Slack Webhook URL\n")
print("To receive messages over Slack, Flathunter will need the Webhook URL\n"
"of your Slack channel.\n")
webhook_url = prompt_with_default("Enter Webhook URL: ", config.slack_webhook_url())
if len(webhook_url) == 0:
raise ConfigurationAborted()
return {
Notifier.SLACK.value: {
"webhook_url": webhook_url
}
}
def configure_notifier(notifier: str, config) -> Dict[str, Any]:
"""Configure the selected / active notifier"""
if notifier == Notifier.TELEGRAM.value:
return configure_telegram(config)
if notifier == Notifier.MATTERMOST.value:
return configure_mattermost(config)
if notifier == Notifier.APPRISE.value:
return configure_apprise(config)
if notifier == Notifier.SLACK.value:
return configure_slack(config)
raise ConfigurationError("Invalid Notifier Selection")
def configure_captcha(urls: List[str], config: YamlConfig) -> Optional[Dict[str, Any]]:
"""Configure the captcha solver, where required"""
is_immoscout = reduce(lambda a,b: a or b,
[re.search(immobilienscout.STATIC_URL_PATTERN, url) for url in urls],
False)
if not is_immoscout:
return None
clear()
print("Captcha configuration\n")
print("Your search configuration includes URLs from ImmobilienScout24\n"
"To crawl ImmoScout, we need to browse the site with a real Chrome browser instance\n"
"and solve the Captcha that shows up on the ImmoScout site.\n")
print("You WILL NEED TO INSTALL google-chrome / chromium to solve Captchas\n")
print("We recommend using Capmonster (https://capmonster.cloud/) as your captcha-solving\n"
"service. You will need an account there with some credit on it.\n"
"IMPORTANT NOTICE: Buying captcha credit does not guarantee that Flathunter will be\n"
"able to bypass the bot detection on the ImmoScout site - pay at your own risk!!\n")
print("Once you have an account and have paid, enter the API Key here (or hit Enter\n"
"to skip Captcha configuration, but be aware that ImmoScout scraping will fail...)\n")
if config.get_capmonster_key() is not None:
api_key = prompt("Enter Capmonster API Key: ", default=config.get_capmonster_key())
else:
api_key = prompt("Enter Capmonster API Key: ")
if len(api_key) == 0:
return None
return {
"captcha": {
"capmonster": {
"api_key": api_key
},
"driver_arguments": [
"--no-sandbox",
"--headless",
"--disable-gpu",
"--remote-debugging-port=9222",
"--disable-dev-shm-usage",
"window-size=1024,768"
]
}
}
def load_config(existing) -> YamlConfig:
"""Load the existing (or default) config from disk"""
yaml = YAML()
source_file = "config.yaml.dist"
if existing:
source_file = "config.yaml"
with open(source_file, "r", encoding="utf-8") as dist_config:
config = yaml.load(dist_config)
return YamlConfig(config)
def save_config(config: Dict):
"""Save the configuration as 'config.yaml'"""
clear()
yaml = YAML()
with open("config.yaml", "w", encoding="utf-8") as config_file:
yaml.dump(config, config_file)
print("Configuration saved to 'config.yaml' - you're all set!")
def check_existing() -> bool:
"""Check to see if a configuration file already exists, prompt if so"""
if not os.path.exists("config.yaml"):
return False
result = button_dialog(
title="Config File Exists",
text="We found an existing 'config.yaml' file in the current directory\n"
"Running the wizard will update / edit this file. Do you want to proceed?",
buttons=[("Yes", True), ("No", False)],
).run()
if not result:
raise ConfigurationAborted()
return True
def main():
"""Run the configuration wizard"""
try:
welcome()
existing = check_existing()
config = load_config(existing)
config.init_searchers()
urls = gather_urls(config)
config.set_keys({ "urls": urls })
notifier = select_notifier(config)
config.set_keys({ "notifiers": [ notifier ]})
notifier_config = configure_notifier(notifier, config)
config.set_keys(notifier_config)
captcha_config = configure_captcha(urls, config)
if captcha_config is not None:
config.set_keys(captcha_config)
save_config(config.config)
except ConfigurationAborted:
print("Configuration was aborted by user action")
sys.exit(1)
if __name__ == "__main__":
main()