From dd9224edafc981627af6c2fee4230aa1cab70708 Mon Sep 17 00:00:00 2001 From: John Smith Date: Mon, 30 Dec 2024 21:22:17 +0100 Subject: [PATCH] First commit --- .gitignore | 3 + README.md | 18 +++ bootstrap.ps1 | 31 +++++ library/commons.py | 101 ++++++++++++++++ library/inputimeout.py | 75 ++++++++++++ resources/speedometer.ico | Bin 0 -> 41662 bytes resources/speedometer.svg | 10 ++ setup.bat | 2 + setup.py | 93 +++++++++++++++ speedtest.ini | 20 ++++ speedtest.py | 244 ++++++++++++++++++++++++++++++++++++++ urls.txt | 1 + 12 files changed, 598 insertions(+) create mode 100644 .gitignore create mode 100644 README.md create mode 100644 bootstrap.ps1 create mode 100644 library/commons.py create mode 100644 library/inputimeout.py create mode 100644 resources/speedometer.ico create mode 100644 resources/speedometer.svg create mode 100644 setup.bat create mode 100644 setup.py create mode 100644 speedtest.ini create mode 100644 speedtest.py create mode 100644 urls.txt diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..0ac85d4 --- /dev/null +++ b/.gitignore @@ -0,0 +1,3 @@ +__pycache__ +*.pyc +speedtest.log \ No newline at end of file diff --git a/README.md b/README.md new file mode 100644 index 0000000..40d7c6d --- /dev/null +++ b/README.md @@ -0,0 +1,18 @@ +# Speedtest + +A speedtest utiliy that is supposed to be ran on the end user machine. + +The utility sends email reports about average and maximum network speed to the pre-configured email address. + +# Setting up + +In order for the utility to perform any tests, consider filling two configuration files: + +* `speedtest.ini` — the file where you would put the logging, chunk size, number of samples and email settings. +* `urls.txt` — the file where you put direct links to the files you want the tests to be performed on. + +# Usage + +Just run `speedtest.py` or `speedtest.exe` after completing the configuration. + +Done! You're amazing! diff --git a/bootstrap.ps1 b/bootstrap.ps1 new file mode 100644 index 0000000..6fa771d --- /dev/null +++ b/bootstrap.ps1 @@ -0,0 +1,31 @@ +$url = "https://www.python.org/ftp/python/3.11.5/python-3.11.5-amd64.exe" +$output = "python311_installer.exe" +$install_dir = "C:\Python311" + +if (!(Test-Path $install_dir)) { + Write-Output "Python 3.11 installation not found. Installing..." + if (!(Test-Path $output)) { + $start_time = Get-Date + $wc = New-Object System.Net.WebClient + $wc.DownloadFile($url, $output) + #OR + (New-Object System.Net.WebClient).DownloadFile($url, $output) + Write-Output "Python 3.11 Downloaded. Time taken: $((Get-Date).Subtract($start_time).Seconds) second(s)" + } + Write-Output "Installing Python 3.11 to $install_dir" + Start-Process $output -ArgumentList "/quiet","InstallAllUsers=1","TargetDir=$install_dir" -NoNewWindow -Wait +} + +if (!(Test-Path $install_dir)) { + Write-Output "Python 3.11 installation still not found at $install_dir. Please repair your installation" + Read-Host -Prompt "Press Enter to continue" +} + +Write-Output "Installing packaging" +Start-Process "$install_dir\python.exe" -ArgumentList "-m pip install packaging" -Wait + +Write-Output "Installing setuptools" +Start-Process "$install_dir\python.exe" -ArgumentList "-m pip install setuptools" -Wait + +Write-Output "Installing py2exe" +Start-Process "$install_dir\python.exe" -ArgumentList "-m pip install py2exe" -Wait diff --git a/library/commons.py b/library/commons.py new file mode 100644 index 0000000..3465027 --- /dev/null +++ b/library/commons.py @@ -0,0 +1,101 @@ +# This file is a part of Teflon. + + +class Format(object): + pass + + +class Message(object): + + def __init__(self, contents: list): + self.contents = contents + self.server = None + + +class Cell(object): + + def __init__(self, content: str, args=None): + self.content = content + self.args = args or [] + + def add_arg(self, arg): + self.args.append(arg) + + +class Row(object): + + def __init__(self, cells=None): + self._cells = cells or [] + + def add_cell(self, cell: Cell): + self._cells.append(cell) + + def get_cells(self) -> list: + return self._cells + + +class TableFormat(Format): + + def __init__(self, rows: int, cols: int, rows_=None): + self.rows = rows + self.cols = cols + self._rows = rows_ or [] + + def add_row(self, row: Row): + self._rows.append(row) + + def get_rows(self) -> list: + return self._rows + + +class Text(object): + + def __init__(self, content, args=None): + self.content = content + self.args = args or [] + + def add_arg(self, arg): + self.args.append(arg) + + +class LineFormat(Format): + + def __init__(self, text=None): + self.text = text or [] + + def add_text(self, text): + self.text.append(text) + + +class Content(object): + + def __init__(self, head: str = "", body: str = "", footer: str ="\n"): + self.head = head + # body is a list of Body joined with EOL + self.body = body + self.footer = footer + + +class Connection(object): + + """ + Common connection interface + """ + def __init__(self, name: str): + self.name = name + self._notify_methods = [] + + def set_notify_methods(self, methods: list): + self._notify_methods = methods + + def get_notify_methods(self) -> list: + return self._notify_methods + + def connect(self): + raise NotImplementedError("Method should be overridden") + + def get_executor(self): + raise NotImplementedError("Method should be overridden") + + def disconnect(self): + raise NotImplementedError("Method should be overridden") diff --git a/library/inputimeout.py b/library/inputimeout.py new file mode 100644 index 0000000..4b0c8e9 --- /dev/null +++ b/library/inputimeout.py @@ -0,0 +1,75 @@ +# Code from https://github.com/johejo/inputimeout/ +import sys + +DEFAULT_TIMEOUT = 30.0 +INTERVAL = 0.05 + +SP = ' ' +CR = '\r' +LF = '\n' +CRLF = CR + LF + + +class TimeoutOccurred(Exception): + pass + + +def echo(string): + sys.stdout.write(string) + sys.stdout.flush() + + +def posix_inputimeout(prompt='', timeout=DEFAULT_TIMEOUT): + echo(prompt) + sel = selectors.DefaultSelector() + sel.register(sys.stdin, selectors.EVENT_READ) + events = sel.select(timeout) + + if events: + key, _ = events[0] + return key.fileobj.readline().rstrip(LF) + else: + echo(LF) + termios.tcflush(sys.stdin, termios.TCIFLUSH) + raise TimeoutOccurred + + +def win_inputimeout(prompt='', timeout=DEFAULT_TIMEOUT): + echo(prompt) + begin = time.monotonic() + end = begin + timeout + line = '' + + while time.monotonic() < end: + if msvcrt.kbhit(): + c = msvcrt.getwche() + if c in (CR, LF): + echo(CRLF) + return line + if c == '\003': + raise KeyboardInterrupt + if c == '\b': + line = line[:-1] + cover = SP * len(prompt + line + SP) + echo(''.join([CR, cover, CR, prompt, line])) + else: + line += c + time.sleep(INTERVAL) + + echo(CRLF) + raise TimeoutOccurred + + +try: + import msvcrt + +except ImportError: + import selectors + import termios + + inputimeout = posix_inputimeout + +else: + import time + + inputimeout = win_inputimeout diff --git a/resources/speedometer.ico b/resources/speedometer.ico new file mode 100644 index 0000000000000000000000000000000000000000..0f1e40e3ecded9c75d93cd968ae026444985576d GIT binary patch literal 41662 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 + + + + + + + + + diff --git a/setup.bat b/setup.bat new file mode 100644 index 0000000..6b8cea2 --- /dev/null +++ b/setup.bat @@ -0,0 +1,2 @@ +C:\Python311\python.exe setup.py +pause diff --git a/setup.py b/setup.py new file mode 100644 index 0000000..cae487b --- /dev/null +++ b/setup.py @@ -0,0 +1,93 @@ +# Windows py2exe build script +# running: python3 setup.py py2exe + +import os +import sys +import traceback +# noinspection PyUnresolvedReferences +import py2exe +import importlib + +from distutils.core import setup +from glob import glob + +sys.argv.append("py2exe") + +PYTHON_PATH = r"C:\Python311" + + +SETUP_DICT = { + "console": [{ + "script": "speedtest.py", + "icon_resources": [(1, "resources/speedometer.ico")], + "dest_base": "Speedtest" + + }], + + "zipfile": "lib/library.zip", + + "data_files": [ + ("", glob(os.path.join(WINDOWS_PATH, r"SYSTEM32\msvcp100.dll"))), + + ("", glob(os.path.join(WINDOWS_PATH, r"SYSTEM32\msvcr100.dll"))), + + ], + + "options": { + "py2exe": { + "bundle_files": 3, + "excludes": ["tcl", "tk", "tkinter", "idna", "lib2to3", "xmlrpc", "multiprocessing", "urllib3", "chardet", + "asyncio", "pydoc_data", "requests", "unittest", "pydoc", "lzma", "bz2"], + "dll_excludes": ["tcl86.dll", "tk86.dll"], + "compressed": True, + "optimize": 2 + + }, + }, + # fix for conflicting module dirs + "py_modules": [], + +} + +# Check if all files can be found +for num, file in enumerate(SETUP_DICT["data_files"]): + type_, file = file + if len(file) > 0: + file = file[0] + if os.path.exists(file): + print("Found file:", file) + else: + print("File not found:", file) + sys.exit(-1) + else: + if type_: + print("Data file of type", type_, "not found") + else: + print("Data file with index", num, "and name", file, "not found") + sys.exit(-1) + +try: + setup(**SETUP_DICT) +except Exception: + traceback.print_exc() + os.system("pause") +else: + # DDLs / exe files that are safe to be compressed with UPX + OPTIMIZE = [glob(r"dist\python311.dll"), + glob("dist\adb.exe"), + glob(r"dist\lib\Qt*"), + glob(r"dist\lib\*.pyd"), + glob(r"dist\lib\libcrypto-1_1.dll"), + glob(r"dist\lib\libssl-1_1.dll")] + + if os.path.exists("dist") and os.path.exists("upx.exe"): + for files in OPTIMIZE: + for file in files: + if os.path.exists(file): + # todo use subprocessing to run UPX on ALL CPU cores + os.system("upx --best %s" % file) + else: + print("Warning: not going to optimize dist size because upx.exe doesn't exist\n") + print("Please put upx.exe next to setup.py in order to have the build reduced in size by ~50%") + + os.system("pause") diff --git a/speedtest.ini b/speedtest.ini new file mode 100644 index 0000000..5c3ba7b --- /dev/null +++ b/speedtest.ini @@ -0,0 +1,20 @@ +[main] +# logging (DEBUG/INFO/WARNING/ERROR/CRITICAL) +log_level=DEBUG +log_filename=speedtest.log +# download chunk size in bytes +chunk_size=1024 +# sample every N chunks for averaging speed +sample_every=5 +# connection timeout. This is the timeout for the server to send the first response, NOT read timeout +connection_timeout=60 + +[mail] +smtp_host=smtp.mzjtechnology.com +smtp_login=test_account@mzjtechnology.com +smtp_password=test123456789 +smtp_port=587 +smtp_starttls=True +# optional +smtp_from=test@wiphone.io +smtp_sendto=me@helldev.net \ No newline at end of file diff --git a/speedtest.py b/speedtest.py new file mode 100644 index 0000000..5c0f68d --- /dev/null +++ b/speedtest.py @@ -0,0 +1,244 @@ +#!/usr/bin/env python3 +# This file is a part of speedtest +# Created at 12/28/2024 + +import os +import sys + + +core = getattr(sys.modules["__main__"], "__file__", None) +if core: + core = os.path.abspath(core) + root = os.path.dirname(core) + if root: + os.chdir(root) + + +sys.path.insert(0, "library") + + +import email.message +import email.utils +import io +import logging +import requests +import smtplib +import time +import traceback + +from configparser import ConfigParser + +from inputimeout import inputimeout, TimeoutOccurred +from commons import * + + +logger = logging.getLogger("speedtest") +formatter = logging.Formatter( + "%(asctime)s %(levelname)s: %(name)s: %(message)s", "%d.%m.%Y %H:%M:%S") + + +CONFIG_FILENAME = "speedtest.ini" +URLS_FILENAME = "urls.txt" + +MAIN_SECTION = "main" +MAIL_SECTION = "mail" + +CHUNK_SIZE = "chunk_size" +SAMPLE_EVERY = "sample_every" +CONNECTION_TIMEOUT = "connection_timeout" + +SMTP_HOST = "smtp_host" +SMTP_STARTTLS = "smtp_starttls" +SMTP_HOST = "smtp_host" +SMTP_PORT = "smtp_port" +SMTP_PASSWORD = "smtp_password" +SMTP_LOGIN = "smtp_login" +SMTP_SENDTO = "smtp_sendto" + +LOG_LEVEL = "log_level" +LOG_FILENAME = "log_filename" + +# YOU HAVE 20 SECONDS TO COMPLY +INPUT_TIMEOUT = 20 + + +def get_client(config: ConfigParser): + server = config[SMTP_HOST] + if config[SMTP_STARTTLS] == "True": + logger.debug("connecting to %s using starttls", server) + client = smtplib.SMTP(config[SMTP_HOST], config[SMTP_PORT]) + client.starttls() + + else: + client = smtplib.SMTP_SSL(config[SMTP_HOST], config[SMTP_PORT]) + client.login(config[SMTP_LOGIN], config[SMTP_PASSWORD]) + return client + + +def send_data(config: ConfigParser, body: str, messages: list): + client = get_client(config) + email_content = "" + for message in messages: + if not message.contents: + continue + email_content += "# Server: %s\n" % (message.server.name) + for content in message.contents: + email_content += content.head + email_content += "\n" + body = content.body + if isinstance(body, str): + email_content += body + + elif isinstance(body, LineFormat): + texts = body.text + for text in texts: + content_, args = text.content, text.args + print("args", args) + email_content += content_.format(*text.args) + email_content += "\n" + + email_content += content.footer + email_content += "\n" + email_content += "_" * 25 + email_content += "\n" + from_addr = "Speedtest <%s>" % (config.get(SMTP_LOGIN)) + msg = email.message.Message() + msg.add_header("Content-Type", "text") + msg.set_payload(email_content) + msg["Subject"] = "Speedtest report" + msg["From"] = from_addr + msg["To"] = config.get(SMTP_SENDTO) + client.sendmail(config.get(SMTP_LOGIN), + config.get(SMTP_SENDTO), msg.as_string()) + + +def measure_speed(url: str, chunk_size: int = 1024, sample_every: int = 5, timeout: int = 10) -> tuple[int, int]: + with io.BytesIO() as f: + start = time.perf_counter() + r = requests.get(url, stream=True, timeout=timeout) + total_length = r.headers.get("content-length") + dl = 0 + min_ = sys.maxsize + max_ = 0 + samples = [] + counter = 0 + if total_length is None: # no content length header + # fixme: we can pre-configure the length + raise ValueError("Content length header not found!") + else: + for chunk in r.iter_content(chunk_size): + dl += len(chunk) + f.write(chunk) + speed = dl // (time.perf_counter() - start) / 1000000 + if (counter % sample_every) == 0: + samples.append(speed) + max_ = max(max_, speed) + min_ = min(min_, speed) + counter += 1 + avg = sum(samples) / len(samples) + return (avg, max_) + + +def test_url(url: str, chunk_size: int, sample_every: int, timeout: int) -> Message: + avg = 0 + max_ = 0 + try: + avg, max_ = measure_speed(url, chunk_size, sample_every, timeout) + except requests.exceptions.Timeout: + logger.error("Timeout to reach url: %s" % url, exc_info=True) + except Exception: + logger.error("Unable to test url: %s" % url, exc_info=True) + else: + text = [] + text.append(Text("Average speed: {:.2f} MB/s", [avg])) + text.append(Text("Maximum speed: {:.2f} MB/s", [max_])) + content = Content("Speed test results", LineFormat(text)) + message = Message([content]) + message.server = Connection(url) + message.server.name = url + return message + + +def get_test_results(urls: list, chunk_size: int, sample_every: int, timeout: int): + messages = [] + for url in urls: + logger.debug("Going to perform speedtest on %s" % url) + message = test_url(url, chunk_size, sample_every, timeout) + if message is not None: + messages.append(message) + else: + logger.warning("The url %s was not tested" % url) + return messages + + +def _get_urls() -> list: + with open(URLS_FILENAME) as file: + return file.read().splitlines() + + +def _setup_logging(config: ConfigParser): + log_level = config[LOG_LEVEL] + log_filename = config[LOG_FILENAME] + log_level = logging.getLevelName(log_level) + logger.setLevel(log_level) + console_handler = logging.StreamHandler(sys.stdout) + console_handler.setFormatter(formatter) + logger.addHandler(console_handler) + file_handler = logging.FileHandler(log_filename) + file_handler.setFormatter(formatter) + logger.addHandler(file_handler) + + +def excepthook(*exc_info): + text = "".join(traceback.format_exception(*exc_info)) + logger.error("unhandled exception caught: %s", text) + + +def _get_input(prompt: str, timeout: int = INPUT_TIMEOUT, default: str="") -> str: + try: + return inputimeout(prompt, timeout) + except TimeoutOccurred: + print("(fallback to default: %s)" % default) + return default + + +def get_name_and_isp() -> tuple[str, str]: + print("In order to proceed, we need to ask you a couple questions. Every question has a 20 second timeout, then the default value will be used.") + name = _get_input("Please input your name: ", INPUT_TIMEOUT, "undefined") + isp = _get_input("Please input your ISP name: ", INPUT_TIMEOUT, "undefined") + return name, isp + + +def main(): + sys.excepthook = excepthook + parser = ConfigParser() + parser.read(CONFIG_FILENAME) + main_section = parser[MAIN_SECTION] + mail_section = parser[MAIL_SECTION] + + _setup_logging(main_section) + sample_every = int(main_section[SAMPLE_EVERY]) + chunk_size = int(main_section[CHUNK_SIZE]) + timeout = int(main_section[CONNECTION_TIMEOUT]) + + urls = _get_urls() + logger.debug("Found %d urls" % len(urls)) + name, isp = get_name_and_isp() + messages = get_test_results(urls, chunk_size, sample_every, timeout) + if messages is None or len(messages) == 0: + print("There was an error, no tests were performed!") + raise RuntimeError("No tests performed!") + + client = get_client(mail_section) + body = "User name: %s; ISP: %s" % (name, isp) + send_data(mail_section, body, messages) + logger.info("Done! You're amazing!") + + +if __name__ == "__main__": + try: + main() + except Exception: + logger.error("Critical failure: ", exc_info=True) + if os.name == "nt": + os.system("pause") diff --git a/urls.txt b/urls.txt new file mode 100644 index 0000000..2b0de8b --- /dev/null +++ b/urls.txt @@ -0,0 +1 @@ +http://ipv4.download.thinkbroadband.com/100MB.zip