forked from Speyedr/socialclub-notification-blocker
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmain.py
executable file
·193 lines (161 loc) · 9.14 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
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
"""
SocialClub Notification Blocker
Author: Daniel "Speyedr" Summer
License: GPLv3
Source: https://github.com/Speyedr/socialclub-notification-blocker
"""
# 28th February - 4 hours
# 1st March - 9 hours
# 2nd March - 4 hours
# 3rd March - 7 hours
# 4th March - 3 hours (mostly preparing release stuff like repo, guides, posts, etc)
import pydivert # for cx_Freeze?
from filter import DropLengthSettings, Filter, FilterFlags, FilterSettings
from msvcrt import kbhit, getch # For getting keyboard input.
from time import sleep, time
from menu_options import Menu, MenuNames
from os import system
from webbrowser import open as open_in_browser
from logger import Logger
from multiprocessing import Queue as ProxyQueue, freeze_support
from traceback import format_exc
from settings import Settings
from ctypes import windll
from sys import exit as sys_exit
DONATE = "https://ko-fi.com/speyedr"
VERSION = "0.1.1"
WINDOW_NAME = "Speyedr's SocialClub Notification Blocker v" + VERSION
UI_WAIT_TIME = 0.01
UI_TOP_MARGIN = 1
UI_MSG_PREFIX = '\n' * UI_TOP_MARGIN
LOG_FILE = "debug.log"
NULL_BYTE = b'\x00'
def get_filter_flags():
flags = FilterFlags(False)
if Menu.IS_DROP_INC_80: flags |= FilterFlags.DROP_INC_80
if Menu.IS_DROP_CLIENT_POST: flags |= FilterFlags.DROP_CLIENT_POST
if Menu.IS_DROP_LENGTH: flags |= FilterFlags.DROP_LENGTH
return flags
def get_filter_settings():
return FilterSettings(get_filter_flags(), get_drop_lengths())
def get_drop_lengths():
return DropLengthSettings(max_offset=500)
def is_capslock_enabled():
keycode_caps_lock = 0x14
return windll.user32.GetKeyState(keycode_caps_lock)
def wait_for_valid_key(menu, interval=UI_WAIT_TIME):
while True:
if kbhit(): # User has pressed a key
key = getch() # Save what key was pressed
if is_capslock_enabled():
key = key.lower() # Make sure that CAPS LOCK doesn't affect user input
try:
if key != NULL_BYTE:
return menu[key.decode()] # See if there was an option associated with this key
except (KeyError, UnicodeDecodeError):
pass # Invalid keys are ignored
sleep(interval)
def trigger_auto_action(menu_item):
try:
menu_item["action"](menu_item["name"]) # Try to trigger automatic action
except KeyError:
pass # Didn't have an action
def main():
start = time()
Logger.static_add_message("Booting up program.", LOG_FILE)
logger = Logger(ProxyQueue(), LOG_FILE)
logger.add_message("Logger initialized.")
logger.start() # Run the logger all the time. Pass the queue to Filter when necessary.
logger.add_message("Logger process is now running.")
is_filter_running = False # filter will not be running at boot
config = Settings(Menu.get_options()) # load settings, will save default options if doesn't exist
Menu.set_options(config.get_settings()) # set options according to config file
logger.add_message("Loaded config settings: " + str(config.get_settings()))
Menu.update_menu() # force update
filter_obj = Filter(get_filter_settings())
previous_menu_state = None # helps check if overall state changed
should_exit = False # shouldn't exit immediately
while not should_exit:
# At this point in the UI loop, we should check if menu state changed.
Menu.update_menu() # Update settings before printing
system('cls') # Clear the command prompt screen
current_menu_state = Menu.get_options()
#if previous_menu_state != current_menu_state: # If a setting has changed
if True: # DEBUG: Force update every time
logger.add_message("Menu state has changed: " + str(previous_menu_state) + " -> " + str(current_menu_state))
config.write_settings(current_menu_state) # Update config file
previous_menu_state = current_menu_state
if Menu.IS_FILTER_RUNNING: # Filter should be running
settings = get_filter_settings()
if settings.flags == FilterFlags(False): # No filters are being applied
logger.add_message("WARN: User attempted to start network filter with no filters applied.")
print(Menu.ERR_MSG_NO_FILTERS) # Tell the user something's wrong
Menu.IS_FILTER_RUNNING = False # Filter should no longer be running
config.write_settings(Menu.get_options()) # Force updated an option
Menu.update_menu() # Force menu to update (since we're overriding a value)
current_menu_state = Menu.get_options() # The menu state has now changed (probably bad fix)
else:
logger.add_message("Starting network filter...")
# Because Python multiprocessing is jank as hell, we can't pass the entire Logger object to
# Filter otherwise we get a "weakref" error. Instead, the server / proxy queue object is shared and
# the FilterSettings instance will put items directly in the queue for processing.
logger_queue = logger.queue if Menu.IS_LOGGING else None
filter_obj = Filter(FilterSettings(get_filter_flags(),
drop_lengths=get_drop_lengths(),
logger_queue=logger_queue))
filter_obj.start() # Re-initialize filter with new settings
logger.add_message("Started network filter.")
if is_filter_running and not Menu.IS_FILTER_RUNNING:# Filter is currently running but shouldn't be
logger.add_message("Stopping network filter...")
filter_obj.stop()
logger.add_message("Successfully stopped network filter.")
is_filter_running = Menu.IS_FILTER_RUNNING # Update filter state
# Now we have processed the menu changes, we can prompt again.
print(UI_MSG_PREFIX+Menu.construct_print_options(Menu.MAIN_OPTIONS))
option = wait_for_valid_key(Menu.MAIN_OPTIONS) # Wait until we get a valid option
trigger_auto_action(option) # Attempt to trigger its' automatic action
if option["name"] == MenuNames.ADJUST_FILTER:
should_go_back = False
while not should_go_back:
Menu.update_menu()
system('cls')
print(UI_MSG_PREFIX+Menu.construct_print_options(Menu.FILTER_OPTIONS)) # New Menu to print
option2 = wait_for_valid_key(Menu.FILTER_OPTIONS)
trigger_auto_action(option2)
if option2["name"] == MenuNames.FILTER_SETTINGS_GO_BACK: # Time to go back
should_go_back = True
if option["name"] == MenuNames.OPEN_DONATION_URL:
logger.add_message("User selected OPEN_DONATION_URL...")
open_in_browser(DONATE)
sleep(3) # Pause the UI thread so we don't open 10 billion pages
if option["name"] == MenuNames.EXIT_PROGRAM:
logger.add_message("User selected EXIT_PROGRAM...")
if is_filter_running:
logger.add_message("Stopping network filter...")
filter_obj.stop() # Stop the filter if it is running
logger.add_message("Successfully stopped network filter.")
should_exit = True # Inform the UI loop that we're done
# Bye!
logger.add_message("Program ended from menu")
logger.stop()
logger.add_message("Logger process has been stopped.")
if __name__ == "__main__":
freeze_support()
system('mode con: cols=92 lines=30') # make console wider to fit all text
windll.kernel32.SetConsoleTitleW(WINDOW_NAME)
if not windll.shell32.IsUserAnAdmin(): # need admin perms
Logger.static_add_message("User attempted to run program without Administrator permissions", LOG_FILE)
print("\n This program requires Administrator permissions to run.\n"
" Press any key to continue...")
while True:
if kbhit(): sys_exit(1)
sleep(0.5)
try:
main()
except (Exception, KeyboardInterrupt) as e:
if isinstance(e, KeyboardInterrupt):
Logger.static_add_message("User ended program forcefully", LOG_FILE)
else:
Logger.static_add_message("Oops! Looks like the program crashed. Reason: " + str(e), LOG_FILE)
Logger.static_add_message(format_exc(), LOG_FILE)
raise # still need to "crash" the program after logging the error