diff --git a/packages/internet_detector.vm/internet_detector.vm.nuspec b/packages/internet_detector.vm/internet_detector.vm.nuspec new file mode 100644 index 000000000..3dd20ba9f --- /dev/null +++ b/packages/internet_detector.vm/internet_detector.vm.nuspec @@ -0,0 +1,14 @@ + + + + internet_detector.vm + 1.0.0 + Elliot Chernofsky and Ana Martinez Gomez + Tool that changes the background and a taskbar icon if it detects internet connectivity + + + + + + + diff --git a/packages/internet_detector.vm/tools/chocolateyinstall.ps1 b/packages/internet_detector.vm/tools/chocolateyinstall.ps1 new file mode 100644 index 000000000..163f17728 --- /dev/null +++ b/packages/internet_detector.vm/tools/chocolateyinstall.ps1 @@ -0,0 +1,37 @@ +$ErrorActionPreference = 'Stop' +Import-Module vm.common -Force -DisableNameChecking + +$toolName = 'internet_detector' +$category = 'Utilities' +$packageToolDir = "$(Split-Path -parent $MyInvocation.MyCommand.Definition)" + +# Create tool directory +$toolDir = Join-Path ${Env:RAW_TOOLS_DIR} $toolName +New-Item -Path $toolDir -ItemType Directory -Force -ea 0 +VM-Assert-Path $toolDir + +# Install dependency for windows api +VM-Pip-Install "pywin32" +# Install dependency for creating python executable +VM-Pip-Install "pyinstaller" + +Start-Process -FilePath 'cmd.exe' -ArgumentList "/c pyinstaller --onefile -w --distpath $toolDir --workpath $packageToolDir --specpath $packageToolDir $packageToolDir\internet_detector.pyw" -Wait + +# Move images to %VM_COMMON_DIR% directory +$imagesPath = Join-Path $packageToolDir "images" +Copy-Item "$imagesPath\*" ${Env:VM_COMMON_DIR} -Force + +VM-Install-Shortcut $toolName $category $toolDir + +# TODO - Uncomment when FakeNet BlackList for DNS is fixed. https://github.com/mandiant/flare-fakenet-ng/issues/190 +# # Create scheduled task for tool to run every 2 minutes. +# $action = New-ScheduledTaskAction -Execute $rawToolPath +# $trigger = New-ScheduledTaskTrigger -Once -At (Get-Date) -RepetitionInterval (New-TimeSpan -Minutes 2) +# Register-ScheduledTask -Action $action -Trigger $trigger -TaskName 'Internet Detector' -Force + +# Copy updated `default.ini` with tool added to ProcessBlackList to FakeNet Config Directory +$fakenetConfigDir = Join-Path ${Env:UserProfile} "Desktop\fakenet_logs.lnk" +if (Test-Path $fakenetConfigDir) { # Check if the shortcut exists + $targetPath = (New-Object -ComObject WScript.Shell).CreateShortcut($fakenetConfigDir).TargetPath + Copy-Item "$packageToolDir\default.ini" -Destination $targetPath +} diff --git a/packages/internet_detector.vm/tools/chocolateyuninstall.ps1 b/packages/internet_detector.vm/tools/chocolateyuninstall.ps1 new file mode 100644 index 000000000..94da3c136 --- /dev/null +++ b/packages/internet_detector.vm/tools/chocolateyuninstall.ps1 @@ -0,0 +1,8 @@ +$ErrorActionPreference = 'Continue' +Import-Module vm.common -Force -DisableNameChecking + +$toolName = 'internet_detector' +$category = 'Utilities' + +VM-Uninstall $toolName $category +Unregister-ScheduledTask -TaskName 'Internet Detector' -Confirm:$false diff --git a/packages/internet_detector.vm/tools/default.ini b/packages/internet_detector.vm/tools/default.ini new file mode 100644 index 000000000..ec6c82a10 --- /dev/null +++ b/packages/internet_detector.vm/tools/default.ini @@ -0,0 +1,349 @@ +############################################################################### +# Fakenet Configuration + +[FakeNet] + +# Specify whether or not FakeNet should divert traffic. Disable if you want to +# just start listeners and direct traffic manually (e.g. modify DNS server) +DivertTraffic: Yes + +############################################################################### +# Diverter Configuration + +[Diverter] + +# Specify what mode of operation to use. Options: +# SingleHost - manipulate local traffic +# MultiHost - manipulate traffic from foreign hosts +# Auto - Use SingleMode on Windows or use MultiHost on Linux +# +# The current support for these modes on each supported platform is as follows: +# | Windows | Linux | +# -----------+------------+--------------+ +# SingleHost | Functional | Experimental | +# MultiHost | - | Functional | +# NetworkMode: SingleHost +# NetworkMode: MultiHost +NetworkMode: Auto + +# DebugLevel: specify fine-grained debug print flags to enable. Enabling all +# logging when verbose mode is selected results in overwhelming output, hence +# this setting. Valid values (comma-separated) are: +# +# GENPKT Generic packet information +# GENPKTV Packet analysis, displays IP, TCP, UDP fields, very wide output +# CB Diverter packet handler callback start/finish logging +# NONLOC Nonlocal packet verbose logging +# DPF Dynamic port forwarding decisions +# DPFV Dynamic port forwarding table activity +# IPNAT NAT decisions +# MANGLE Packet mangling (modification) activity +# PCAP PCAP writes of original and mangled packets +# IGN Cases where packets are forwarded as is +# FTP FTP-specific logic +# IGN-FTP Cases where packets are forwarded as is due to FTP Active Mode +# MISC Miscellaneous +# NFQUEUE NetfilterQueue activity (Linux only) +# PROCFS Procfs read/write activity (Linux only) +# IPTABLES iptables firewall rule activity (Linux only) +DebugLevel: Off + +# Restrict which interface on which Fakenet-NG will intercept and handle +# packets. Specify (only) one interface and Fakenet-NG will ignore all other +# interfaces. This feature only applies to interfaces on different subnets. +# Specify interface by name only (ex: eth0). To disable, set to "Off". In +# order to run multiple instance of Fakenet-NG on different interfaces within +# the same guest, LinuxFlushIptables must be turned off to avoid the latest +# instance flushing the rules associated with other instances or restoring +# rules to an incorrect state upon exit. +LinuxRestrictInterface: Off + +# Set LinuxFlushIptables to Yes to have the Linux Diverter flush all iptables +# rules before adding its FakeNet-NG-specific rules to iptables. This setting +# also restores rules via `iptables-restore` when it exits, unless its +# termination is interrupted. +LinuxFlushIptables: Yes + +# Incorporated so that users of the binary release may make this work for +# various Linux distros. On Ubuntu, this is `service dns-clean restart`. For +# other distributions, it may be `nscd -I hosts`. Check your manual for +# details. +LinuxFlushDNSCommand: service dns-clean restart + +# Specify whether or not to save captured traffic. You can also change +# the file prefix for the generated PCAPs. +DumpPackets: Yes +DumpPacketsFilePrefix: packets + +# DHCP server running under VMWare Host-Only networking does not configure +# interface gateway and DNS server. Gateway must be configured to allow +# Windows to attempt to route external traffic so that FakeNet could +# could intercept it. This option will automatically generate and set +# appropriate gateway and DNS addresses to allow normal operation. +FixGateway: Yes +FixDNS: Yes + +# Enable 'ModifyLocalDNS' to statically set DNS server to the local machine. +# Linux: Modifies (and restores) /etc/resolv.conf on Linux to make this an +# ephemeral change. +ModifyLocalDNS: Yes + +# Enable 'StopDNSService' to stop Windows DNS client to see the actual +# processes resolving domains. This is a no-op on Linux, until such time as DNS +# caching is observed to interfere with finding the pid associated with a DNS +# request. +StopDNSService: Yes + +# Enable 'RedirectAllTraffic' to optionally divert traffic going to ports not +# specifically listed in one of the listeners below. 'DefaultTCPListener' and +# 'DefaultUDPListener' will handle TCP and UDP traffic going to unspecified ports. +# +# NOTE: Setting default UDP listener will intercept all DNS traffic unless you +# enable a dedicated UDP port 53 DNS listener or add UDP port 53 to the +# 'BlackListPortsUDP' below so that system's default DNS server is used instead. + +RedirectAllTraffic: Yes +DefaultTCPListener: ProxyTCPListener +DefaultUDPListener: ProxyUDPListener + +# Specify TCP and UDP ports to ignore when diverting packets. +# For example, you may want to avoid diverting UDP port 53 (DNS) traffic +# when trying to intercept a specific process while allowing the rest to +# function normally +# +# NOTE: This setting is only honored when 'RedirectAllTraffic' is enabled. + +BlackListPortsTCP: 139 +BlackListPortsUDP: 67, 68, 137, 138, 443, 1900, 5355, 53 + +# Specify processes to ignore when diverting traffic. Windows example used +# here. +ProcessBlackList: internet_detector.exe + +# Specify processes to consider when diverting traffic (others will be +# ignored). Linux examples used here. +# ProcessWhiteList: wget, nc + +# Specify hosts to ignore when diverting traffic. +# HostBlackList: 6.6.6.6 + +############################################################################### +# Listener Configuration +# +# Listener configuration consists of generic settings used by the diverter which +# are the same for all listeners and listener specific settings. +# +# NOTE: Listener section names will be used for logging. +# +# NOTE: Settings labels are not case-sensitive. +# +# The following settings are available for all listeners: +# * Enabled - specify whether or not the listener is enabled. +# * Port - TCP or UDP port to listen on. +# * Protocol - TCP or UDP +# * Listener - Listener name to handle traffic. +# * ProcessWhiteList - Only traffic from these processes will be modified +# and the rest will simply be forwarded. +# * ProcessBlackList - Traffic from all but these processes will be simply forwarded +# and the rest will be modified as needed. +# * HostWhiteList - Only traffic to these hosts will be modified and +# the rest will be simply forwarded. +# * HostBlackList - Traffic to these hosts will be simply forwarded +# and the rest will be modified as needed. +# * ExecuteCmd - Execute command on the first connection packet. This is feature is useful +# for extending FakeNet-NG's functionality (e.g. launch a debugger on the +# connecting pid to help with unpacking and decoding.) +# +# The following format string variables are made available: +# * {pid} - process id +# * {procname} - process executable name +# * {src_addr} - source address +# * {src_port} - source port +# * {dst_addr} - destination address +# * {dst_port} - destination port +# +# Listener entry which does not specify a specific listener service +# will still redirect all packets to the local machine on the specified port and +# subject to all the filters (processes, hosts, etc.). However, you must set-up a +# third party service (e.g. proxy servers) to accept these connections. This feature can be +# used to provide FakeNet-NG's passive traffic diverting and filtering capabilities to other +# applications. +# +# Listener specific settings: +# +# * Timeout - Set connection timeout for any listeners that support +# TCP connections (e.g. RawListener, DNSListener, HTTPListener +# SMTPListener). +# * UseSSL - Enable SSL support on the listener (RawListener, HTTPListener) +# * Webroot - Set webroot path for HTTPListener. +# * DumpHTTPPosts - Store HTTP Post requests for the HTTPListener. +# * DumpHTTPPostsFilePrefix - File prefix for the stored HTTP Post requests used by the HTTPListener. +# * TFTPFilePrefix - File prefix for the stored tftp uploads used by the TFTPListener. +# * DNSResponse - IP address to respond with for A record DNS queries. (DNSListener) +# * NXDomains - A number of DNS requests to ignore to let the malware cycle through +# all of the backup C2 servers. (DNSListener) +# * Banner - FTPListener, IRCListener: FTP or IRC banner to display. +# Valid settings are any banner string, or ! where +# is a valid key in the BANNERS dictionary within +# FTPListener.py or IRCListener.py, or !random to +# randomize among the banners in the BANNERS dictionary. +# The default value if none is specified is !generic, +# which selects the banner in the BANNERS dictionary going +# by that key. Banner string may specify the following +# escapes/insertions: +# {servername} - ServerName setting value +# {tz} - Time zone, currently hard-coded to 'UTC' +# * ServerName - FTPListener, IRCListener: FTP or IRC server name for +# insertion into selected default banners or into a +# user-specified banner string. Valid settings are any +# hostname string, !hostname to insert the actual hostname +# of the system, or !random to generate a random hostname +# between 1 and 15 characters (inclusive). + +[ProxyTCPListener] +Enabled: True +Protocol: TCP +Listener: ProxyListener +Port: 38926 +Listeners: HTTPListener, RawListener, FTPListener, DNSListener, POPListener, SMTPListener, TFTPListener, IRCListener +Hidden: False + +[ProxyUDPListener] +Enabled: True +Protocol: UDP +Listener: ProxyListener +Port: 38926 +Listeners: RawListener, DNSListener, TFTPListener, FTPListener +Hidden: False + +[Forwarder] +Enabled: False +Port: 8080 +Protocol: TCP +ProcessWhiteList: chrome.exe +Hidden: False + +[RawTCPListener] +Enabled: True +Port: 1337 +Protocol: TCP +Listener: RawListener +UseSSL: No +Timeout: 10 +Hidden: False +# To read about customizing responses, see docs/CustomResponse.md +# Custom: sample_custom_response.ini + +[RawUDPListener] +Enabled: True +Port: 1337 +Protocol: UDP +Listener: RawListener +UseSSL: No +Timeout: 10 +Hidden: False +# To read about customizing responses, see docs/CustomResponse.md +# Custom: sample_custom_response.ini + +[FilteredListener] +Enabled: False +Port: 31337 +Protocol: TCP +Listener: RawListener +UseSSL: No +Timeout: 10 +ProcessWhiteList: ncat.exe, nc.exe +HostBlackList: 5.5.5.5 +Hidden: False + +[DNS Server] +Enabled: True +Port: 53 +Protocol: UDP +Listener: DNSListener +ResponseA: 192.0.2.123 +ResponseMX: mail.evil2.com +ResponseTXT: FAKENET +NXDomains: 0 +Hidden: False + +[HTTPListener80] +Enabled: True +Port: 80 +Protocol: TCP +Listener: HTTPListener +UseSSL: No +Webroot: defaultFiles/ +Timeout: 10 +DumpHTTPPosts: Yes +DumpHTTPPostsFilePrefix: http +Hidden: False +# To read about customizing responses, see docs/CustomResponse.md +# Custom: sample_custom_response.ini + +[HTTPListener443] +Enabled: True +Port: 443 +Protocol: TCP +Listener: HTTPListener +UseSSL: Yes +Webroot: defaultFiles/ +Timeout: 10 +DumpHTTPPosts: Yes +DumpHTTPPostsFilePrefix: http +Hidden: False + +[SMTPListener] +Enabled: True +Port: 25 +Protocol: TCP +Listener: SMTPListener +UseSSL: No +Hidden: False + +[FTPListener21] +Enabled: True +Port: 21 +Protocol: TCP +Listener: FTPListener +UseSSL: No +FTProot: defaultFiles/ +PasvPorts: 60000-60010 +Hidden: False +Banner: !generic +ServerName: !gethostname + +[FTPListenerPASV] +Enabled: True +Port: 60000-60010 +Protocol: TCP +Hidden: False + +[IRCServer] +Enabled: True +Port: 6667 +Protocol: TCP +Listener: IRCListener +UseSSL: No +Banner: !generic +ServerName: !gethostname +Timeout: 30 +Hidden: False + +[TFTPListener] +Enabled: True +Port: 69 +Protocol: UDP +Listener: TFTPListener +TFTPRoot: defaultFiles/ +Hidden: False +TFTPFilePrefix: tftp + +[POPServer] +Enabled: True +Port: 110 +Protocol: TCP +Listener: POPListener +UseSSL: No +Hidden: False + diff --git a/packages/internet_detector.vm/tools/images/background-internet.png b/packages/internet_detector.vm/tools/images/background-internet.png new file mode 100644 index 000000000..b380cb483 Binary files /dev/null and b/packages/internet_detector.vm/tools/images/background-internet.png differ diff --git a/packages/internet_detector.vm/tools/images/indicator_off.ico b/packages/internet_detector.vm/tools/images/indicator_off.ico new file mode 100644 index 000000000..c58bc6871 Binary files /dev/null and b/packages/internet_detector.vm/tools/images/indicator_off.ico differ diff --git a/packages/internet_detector.vm/tools/images/indicator_on.ico b/packages/internet_detector.vm/tools/images/indicator_on.ico new file mode 100644 index 000000000..e65a6e2f1 Binary files /dev/null and b/packages/internet_detector.vm/tools/images/indicator_on.ico differ diff --git a/packages/internet_detector.vm/tools/internet_detector.pyw b/packages/internet_detector.vm/tools/internet_detector.pyw new file mode 100644 index 000000000..3b22c25ed --- /dev/null +++ b/packages/internet_detector.vm/tools/internet_detector.pyw @@ -0,0 +1,425 @@ +# This tool checks if internet connectivity exists by reaching out to specific websites and checking if they return expected values and +# display the current state via changes to the background, theme, and icon in the taskbar. +# * It works even with a tool like FakeNet running (provided it uses the default configuration) +# If internet is detected, the tool: +# - Changes the background to the image "%VM_COMMON_DIR%\background-dynamic.png". +# - Change the the taskbar to be transparent and if the theme is dark mode also the color to "Rose". +# - Shows the icon "%VM_COMMON_DIR%\indicator_on.ico" in the taskbar. +# If internet is NOT detected, the tool: +# - Changes the background to the image "%VM_COMMON_DIR%\background.png" (default background image). +# - Changes the taskbar to be transparent keeping the current theme/color (dark or light). +# - Shows the icon "%VM_COMMON_DIR%\indicator_off.ico" in the taskbar. +VERSION = "1.0.0" +TOOL_NAME = "internet_detector" + +import threading +import requests +import win32api +import win32gui +import win32con +import urllib3 +import winreg +import signal +import ctypes +import time +import os +import re + + +# Define constants +CHECK_INTERVAL = 2 # Seconds +CONNECT_TEST_URL_AND_RESPONSES = { + "https://www.msftconnecttest.com/connecttest.txt": "Microsoft Connect Test", # HTTPS Test #1 + "http://www.google.com": "Google", # HTTP Test + "https://www.wikipedia.com": "Wikipedia", # HTTPS Test #2 + "https://www.youtube.com": "YouTube", # HTTPS Test #3 +} +SPI_SETDESKWALLPAPER = 20 +SPIF_UPDATEINIFILE = 0x01 +SPIF_SENDWININICHANGE = 0x02 +WM_DWMCOLORIZATIONCOLORCHANGED = 0x0320 +ACCENT_PREVALENCE = 1 +ROSE_COLOR = "#C30052" # Rose Color = #ff5200c3 which correlates to #C30052 before parsing +ROSE_ACCENT_PALETTE = "FFABCE00FF7FB400F74A9200C30052008C003A0069002C004D002000567C7300" +ICON_INDICATOR_ON = os.path.join(os.environ.get("VM_COMMON_DIR"), "indicator_on.ico") +ICON_INDICATOR_OFF = os.path.join(os.environ.get("VM_COMMON_DIR"), "indicator_off.ico") +DEFAULT_BACKGROUND = os.path.join(os.environ.get("VM_COMMON_DIR"), "background.png") +INTERNET_BACKGROUND = os.path.join(os.environ.get("VM_COMMON_DIR"), "background-internet.png") + +# Global variables +tray_icon = None +stop_event = threading.Event() # To signal the background thread to stop +hwnd = None # We'll assign the window handle here later +check_thread = None +tray_icon_thread = None +default_transparency = None +default_color_prevalence = None +default_color = None +default_palette = None +# Win32 API icon handles +hicon_indicator_off = None +hicon_indicator_on = None + + +def signal_handler(sig, frame): + global check_thread, tray_icon_thread, tray_icon + print("Ctrl+C detected. Exiting...") + stop_event.set() # Signal the background thread to stop + if check_thread: + check_thread.join() + if tray_icon_thread: + tray_icon_thread.join() + if tray_icon: + del tray_icon + exit(0) + + +def load_icon(icon_path): + try: + return win32gui.LoadImage(None, icon_path, win32con.IMAGE_ICON, 0, 0, win32con.LR_LOADFROMFILE) + except Exception as e: + print(f"Error loading indicator icon: {e}") + return None + + +class SysTrayIcon: + def __init__(self, hwnd, icon, tooltip): + self.hwnd = hwnd + self.icon = icon + self.tooltip = tooltip + # System tray icon data structure + self.nid = ( + self.hwnd, + 0, + win32gui.NIF_ICON | win32gui.NIF_MESSAGE | win32gui.NIF_TIP, + win32con.WM_USER + 20, + self.icon, + self.tooltip, + ) + # Add the icon to the system tray + win32gui.Shell_NotifyIcon(win32gui.NIM_ADD, self.nid) + + def set_tooltip(self, new_tooltip): + self.tooltip = new_tooltip + self.nid = ( + self.hwnd, + 0, + win32gui.NIF_ICON | win32gui.NIF_MESSAGE | win32gui.NIF_TIP, + win32con.WM_USER + 20, + self.icon, + self.tooltip, + ) + win32gui.Shell_NotifyIcon(win32gui.NIM_MODIFY, self.nid) + + def set_icon(self, icon): + self.icon = icon + self.nid = ( + self.hwnd, + 0, + win32gui.NIF_ICON | win32gui.NIF_MESSAGE | win32gui.NIF_TIP, + win32con.WM_USER + 20, + self.icon, + self.tooltip, + ) + win32gui.Shell_NotifyIcon(win32gui.NIM_MODIFY, self.nid) + + def __del__(self): + # Remove the icon from the system tray when the object is destroyed + win32gui.Shell_NotifyIcon(win32gui.NIM_DELETE, self.nid) + + +def get_current_color_prevalence(): + try: + # Open the registry key for accent palette settings + key = winreg.OpenKey( + winreg.HKEY_CURRENT_USER, + r"Software\Microsoft\Windows\CurrentVersion\Themes\Personalize", + 0, + winreg.KEY_READ + ) + value, _ = winreg.QueryValueEx(key, "ColorPrevalence") + winreg.CloseKey(key) + return value + + except WindowsError: + print("Error accessing registry.") + return None + + +# Reads color palette binary data from a registry key and returns it as a hex string. +def get_current_color_palette(): + try: + # Open the registry key for accent palette settings + key = winreg.OpenKey( + winreg.HKEY_CURRENT_USER, + r"Software\Microsoft\Windows\CurrentVersion\Explorer\Accent", + 0, + winreg.KEY_READ + ) + value, _ = winreg.QueryValueEx(key, "AccentPalette") + winreg.CloseKey(key) + # return the data as hex string + return value.hex() + + except WindowsError: + print("Error accessing registry.") + return None + + +# Attempts to get the current taskbar color based on user personalization settings and returns it in hex format. +def get_current_taskbar_color(): + try: + # Open the registry key for personalization settings + key = winreg.OpenKey( + winreg.HKEY_CURRENT_USER, + r"Software\Microsoft\Windows\CurrentVersion\Explorer\Accent", + 0, + winreg.KEY_READ + ) + accent_color, _ = winreg.QueryValueEx(key, "AccentColorMenu") + winreg.CloseKey(key) + + # Convert the accent color value (DWORD) to RGB + r = accent_color & 0xFF + g = (accent_color >> 8) & 0xFF + b = (accent_color >> 16) & 0xFF + # Convert RGB to hex + return f"#{r:02X}{g:02X}{b:02X}" + + except WindowsError: + print("Error accessing registry.") + return None + + +def set_taskbar_accent_color(hex_color, hex_color_palette, color_prevalence): + """ + Sets the accent color location in the Windows registry. + + Args: + hex_color = the hex value of RGB color with starting "#" (e.g., '#RRGGBB') + hex_color_palette = the hex string of the corresponding color paletter for the provided hex_color + color_prevalence = 0 - don't use accent color; 1 - use accent color + """ + try: + print("Setting taskbar color to:", hex_color) + # Convert hex color to RGB + r = int(hex_color[1:3], 16) + g = int(hex_color[3:5], 16) + b = int(hex_color[5:7], 16) + + # Convert RGB to DWORD + color_value = (b << 16) | (g << 8) | r + + key = winreg.OpenKey( + winreg.HKEY_CURRENT_USER, + r"Software\Microsoft\Windows\CurrentVersion\Themes\Personalize", + 0, + winreg.KEY_ALL_ACCESS + ) + winreg.SetValueEx(key, "AccentColorPolicy", 0, winreg.REG_DWORD, 1) # Set color for taskbar + winreg.SetValueEx(key, "ColorPrevalence", 0, winreg.REG_DWORD, color_prevalence) # Enable accent color + winreg.SetValueEx(key, "AccentColorInactive", 0, winreg.REG_DWORD, 0) # Disable inactive color + winreg.CloseKey(key) + + key = winreg.OpenKey( + winreg.HKEY_CURRENT_USER, + r"Software\Microsoft\Windows\CurrentVersion\Explorer\Accent", + 0, + winreg.KEY_ALL_ACCESS, + ) + binary_data = bytes.fromhex(hex_color_palette) + winreg.SetValueEx(key, "AccentColorMenu", 0, winreg.REG_DWORD, color_value) + winreg.SetValueEx(key, "AccentPalette", 0, winreg.REG_BINARY, binary_data) + winreg.CloseKey(key) + + except WindowsError as e: + print(f"Error accessing or modifying registry: {e}") + + +def get_transparency_effects(): + try: + key = winreg.OpenKey( + winreg.HKEY_CURRENT_USER, + r"Software\Microsoft\Windows\CurrentVersion\Themes\Personalize", + 0, + winreg.KEY_ALL_ACCESS, + ) + value, _ = winreg.QueryValueEx(key, "EnableTransparency") + winreg.CloseKey(key) + return value + + except WindowsError as e: + print(f"Error accessing or modifying registry: {e}") + + +def set_transparency_effects(value): + try: + key = winreg.OpenKey( + winreg.HKEY_CURRENT_USER, + r"Software\Microsoft\Windows\CurrentVersion\Themes\Personalize", + 0, + winreg.KEY_ALL_ACCESS, + ) + + winreg.SetValueEx(key, "EnableTransparency", 0, winreg.REG_DWORD, value) + winreg.CloseKey(key) + except WindowsError as e: + print(f"Error accessing or modifying registry: {e}") + + +# Attempt to extract a known good value in response. +def extract_title(data): + match = re.search(r"(.*?)", data) + if match: + return match.group(1) + else: + return None + + +def check_internet(): + for url, expected_response in CONNECT_TEST_URL_AND_RESPONSES.items(): + try: + # Perform internet connectivity tests + response = requests.get(url, timeout=5, verify=False) + if expected_response in (extract_title(response.text) or response.text): + print(f"Internet connectivity detected via URL: {url}") + return True + except: + pass + return False + + +def check_internet_and_update_tray_icon(): + global tray_icon, hicon_indicator_off, hicon_indicator_on, default_color + if check_internet(): + tray_icon.set_icon(hicon_indicator_on) + tray_icon.set_tooltip("Internet Connection: Detected") + set_transparency_effects(1) + set_taskbar_accent_color(ROSE_COLOR, ROSE_ACCENT_PALETTE, ACCENT_PREVALENCE) + # Set the background to internet connection background + if get_wallpaper_path() != INTERNET_BACKGROUND: # Checked so program isn't continuously setting the wallpaper + set_wallpaper(INTERNET_BACKGROUND) + else: + tray_icon.set_icon(hicon_indicator_off) + tray_icon.set_tooltip("Internet Connection: Not Detected") + set_transparency_effects(default_transparency) + set_taskbar_accent_color(default_color, default_palette, default_color_prevalence) # change color back to what user had originally + # Reset background when internet is not detected + if get_wallpaper_path() != DEFAULT_BACKGROUND: # Checked so program isn't continuously setting the wallpaper + set_wallpaper(DEFAULT_BACKGROUND) + + +def check_internet_loop(): + while not stop_event.is_set(): + check_internet_and_update_tray_icon() + time.sleep(CHECK_INTERVAL) + + +def tray_icon_loop(): + global hwnd, tray_icon, hicon_indicator_off, hicon_indicator_on, stop_event + # Load icons + hicon_indicator_off = load_icon(ICON_INDICATOR_OFF) + hicon_indicator_on = load_icon(ICON_INDICATOR_ON) + + # Wait for hwnd to be initialized + while hwnd is None: + time.sleep(0.1) + + if hicon_indicator_off is None or hicon_indicator_on is None: + print("Error: Failed to load icons. Exiting TrayIconThread.") + return + + tray_icon = SysTrayIcon(hwnd, hicon_indicator_off, "Internet Detector") + + while not stop_event.is_set(): + msg = win32gui.PeekMessage(hwnd, 0, 0, 0) + if msg and len(msg) == 6: + win32gui.TranslateMessage(msg) + win32gui.DispatchMessage(msg) + time.sleep(0.1) + + +def get_wallpaper_path(): + """Attempts to retrieve the path to the current wallpaper image.""" + # Try to get the path from the registry (for wallpapers set through Windows settings) + try: + key = winreg.OpenKey(winreg.HKEY_CURRENT_USER, r"Control Panel\Desktop", 0, winreg.KEY_READ) + value, _ = winreg.QueryValueEx(key, "Wallpaper") + winreg.CloseKey(key) + if value: + return value + except WindowsError: + pass + + # Check for cached wallpaper files (if the above fails) + cached_files_dir = os.path.join(os.getenv("APPDATA"), r"Microsoft\Windows\Themes\CachedFiles") + transcoded_wallpaper_path = os.path.join(os.getenv("APPDATA"), r"Microsoft\Windows\Themes\TranscodedWallpaper") + + for file in os.listdir(cached_files_dir): + if file.endswith((".jpg", ".jpeg", ".bmp", ".png")): + return os.path.join(cached_files_dir, file) + + if os.path.exists(transcoded_wallpaper_path): + return transcoded_wallpaper_path + + # If all else fails, return None + return None + + +def set_wallpaper(image_path): + """Sets the desktop wallpaper to the image at the specified path.""" + print(f"Setting wallpaper to: {image_path}") + result = ctypes.windll.user32.SystemParametersInfoW( + SPI_SETDESKWALLPAPER, 0, image_path, SPIF_UPDATEINIFILE | SPIF_SENDWININICHANGE + ) + if not result: + print("Error setting wallpaper. Make sure the image path is correct.") + + +def main_loop(): + global stop_event, check_thread, tray_icon_thread, tray_icon + # Create and start the threads + tray_icon_thread = threading.Thread(target=tray_icon_loop) + check_thread = threading.Thread(target=check_internet_loop) + + tray_icon_thread.start() + # Wait for the tray icon to finish initializing + while tray_icon is None: + time.sleep(0.1) + + check_thread.start() + + while not stop_event.is_set(): + time.sleep(1) + + +if __name__ == "__main__": + signal.signal(signal.SIGINT, signal_handler) + urllib3.disable_warnings(urllib3.exceptions.InsecureRequestWarning) + default_transparency = get_transparency_effects() + default_color_prevalence = get_current_color_prevalence() + default_color = get_current_taskbar_color() + default_palette = get_current_color_palette() + + # Create a hidden window to receive messages (required for system tray icons) + def wndProc(hwnd, msg, wparam, lparam): + if lparam == win32con.WM_LBUTTONDBLCLK: + print("Left button double clicked") + elif msg == win32con.WM_COMMAND: + if wparam == 1023: # Example menu item ID + print("Exit selected") + win32gui.DestroyWindow(hwnd) + return win32gui.DefWindowProc(hwnd, msg, wparam, lparam) + + wc = win32gui.WNDCLASS() + hinst = wc.hInstance = win32api.GetModuleHandle(None) + wc.lpszClassName = "Internet Detector" + wc.lpfnWndProc = wndProc + classAtom = win32gui.RegisterClass(wc) + hwnd = win32gui.CreateWindow(classAtom, "Internet Detector", 0, 0, 0, 0, 0, 0, 0, hinst, None) + + print(f"{TOOL_NAME} Version: {VERSION}") + print(f"Current wallpaper: {get_wallpaper_path()}") + print(f"Current color: {default_color}") + + main_loop()