""" This is PyWebServer, an ultra minimalist webserver, meant to still have a lot standard webserver features. A comprehensive list is below: Features: HTTP and HTTPS support. Automatically generate certificates using AutoCertGen plugin. Blocking per host. Easy port configuration. Easy to understand documentation and configuration. Very small size, compared to something like Apache and NGINX. No compromise(-ish) security: Directory traversal attack prevention. No fuss HTTPS setup. Per-host blocking. Ability for per-IP blocking. Ability for per-UA blocking. Simple to understand and mod codebase. All GNU GPL-3-or-above license. (Do with it what you want.) Library aswell as a standalone script: You can easily get access to other parts of the script if you need it. """ import os import threading import ssl import socket import signal import sys try: from autocertgen import AutoCertGen except ImportError: print( "WARN: You need the AutoCertGen plugin! Please install it from\n" "https://git.novacow.ch/Nova/AutoCertGen/" ) class FileHandler: CONFIG_FILE = "pywebsrv.conf" DEFAULT_CONFIG = ( "port:8080\nport-https:8443\nhttp:1" "\nhttps:0\ndirectory:{cwd}\nhost:localhost" "allow-all:1\nallow-localhost:1" ) def __init__(self, base_dir=None): self.base_dir = base_dir or os.path.join(os.getcwd(), "html") self.config_path = os.path.join(os.getcwd(), self.CONFIG_FILE) def check_first_run(self): if not os.path.isfile(self.config_path): self.on_first_run() return True return False def on_first_run(self): with open(self.config_path, "w") as f: f.write(self.DEFAULT_CONFIG.format(cwd=os.getcwd())) def read_file(self, file_path): if "../" in file_path: return 403 full_path = os.path.join(self.base_dir, file_path.lstrip("/")) if not os.path.isfile(full_path): return 404 try: with open(full_path, "rb") as f: return f.read() except Exception as e: print(f"Error reading file {full_path}: {e}") return 500 def write_file(self, file_path, data): if "../" in file_path: return 403 full_path = os.path.join(self.base_dir, file_path.lstrip("/")) with open(full_path, "a") as f: f.write(data) return 0 def read_config(self, option): """ clean code, whats that???? TODO: docs """ option = option.lower() valid_options = [ "port", "directory", "host", "http", "https", "port-https", "allow-all", "allow-nohost", "allow-localhost", "disable-autocertgen", ] if option not in valid_options: return None with open(self.config_path, "r") as f: for line in f: if line.startswith("#"): continue try: key, value = line.strip().split(":", 1) except ValueError: return None key = key.lower() if key == option: if option == "host": seperated_values = value.split(",", 0) return [value.lower() for value in seperated_values] if option == "port" or option == "port-https": return int(value) if ( option == "http" or option == "https" or option == "allow-all" or option == "allow-localhost" or option == "disable-autocertgen" or option == "allow-nohost" ): return bool(int(value)) return value return None def autocert(self): """ Generate some self-signed certificates using AutoCertGen """ autocert = AutoCertGen() pk = autocert.generate_private_key() sub, iss = autocert.generate_issuer_and_subject() cert = autocert.build_cert(pk, iss, sub) autocert.write_cert(pk, cert) class RequestParser: def __init__(self): self.allowed_methods_file = "allowedmethods.conf" self.file_handler = FileHandler() self.hosts = self.file_handler.read_config("host") self.all_allowed = self.file_handler.read_config("allow-all") def parse_request_line(self, line): """Parses the HTTP request line.""" try: method, path, version = line.split(" ") if path.endswith("/"): path += "index.html" return method, path, version except ValueError: return None, None, None def is_method_allowed(self, method): """ Checks if the HTTP method is allowed. Reads allowed methods from a configuration file. Falls back to allowing only 'GET' if the file does not exist. Should (for now) only be GET as I haven't implemented the logic for PUT """ allowed_methods = ["GET"] if os.path.isfile(self.allowed_methods_file): with open(self.allowed_methods_file, "r") as f: allowed_methods = [line.strip() for line in f] return method in allowed_methods def host_parser(self, host): """ Parses the host and makes sure it's allowed in Mfw im in an ugly code writing contest and my opponent is nova while writing a side project """ host = str(host) if ":" in host: host = host.split(":", 1)[0] host = host.lstrip() if ( host == "localhost" or host == "127.0.0.1" ) and self.file_handler.read_config("allow-localhost"): return True if host not in self.hosts and self.all_allowed is False: return False elif host not in self.hosts and self.all_allowed is True: return True class WebServer: def __init__( self, http_port=8080, https_port=8443, cert_file="cert.pem", key_file="key.pem" ): self.http_port = http_port self.https_port = https_port self.cert_file = cert_file self.key_file = key_file self.file_handler = FileHandler() self.parser = RequestParser() self.skip_ssl = False # me when no certificate and key file if not os.path.exists(self.cert_file) or not os.path.exists(self.key_file): print("WARN: No HTTPS certificate was found!") if self.file_handler.read_config("disable-autocertgen") is True: print("WARN: AutoCertGen is disabled, ignoring...") self.skip_ssl = True else: choice = input("Do you wish to generate an HTTPS certificate? [y/N] ") if choice.lower() == "y": self.file_handler.autocert() else: self.skip_ssl = True # TODO: change this to something like oh no you fucked up, go fix idiot self.no_host_req_response = ( "Connecting via this host is disallowed\r\n" "You may also be using a very old browser!\r\n" "Ask the owner of this website to set allow-all to 1!" ) # ipv6 when????/??//?????//? self.http_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM) self.http_socket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1) self.http_socket.bind(("0.0.0.0", self.http_port)) self.https_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM) self.https_socket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1) self.https_socket.bind(("0.0.0.0", self.https_port)) if self.skip_ssl is False: # https gets the ssl treatment!! yaaaay :3 self.ssl_context = ssl.create_default_context(ssl.Purpose.CLIENT_AUTH) self.ssl_context.load_cert_chain( certfile=self.cert_file, keyfile=self.key_file ) self.https_socket = self.ssl_context.wrap_socket( self.https_socket, server_side=True ) self.http_404_html = ( "
Running PyWebServer/1.1
" "Running PyWebServer/1.1
" "Running PyWebServer/1.1
" "