pyTooling Documentation
pyTooling is a powerful collection of arbitrary and useful (abstract) data models, lacking classes, decorators, a new performance boosting meta-class and enhanced exceptions. It also provides lots of helper functions e.g. to ease the handling of package descriptions or to unify multiple existing APIs into a single API.
It’s useful ‒ if not even essential ‒ for any Python-based project independent if it’s a library, framework, CLI tool or just a “script”.
In addition, pyTooling provides a collection of CI job templates for GitHub Actions. This drastically simplifies GHA-based CI pipelines for Python projects.
Package Details
The following descriptions and code examples give peak onto pyTooling’s highlights. But be ensured, there is more to explore, which can’t be highlighted on the main landing page.
Attributes
The pyTooling.Attributes module offers the base implementation of
.NET-like attributes
realized with Python decorators. The annotated and declarative data is stored as instances of
Attribute classes in an additional __pyattr__
field per class, method or
function.
The annotation syntax (decorator syntax) allows users to attach any structured data to classes, methods or
functions. In many cases, a user will derive a custom attribute from Attribute
and override the __init__
method, so user-defined parameters can be accepted when the attribute is constructed.
Later, classes, methods or functions can be searched for by querying the attribute class for attribute instance usage locations (see example to the right). Another option for class and method attributes is declaring a classes using pyTooling’s ExtendedType meta-class. Here the class itself offers helper methods for discovering annotated methods.
A SimpleAttribute class is offered accepting any positional and keyword parameters. In a more advanced use case, users are encouraged to derive their own attribute class hierarchy from Attribute.
from pyTooling.Attributes import Attribute
class Command(Attribute):
def __init__(self, cmd: str, help: str = "") -> None:
pass
class Flag(Attribute):
def __init__(self, param: str, short: str = None, long: str = None, help: str = "") -> None:
pass
@Command(cmd="version", help="Print version information.")
@Flag(param="verbose", short="-v", long="--verbose", help="Default handler.")
def Handler(self, args) -> None:
pass
for function in Command.GetFunctions():
pass
from pyTooling.Attributes import Attribute
from pyTooling.MetaClasses import ExtendedType
class TestCase(Attribute):
def __init__(self, name: str) -> None:
pass
class Program(metaclass=ExtendedType):
@TestCase(name="Handler routine")
def Handler(self, args) -> None:
pass
prog = Program()
for method, attributes in prog.GetMethodsWithAttributes(predicate=TestCase):
pass
from pyTooling.Attributes import Attribute
from pyTooling.MetaClasses import ExtendedType
class TestSuite(Attribute):
def __init__(self, name: str) -> None:
pass
@TestSuite(name="Command line interface tests")
class Program(metaclass=ExtendedType):
def Handler(self, args) -> None:
pass
prog = Program()
for testsuite in TestSuite.GetClasses():
pass
ArgParse
Defining commands, arguments or flags for a command line argument parser like argparse
is done imperatively.
This means code executed in-order defines how the parser will accept inputs. Then more user-defined code is needed
to dispatch the collected and type-converted arguments to handler routines. See an example to the right as
“Traditional argparse”.
In contrast, pyTooling.Attributes.ArgParse allows the definition of commands, arguments or flags as declarative code attached to handler routines using pyTooling’s attributes. This allow a cleaner and more readable coding style. Also maintainability is improved, as arguments are defined using clear attribute names attached to the matching handler routine. Thus parser and handler code is not separated.
If the command line interface uses many commands, handlers and their arguments can be spread across mixin classes. Later, the whole CLI is assembled by using multiple inheritance. In case handlers use shared argument sets, arguments can be grouped and shared by defining grouping attributes.
class Program:
def __init__(self) -> None:
mainParser = argparse.ArgumentParser()
mainParser.set_defaults(func=self.HandleDefault)
mainParser.add_argument("-v", "--verbose")
subParsers = mainParser.add_subparsers()
newUserParser = subParsers.add_parser("new-user", help="Add a new user.")
newUserParser.add_argument(dest="username", metaName="username", help="Name of the new user.")
newUserParser.add_argument("--quota", dest="quota", help="Max usable disk space.")
newUserParser.set_defaults(func=self.NewUserHandler)
deleteUserParser = subParsers.add_parser("delete-user", help="Delete a user.")
deleteUserParser.add_argument(dest="username", metaName="username", help="Name of the user.")
deleteUserParser.add_argument("-f", "--force", dest="force", help="Ignore internal checks.")
deleteUserParser.set_defaults(func=self.DeleteUserHandler)
listUserParser = subParsers.add_parser("list-user", help="List all users.")
listUserParser.set_defaults(func=self.ListUserHandler)
def HandleDefault(self, args) -> None:
pass
def NewUserHandler(self, args) -> None:
pass
def DeleteUserHandler(self, args) -> None:
pass
def ListUserHandler(self, args) -> None:
pass
class Program:
@DefaultHandler()
@FlagArgument(short="-v", long="--verbose", dest="verbose", help="Show verbose messages.")
def HandleDefault(self, args) -> None:
pass
@CommandHandler("new-user", help="Add a new user.")
@StringArgument(dest="username", metaName="username", help="Name of the new user.")
@LongValuedFlag("--quota", dest="quota", help="Max usable disk space.")
def NewUserHandler(self, args) -> None:
pass
@CommandHandler("delete-user", help="Delete a user.")
@StringArgument(dest="username", metaName="username", help="Name of the user.")
@FlagArgument(short="-f", long="--force", dest="force", help="Ignore internal checks.")
def DeleteUserHandler(self, args) -> None:
pass
@CommandHandler("list-user", help="List all users.")
def ListUserHandler(self, args) -> None:
pass
CLI Abstraction
pyTooling.CLIAbstraction offers an abstraction layer for command line programs, so they can be
used easily in Python. There is no need for manually assembling parameter lists or considering the order of
parameters. All parameters like -v
or --value=42
are described using nested classes on a
Program class. Each nested class derived from predefined argument classes knows about the
correct formatting pattern, character escaping, and if needed about necessary type conversions.
Such an instance of a program can be converted to an argument list suitable for subprocess.Popen
.
In stead of deriving from Program, abstracted command line tools can derive from
Executable which offers embedded Popen
behavior.
class Git(Executable):
def __new__(cls, *args: Tuple[Any, ...], **kwargs: Dict[str, Any]) -> self:
cls._executableNames = {
"Darwin": "git",
"FreeBSD": "git",
"Linux": "git",
"Windows": "git.exe"
}
return super().__new__(cls)
@CLIArgument()
class FlagHelp(ShortFlag, name="h"): ...
@CLIArgument()
class FlagVersion(LongFlag, name="version"): ...
@CLIArgument()
class CommandHelp(CommandArgument, name="help"): ...
@CLIArgument()
class CommandCommit(CommandArgument, name="commit"): ...
@CLIArgument()
class ValueCommitMessage(ShortTupleFlag, name="m"): ...
tool = Git()
tool[tool.FlagVersion] = True
tool.StartProcess()
Common Helper Functions
This is a set of useful helper functions:
firstElement, lastElement get the first/last element from an indexable.
firstItem, lastItem get the first/last item from an iterable.
firstKey, firstValue, firstPair get the first key/value/pair from an ordered dictionary.
getsizeof calculates the “real” size of a data structure.
isnestedclass checks if a class is nested inside another class.
mergedicts merges multiple dictionaries into a new dictionary.
zipdicts iterate multiple dictionaries simultaneously.
def myFunction(condition: bool) -> Iterable:
myList = [3, 21, 5, 7]
if condition:
return myList[0:2]
else
return myList[1:3]
beginOfSequence = myFunction(True)
first = firstItem(beginOfSequence)
# 3
from pyTooling.Common import mergedicts
dictA = {"a": 11, "b": 12}
dictB = {"x": 21, "y": 22}
for key, value in mergedicts(dictA, dictB):
pass
# ("a", 11)
# ("b", 12)
# ("x", 21)
# ("y", 22)
from pyTooling.Common import zipdicts
dictA = {"a": 11, "b": 12, "c": 13}
dictB = {"a": 21, "b": 22, "c": 23}
for key, valueA, valueB in zipdicts(dictA, dictB):
pass
# ("a", 11, 21)
# ("a", 12, 22)
# ("a", 13, 23)
Common Classes
Call-by-reference parameters: Python doesn’t provide call-by-reference parameters for simple types.
This behavior can be emulated with classes provided by thepyTooling.CallByRef
module.Unified license names: Setuptools, PyPI, and others have a varying understanding of license names.
ThepyTooling.Licensing
module provides unified license names as well as license name mappings or translations.Unified platform and environment description: Python has many ways in figuring out the current platform using APIs from
sys
,platform
,os
, …. Unfortunately, none of the provided standard APIs offers a comprehensive answer. pyTooling provides a CurrentPlatform singleton summarizing multiple platform APIs into a single class instance.Representations of version numbers: While Python itself has a good versioning schema, there are no classes provided to abstract a version numbers. pyTooling provides such representations following semantic versioning (SemVer) and calendar versioning (CalVer) schemes. The implementation can parse many common formats and allows user defined formatting. In addition, versions can be compared with various operators including PIPs
~=
operator.Measuring execution times can be achieved by using a stopwatch implementation providing start, pause, resume, split and stop features. Internally, Python’s high resolution clock is used. The stopwatch also provides a context manager, so it can be used in a
with
-statement.
pass
from pytest import mark
from unittest import TestCase
from pyTooling.Common import CurrentPlatform
class MyTests(TestCase):
@mark.skipif(not CurrentPlatform.IsNativeWindows, reason="Skipped, if platform isn't native Windows.")
def test_OnlyNativeWindows(self) -> None:
pass
@mark.skipif(not CurrentPlatform.IsMinGW64OnWindows, reason="Skipped, if platform isn't MinGW64.")
def test_OnlyMinGW64(self) -> None:
pass
@mark.skipif(CurrentPlatform.IsPyPy, reason="getsizeof: not supported on PyPy")
def test_ObjectSize(self) -> None:
pass
from pyTooling.Versioning import SemanticVersion, PythonVersion, CalendarVersion
version = SemanticVersion("v2.5.4")
version.Major
version.Minor
version.Patch
if version >= "2.5":
print(f"{version:%p%M.%m.%u}")
# Python versioning from sys.version_info
from pyTooling.Versioning import PythonVersion, CalendarVersion
pythonVersion = PythonVersion.FromSysVersionInfo()
# Calendar versioning
from pyTooling.Versioning import CalendarVersion
osvvmVersion = CalendarVersion.Parse("2024.07")
from pyTooling.Stopwatch import Stopwatch
sw = Stopwatch("my name", preferPause=True)
sw.Start()
# do something
sw.Pause()
with sw:
# do something
sw.Resume()
# do something
sw.Stop()
print(f"Start: {sw.StartTime}")
print(f"Stop: {sw.StopTime}")
print(f"Duration: {sw.Duration}")
print(f"Activity: {sw.Activity}")
print(f"Inactivity: {sw.Inactivity}")
print("Splits:")
for duration, activity in sw:
print(f" {'running for' if activity else 'paused for '} {duration}")
Configuration
Various file formats suitable for configuration information share the same features supporting: key-value pairs (dictionaries), sequences (lists), and simple types like string, integer and float. pyTooling provides an abstract configuration file data model supporting these features. Moreover, concrete configuration file format reader implementations are provided as well.
JSON configuration reader for the JSON file format.
TOML configuration reader → To be implemented.
YAML configuration reader for the YAML file format.
from pathlib import Path
from pyTooling.Configuration.JSON import Configuration
configFile = Path("config.json")
config = Configuration(configFile)
# Accessing root-level scalar value
configFileFormatVersion = config["version"]
# Accessing value in a sequence
firstItemInList = config["list"][0]
# Accessing first value in dictionary
firstItemInDict = config["dict"]["key_1"]
# Iterate simple list
simpleList = config["list"]
for item in simpleList:
pass
Todo
Needs example code
pass
from pathlib import Path
from pyTooling.Configuration.YAML import Configuration
configFile = Path("config.yml")
config = Configuration(configFile)
# Accessing root-level scalar value
configFileFormatVersion = config["version"]
# Accessing value in a sequence
firstItemInList = config["list"][0]
# Accessing first value in dictionary
firstItemInDict = config["dict"]["key_1"]
# Iterate simple list
simpleList = config["list"]
for item in simpleList:
pass
Todo
Needs example code
pass
Data Structures
pyTooling also provides fast and powerful data structures offering object-oriented APIs:
Graph data structure
→ A directed graph implementation using aVertex
and anEdge
class.Path data structure
→ To be documented.Finite State Machine data structure
→ A data model for state machines using aState
and aTransition
class.Tree data structure
→ A fast and simple implementation using a singleNode
class.
from pyTooling.Graph import Graph, Vertex
graph = Graph(name="myGraph")
# Create new vertices and an edge between them
vertex1 = Vertex(vertexID=1, graph=graph)
vertex2 = Vertex(vertexID=2, value="2", graph=graph)
edge12 = vertex1.EdgeToVertex(vertex2, edgeValue="1 -> 2", weight=15)
# Create an edge to a new vertex
edge2x = vertex2.EdgeToNewVertex(vertexID=3)
vertex3 = edge2x.Destination
# Create a link between two vertices
link31 = vertex3.LinkToVertex(vertex1)
Todo
Needs example code
pass
from pyTooling.Tree import Node
# Create a new tree by creating a root node (no parent reference)
root = Node(value="OSVVM Regression Tests")
# Construct the tree top-down
lib = Node(value="Utility Library", parent=root)
# Another standalone node with unique ID (actually an independent tree)
common = Node(nodeID=5, value="Common")
# Construct bottom-up
axi = Node(value="AXI")
axiCommon = Node(value="AXI4 Common")
axi.AddChild(axiCommon)
# Group nodes and handover children at node creation time
vcList = [common, axi]
vcs = Node(value="Verification Components", parent=root, children=vcList)
# Add multiple nodes at once
axiProtocols = (
Node(value="AXI4-Stream"),
Node(value="AXI4-Lite"),
Node(value="AXI4")
)
axi.AddChildren(axiProtocols)
# Create another standalone node and attach it later to a tree.
uart = Node(value="UART")
uart.Parent = vcs
Graph
Statemachine
Tree
Decorators
-
@abstractmethod: Methods marked with
@abstractmethod
are abstract and need to be overwritten in a derived class.
An abstract method might be called from the overwriting method.@mustoverride: Methods marked with
@mustoverride
are abstract and need to be overridden in a derived class.
It’s not allowed to call a mustoverride method.
-
@readonly: Methods marked with
@readonly
get transformed into a read-only property.⚠BROKEN⚠: Methods with @classproperty decorator transform methods to class-properties.
-
@export: Register a given function or class as publicly accessible in a module.
Functions and classes exposed like this are also used by Sphinx extensions to (auto-)document public module members.@InheritDocString: The decorator copies the doc-string from a given base-class to the annotated method.
-
@slotted: Classes marked with
@slotted
get transformed into classes using__slots__
.
This is achieve by exchanging the meta-class toExtendedType
.@mixin: Classes marked with
@mixin
do not store their fields in__slots__
.
When such a mixin-class is inherited by a class using slots, the fields of the mixin become slots.@singleton: Classes marked with
@singleton
get transformed into singleton classes.
This is achieve by exchanging the meta-class toExtendedType
.
-
@notimplemented: This decorator replaces a callable (function or method) with a callable raising a
NotImplementedError
. The original code becomes unreachable.
Todo
Needs example code
pass
Exceptions
EnvironmentException
… is raised when an expected environment variable is missing.PlatformNotSupportedException
… is raise if the platform is not supported.NotConfiguredException
… is raise if the requested setting is not configured.
Meta-Classes
pyTooling provides an enhanced meta-class called ExtendedType
to replace
the default meta-class type
. It combines features like using slots, abstract methods and creating singletons by
applying a single meta-class. In comparison, Python’s approach in to provide multiple specific meta-classes (see
abc
) that can’t be combined e.g. to a singleton using slots.
ExtendedType allows to implement slotted types, mixins, abstract and override methods and singletons, and combinations thereof. Exception messages in case of errors have been improved too.
Slotted types significantly reduce the memory footprint by 4x and decrease the class field access time by 10..25%. While setting up slotted types needed a lot of manual coding, this is now fully automated by this meta-class. It assumes, annotated fields are going to be slots. Moreover, it also takes care deferred slots in multiple-inheritance scenarios by marking secondary base-classes as mixins. This defers slot creation until a mixin is inherited.
class MyClass(metaclass=ExtendedType):
A class definition using the
ExtendedType
meta-class. I can now implement abstract methods using the decorators @abstractmethod or @mustoverride.class MyClass(metaclass=ExtendedType, singleton=True):
A class defined with enabled singleton behavior allows only a single instance of that class to exist. If another instance is going to be created, a previously cached instance of that class will be returned.
class MyClass(metaclass=ExtendedType, slots=True):
A class defined with enabled slots behavior stores instance fields in slots. The meta-class, translates all type-annotated fields in the class definition to slots. Slots allow a more efficient field storage and access compared to dynamically stored and accessed fields hosted in
__dict__
. This improves the memory footprint as well as the field access performance of all class instances. This behavior is automatically inherited to all derived classes.class MyClass(metaclass=ExtendedType, slots=True, mixin=True):
A class defined with enabled mixin behavior collects type-annotated instance fields so they can be added to slots in an inherited class. Thus, slots are not created for mixin-classes but deferred in the inheritance hierarchy.
class MyClass(SlottedObject):
A class definition deriving from
SlottedObject
will bring the slotted type behavior to that class and all its derived classes.
class Application(metaclass=ExtendedType, singleton=True):
_x: int
def __init__(self) -> None:
print("Instance of 'App1WithoutParameters' was created")
self._x = 10
instance1 = Application()
instance2 = Application()
assert instance1 is instance2
class Data(metaclass=ExtendedType, slots=True):
_x: int
_y: int = 12
def __init__(self, x: int) -> None:
self._x = x
data = Data(11)
Todo
Needs example code
def
Packaging
A set of helper functions to describe a Python package for setuptools.
Helper Functions:
pyTooling.Packaging.loadReadmeFile()
Load aREADME.md
file from disk and provide the content as long description for setuptools.pyTooling.Packaging.loadRequirementsFile()
Load arequirements.txt
file from disk and provide the content for setuptools.pyTooling.Packaging.extractVersionInformation()
Extract version information from Python source files and provide the data to setuptools.
Package Descriptions
from setuptools import setup
from pathlib import Path
from pyTooling.Packaging import DescribePythonPackage
pass
from setuptools import setup
from pathlib import Path
from pyTooling.Packaging import DescribePythonPackageHostedOnGitHub
gitHubNamespace = "Paebbels"
packageName = "pyVersioning"
packageDirectory = packageName.replace(".", "/")
packageInformationFile = Path(f"{packageDirectory}/__init__.py")
setup(
**DescribePythonPackageHostedOnGitHub(
packageName=packageName,
description="Write version information collected from (CI) environment for any programming language as source file.",
gitHubNamespace=gitHubNamespace,
sourceFileWithVersion=packageInformationFile,
consoleScripts={
"pyVersioning": "pyVersioning.CLI:main",
}
)
)
Terminal
The pyTooling.TerminalUI package offers a set of helpers to implement a text user interface (TUI) in a terminal. It’s designed on the idea that command line programs emit one line of text per message. Each message can be categorized as normal text, warnings, errors, and many more.
Therefore, this package offers a LineTerminal implementation, derived from a basic Terminal class. Of cause, it also includes colored outputs based on colorama.
Todo
Terminal helpers.
Todo
Needs example code
Contributors
Patrick Lehmann (Maintainer)
License
This Python package (source code) is licensed under Apache License 2.0.
The accompanying documentation is licensed under Creative Commons - Attribution 4.0 (CC-BY 4.0).