Plumbing C++ with a Python frontend via ctypes.
Project description
Funktionsklempner
A library and framework for plumbing C++ code to Python via ctypes.
Why ctypes? It has the advantage of being widely available with Python itself, and interfaces C ABIs that can remain stable for long periods of time. This way, a compiled library can work across many different Python versions in which a dynamically linked extension would have to be recompiled due to binary incompatibility. The maintenance burden due to the recurring need to recompile packages is reduced. This comes at the cost of the additional overhead for having to pipe data through a C/ctypes API. Funktionsklempner generates this wrapping code automatically, which reduces the cost to the computational overhead. You may find Funktionsklempner useful for your open source project with low version iteration rate.
Funktionsklempner's basic workflow is as follows:
- Declare the interface in a C++ header (i.e. functions whose equivalents shall be passed through to the Python world).
- Declare all such headers in a project-wide
funktionsklempner.tomlfile. - Generated wrapper source files via the
funktionsklempnercommand. - Check the generated wrapper files and check them into your version control system.
- Keep the auto-generated files updated via a git pre-commit hook.
Usage
Funktionsklempner assumes that you have created a C++ library with an interface that uses the Funktionsklempner (marker) types. For the sake of brevity, let's use the example in the example subdirectory; i.e. a project that has an example.cpp source file and whose interface with the Python world is designed in example.hpp. Specifically, the important part of the example's interface can be illustrated using the single function
int test_function(
const char* name,
size_t i,
double* result
);
that takes a string name and an integer i, does something with it, and returns a double as result. Note in this interface the pointer-to-double type of result: to transfer values to Python via ctypes, Funktionsklempner uses preallocated ctypes variables and passes pointers to them when calling the compiled library. The C++ side is then expected to write the return value into this provided pointer. The actual return value of the function is unused; Funktionsklempner convention is to use void.
Note that the actual function in example.hpp looks differently! What I have just described is the mechanism that Funktionsklempner makes use of, which is a bit cumbersome to do by hand. To write more explicit code and to make the parser of Funktionsklempner understand the intention of this function definition, it is annoted using macros and alias types defined in funktionsklempner/funktionsklempner.hpp and funktionsklempner/scalar.hpp:
FUNKTIONSKLEMPNER_EXPORT_FUNCTION
int test_function(
funktionsklempner::input<const char*> name,
funktionsklempner::input<size_t> i,
funktionsklempner::output<double> result
);
The first macro FUNKTIONSKLEMPNER_EXPORT_FUNCTION tells the parser that what follows is the declaration of a function that shall be callable from the Python world. The templated type alias funktionsklempner::input<T> tells the parser that the variables name and i are input variables passed by value T, and the alias funktionsklempner::output<T> tells the parser that result is a return value of the function (corresponding to a T* input parameter). Note that you shall name your variables when exporting via the FUNKTIONSKLEMPNER_EXPORT_FUNCTION macro.
What does Funktionsklempner do with this C++ function declaration? On the Python side, it will generate a function that takes a string name, an int i, and returns a float, i.e.
def test_function(
name: str,
i: int
) -> float:
...
On the C++ side, it will wrap test_function defined in the header by an exception handling function (any uncaught exceptions passing through to the Python world would crash the program!) that caches any occurring error messages. Between this C++ wrapper and the Python test_function, glue code will initialize ctypes variables and watch out for such error messages, raising errors on the Python side appropriately.
All glue code is generated via the funktionsklempner generate command after the header example.hpp has been properly described in the project-wide funktionsklempner.toml. The following two sections describe both the funktionsklempner command line tool as well as the funktionsklempner.toml configuration file format.
funktionsklempner.toml
The funktionsklempner.toml file collects all files in a source code repository that shall be wrapped via the funktionsklempner command. It has to be placed in the root of the packages source code directory and it is required for the funktionsklempner command line utility to work. As an example of its layout, consider the funktionsklempner.toml file of the example subdirectory:
[package]
name = "funkybadger"
root = "funkybadger"
[funkybadger]
[funkybadger.example]
header = "cpp/example.hpp"
sources = [ "cpp/example.cpp" ]
funktionsklempner_namespace = "fk"
First, note that the [package] entry---including the name and root fields---is mandatory. The package.name field describes the name of the Python packge, and the package.root field describes the location, relative to the source code directory's root, of the package's source code (see src layout vs flat layout, where package.root would correspond to the awesome_package folder).
Second, there must be one entry per wrapped shared library. In the example, this is [funkybadger.example]. All of these entries must be under a root entry that corresponds to the package name (here [funkybadger]). The entry's key must correspond to the module name which shall be generated and from which the wrapped functions shall be imported (here the module funkybadger.example). Each shared library entry must contain two fields:
header, a string denoting the path to the Funktionsklempner-annotated, API-defining header file (hereexample/example.hpp). The path shall be relative to the source directory's root.sources, a list of (strings denoting paths to) source files which encompass the shared library.
Optional fields
There are further optional fields that can be used to tweak the code that Funktionsklempner injects into the Meson build file. This way, necessary information (such as a static library that must be linked to) can be passed to the autogenerated Meson build code. The following optional fields are available:
link_with, a list of strings denotingstatic_librarytargets of the injection-targeted Meson build file which the Python shared library shall be linked with.dependencies, a list of strings denotingdependencyobjects of the injection-targeted Meson build files on which the Python shared library depends.wrapper_source, a file name specifying the wrapper C++ source file relative to the repository's root.include_root, a directory name specifying the root include directory for the library. If not given, will default to the directory in whichheaderresides.funktionsklempner_namespace, a C++ namespace alias for thefunktionsklempnernamespace (e.g.fkin the above example). This alias will be parsed in theheaderfile instead offunktionsklempner, which allows the header to be less verbose.
As an illustration, the code
[package.library]
header = "..."
sources = [ "..." ]
link_with = [ "linklib" ]
dependencies = [ "a_dependency" ]
wrapper_source = "cpp/src/library_wrapped.cpp"
in the funktionsklempner.toml configuration file will generate a wrapper source file at cpp/src/library_wrapped.cpp (relative to the repository root), and inject the following code into the project's Meson build file:
package_library = shared_library(
'library',
# ...
'cpp/src/library_wrapped.cpp',
dependencies: [
a_dependency,
],
link_with: [
linklib,
],
# ...
)
Command line
The Funktionsklempner package comes with the funktionsklempner command line utility. This program sets up the Meson subprojects and git pre-commit hook, injects the project's main Meson build file, and generates the wrapper code. It has three main commands, two of which are typically used by users themselves.
funktionsklempner init
This command needs to be called once after the funktionsklempner.toml configuration file has been written. The command performs the following tasks:
- Setup the
funktionsklempner,objektbuch,gcem-1.18.0, and (optional)boost-1.90.0Meson subprojects that are required for the automatic error propagation to the Python side. - Setup the git pre-commit hook that internally uses
funktionsklempner verifyto check that the generated code currently residing in the repository is up to date. - Start injecting the Funktionsklempner code into the project's Meson build file.
Boost will be preferably used from the system side, but particulary during the pip package building, Boost might not be found. In this case, the reduced subset of Boost (shipping code relevant to Boost.Units) will act as a fallback subproject.
funktionsklempner generate
This command is the main workhorse. It generates or updates wrapper codes for all libraries described in funktionsklempner.toml, and updates the injection into the project's Meson build file. Call this command whenever a definition in one of the API-defining headers changes.
funktionsklempner verify
This command checks whether the potential result of funktionsklempner generate is identical to what is currently found in the source tree, raising an error if not. This command is used by the git pre-commit hook to ensure that the autogenerated code stays up to date with the API header definitions.
Header annotation syntax
The introductory example lays out the basic syntax for annotating C++ headers using FUNKTIONSKLEMPNER_* macros. The following macros are available:
| Macro | Purpose | Scope |
|---|---|---|
FUNKTIONSKLEMPNER_EXPORT_FUNCTION |
Export the following function to Python module | Top level |
FUNKTIONSKLEMPNER_EXPORT_STRUCT |
Export the following structure to Python module | Top level |
FUNKTIONSKLEMPNER_SIZE_CONTROL("X", "Y") |
Declare variable X as an (output) array whose size is controlled by the (input) array(s) or integer Y |
Function export |
FUNKTIONSKLEMPNER_ELLIPSOID_CONTROL("X", "Y") |
Declare variable X, an output geographic point array (GPA), has its ellipsoid controlled by the input GPA Y |
Function export |
FUNKTIONSKLEMPNER_EXPORT_FUNCTION
This function will generate a wrapper for the following function and export it to the Python module.
FUNKTIONSKLEMPNER_EXPORT_STRUCT
This function will generate a ctypes.Structure wrapping the following struct and export it to the Python module. Functions accepting pointers to this struct (including as return values!) will then be able to pass this data type to and from the Python world. This can be helpful when complex data types shall be transferred with a limited amount of ctypes calls.
FUNKTIONSKLEMPNER_SIZE_CONTROL("X", "Y")
A way to return NumPy arrays with known size. Works with and is required for funktionsklempner::OutputNDArray and funktionsklempner::OutputQuantityArray-type variables. Declares that the variable X is an array of size given by the variable Y. On the Python side, the two C++ parameters X and Y will be translated into a single NumPy array X of shape Y.shape (if Y is an input array) or (Y,) (if Y is not an input parameter of the C++ function, in which case it will be generated as an integer input on the Python side). This allows handling the frequent task of performing computational tasks on vector data kept in Numpy arrays.
The simplest way to automatically derive the size of an output array is by following the size of an input array:
FUNKTIONSKLEMPNER_EXPORT_FUNCTION
FUNKTIONSKLEMPNER_SIZE_CONTROL("out", "in_")
void a_function(
funktionsklempner::InputNDArray<double> in_,
funktionsklempner::OutputNDArray<double> out
);
Here, in_ is an automatically converted input array, and the return value out of the Python wrapper to this function will be a newly generated Numpy array of size out.size = in_.size.
Multidimensional size control
Sometimes, following the size of a single array will not be enough. Think of a two-parameter function evaluated on a meshgrid, for instance. For this particular use case, one can use the size multiplication syntax:
FUNKTIONSKLEMPNER_EXPORT_FUNCTION
FUNKTIONSKLEMPNER_SIZE_CONTROL("X", "Y,Z")
void a_function(
funktionsklempner::InputNDArray<double> Y,
funktionsklempner::InputQuantityArray<boost::units::si::length, double> Z,
funktionsklempner::OutputNDArray<double> out
);
Here, Y is an automatically converted input array of shape Y.shape, Z is an automatically converted input array of shape Z.shape and length dimension, and the return value X of the Python wrapper to this function will be a newly generated Numpy array of shape X.shape = (*Y.shape, *Z.shape).
The size control works equivalently with the QuantityArray, EuclideanPointArray, and GeographicPointArray classes instead of NDArrays.
FUNKTIONSKLEMPNER_ELLIPSOID_CONTROL("X", "Y")
A way to return geographic point arrays with known ellipsoid. Works with funktionsklempner::OutputGeographicPointArray-type variables. Declares that the variable X is a geographic point array with the same ellipsoid as the geographic point array variable Y. On the Python side, the two C++ parameters X and Y will be translated into a single GeographicPointArray X with ellipsoid Y.ellipsoid (if Y is an input geographic point array array) or Y (if Y is not an input parameter of the C++ function, in which case it will be generated as an ellipsoid input on the Python side).
The simplest way to automatically derive the ellipsoid of an output geographic point array is by following the ellipsoid of an input array:
FUNKTIONSKLEMPNER_EXPORT_FUNCTION
FUNKTIONSKLEMPNER_SIZE_CONTROL("out", "in_")
FUNKTIONSKLEMPNER_ELLIPSOID_CONTROL("out", "in_")
int another_function(
funktionsklempner::InputGeographicPointArray<double, fk::geo::UserEllipsoid, false> in_,
funktionsklempner::OutputGeographicPointArray<double, fk::geo::UserEllipsoid, true> out
);
Here, in_ is an automatically converted input geographic point array, and the return value out of the Python wrapper to this function will be a newly generated geographic point array of equal size and ellipsoid.
Specifying ellipsoids in Python
In the Python side of the generated code, ellipsoids can be specified in two ways:
- As a string literal to one of the known ellipsoids:
"WGS84","Bessel 1841"(an their EPSG counterparts"EPSG:7030"and"EPSG:7004"). This corresponds to the C++ static ellipsoid classesfunktionsklempner::geo::ellipsoids::WGS84andfunktionsklempner::geo::ellipsoids::Bessel1841. - As a tuple
((a, a_unit), f), whereais the large half axis given as a in unita_unit, and wherefis the ellipsoid's flattening. This corresponds to the C++funktionsklempner::geo::UserEllipsoidclass.
If the funktionsklempner::geo::UserEllipsoid class was used on the C++ side, the string literals can also be used on the Python side; they will be automatically translated into an equivalent UserEllipsoid.
funktionsklempner::NDArray<NumPyDType T>
This class is a thin random access range wrapper around a scalar pointer (e.g., a NumPy array buffer). It is accessible in the funktionsklempner namespace as InputNDArray<T> (=NDArray<const T>, read-only access to the underlying buffer) and OutputNDArray<T> (=NDArray<T>, can be written to).
The API of the class is as follows:
template<NumPyDType T>
class NDArray
{
public:
/* Constructors: */
constexpr NDArray(T* data, size_t N, const char* dtype);
constexpr NDArray(NDArray&&) noexcept = default;
constexpr NDArray(const NDArray&) noexcept = default;
/* Sized range: */
constexpr size_t size() const noexcept;
/* Random access iterators, pointers to the underlying buffer: */
constexpr T* begin() const noexcept;
constexpr T* end() const noexcept;
/* Random index-based access: */
constexpr T& operator[](size_t i) const;
};
Note that NumPyDType is a concept that captures the basic scalar types of NumPy array buffers (integer and floating point types).
funktionsklempner::InputQuantityArray<Unit U, NumPyDType T>
A rather thin random access wrapper around a scalar pointer that represents values of a certain physical (SI) unit via Boost.Units. Iterating over the quantity array yields value views that act, basically, as physical quantities of the given unit U (boost::unit::quantity<U, T>). It is accessible in the funktionsklempner namespace as InputQuantityArray<U, T> (=QuantityArray<U, const T>, read-only access to the underlying buffer) and OutputQuantityArray<U, T> (=QuantityArray<U, T>, can be written to).
The value type of the iterators of QuantityArray is the the QuantityView<U,T> class, which internally contains a pointer to the numerical buffer of the QuantityArray and a boost::unit::quantity<U, T> that represents the physical unit. The QuantityView can be casted to boost::unit::quantity<U, T>, and addition, subtraction, multiplication, and division operators of QuantityView<U,T> with boost::unit::quantity<U, T> are provided. In case of the OutputQuantityArray, assignment and inplace addition and subtraction operators with boost::unit::quantity<U, T> are provided as well. Essentially, QuantityArray<U,T> can be used to wrap a floating point buffer T* in a way that looks like a buffer of quantities, boost::unit::quantity<U, T>*.
The API of the class is as follows:
template<Unit U, NumPyDType T>
class QuantityArray
{
public:
/* Constructor with given unit of same dimension as `U`. */
constexpr QuantityArray(
T* data,
size_t N,
const char* dtype,
const char* unit
);
/* Constructor with default unit `U` */
constexpr QuantityArray(
T* data,
size_t N,
const char* dtype
);
/* Sized range: */
constexpr size_t size() const noexcept;
/* Random access iterators, pointing to the underlying buffer: */
constexpr QuantityIterator<U, T> begin() const;
constexpr const std::decay_t<T>* end() const noexcept;
/* Random index-based access: */
constexpr QuantityView<U, T> operator[](size_t i) const;
/* Setting the unit to `u`, a quantity of unit `U`.
* Note: this method is enabled only if `T` is not a const value
* (i.e. for OutputQuantityArray<U,T>).
*/
constexpr void set_unit(const boost::units::quantity<U>& u);
};
Note that Unit is a concept that captures boost::units::unit<Dim, System> of some dimension Dim and system System (which essentially has to be the SI system from boost::units::si to be able to interact with the remainder of Funktionsklempner). As before, NumPyDType is a concept that captures the basic scalar types of NumPy array buffers (integer and floating point types).
EuclideanPointArray
A random access input wrapper around a scalar buffer that represents vector values of dimension ndim and physical (SI) length unit via Boost.Units. Iterating over the Euclidean point array yields funktionsklempner::geo::EuclideanPoint values: ndim-dimensional points that act as ndim-sized tuples of quantities of boost::units::si::length (boost::unit::quantity<boost::units::si::length, T>). It is accessible in the funktionsklempner namespace as
funktionsklempner::InputEuclideanPointArray<size_t ndim, NonConstNumPyDType T>(read-only access to the underlying buffer) andfunktionsklempner::OutputEuclideanPointArray<size_t ndim, NonConstNumPyDType T>(write acess to the underlying buffer via the proxy reference classes).
Each of the classes is a std::ranges::random_access_range over values of Euclidean points. The actual type of the iterator values of the Euclidean point ranges is rather complicated template programming (due to the abstraction of points, coordinate units, the underlying storage class Buf, and the proxy referencing that adds units to the floating point buffer), but they all share the same functionality:
- coordinate access via the templated
get<size_t i>()method (withx(),y(), andz()conditionally enabled in dimensions 1--3), - distance computation via the
squared_distance(const EuclidenPoint&)anddistance(const EuclideanPoint&)methods, - assignment from
EuclideanPointvalues (ifBufis mutable), and - individual coordinate assignment (if
Bufis mutable).
If R is to denote an Euclidean point array of dimension ndim and floating point type T, and EP is to denote a compatible Euclidean point specialization (i.e., compatible in dimension), then the API of this class can be roughly summarized as follows:
void Euclidean_functionality(R r, const EP& p)
{
/* Coordinates are quantities of length dimension: */
using Coordinate = boost::units::quantity<boost::units::si::length, T>;
/* Loop over all points of the range: */
for (auto&& v : r){
/* Access coordinate of index i < ndim: */
Coordinate c = v.get<0>();
/* Compute distance.
* d will be the Euclidean distance of length unit L.
* d2 will skip sqrt computation and be of unit L^2
*/
auto d2 = v.squared_distance(p);
auto d = v.distance(p);
/* If 1 <= ndim <= 3 */
Coordinate x = v.x();
/* If 2 <= ndim <= 3 */
Coordinate y = v.y();
/* If 3 == ndim */
Coordinate z = v.z();
/* If the storage class `Buf` is mutable: */
v = p;
v.get<0>() = p.get<0>();
}
}
GeographicPointArray
A random access input wrapper around a scalar buffer that interprets the buffer as multiplexed coordinates of geographic points (with or without height). Iterating over the geographic point array yields funktionsklempner::geo::GeographicPoint values: 2--3-dimensional points that act as 2--3-sized tuples of quantities of either (A,A) or (A,A,L) units (where A is a boost::units::si::plane_angle and L is a boost::units::si::length unit). The geographic point array is accessible in the funktionsklempner namespace (here: fk) as
fk::InputGeographicPointArray<NonConstNumPyDType T, Ellipsoid_t E, bool _height>(read-only access to the underlying buffer) andfk::OutputGeographicPointArray<NonConstNumPyDType T, Ellipsoid_t E, bool _height>(write acess to the underlying buffer via the proxy reference classes).
The Ellipsoid_t type must be one of the predefined static ellipsoids of the fk::geo::ellipsoids namespace (ellipsoid.hpp) or the fk::geo::UserEllipsoid type (which supports runtime ellipsoid specification).
Each of the classes is a std::ranges::random_access_range over values of geographic points. As in the Euclidean case, he actual type of the iterator values of the geographic point ranges is rather complicated but they all share the same functionality:
- coordinate access via the templated
get<size_t i>()method (withlat()andlon()always, andheight()conditionally enabled), - distance computation to compatible geographic point objects via the
squared_distance(const GeographicPoint&)anddistance(const GeographicPoint&)methods, - assignment from compatible
GeographicPointvalues (if the underlying buffer is mutable), and - individual coordinate assignment (if the underlying buffer is mutable).
If R is to denote an geographic point array of ellipsoid E, height switch h and floating point type T, and GP is to denote a compatible geographic point specialization (i.e., compatible in ellipsoid and height switch), then the API of this class can be roughly summarized as follows:
void Geographic_functionality(R r, const GP& p, const E& ellps)
{
/* Coordinates are quantities of length dimension: */
using Length = boost::units::quantity<boost::units::si::length, T>;
using Angle = boost::units::quantity<boost::units::si::plane_angle, T>;
/* Loop over all points of the range: */
for (auto&& v : r){
/* Access coordinates: */
Angle lat = v.get<0>();
Angle lon = v.get<1>();
/* Equivalently: */
lat = v.lat();
lon = v.lon();
/* If height switch has been set: */
if constexpr (h){
Length height = v.get<2>();
height = v.height();
}
/* Compute geodetic distance. This uses (and requires) GeographicLib
* under the hood. d will be the Euclidean distance of length unit L.
* This does not work if the coordinates have height:
*/
if constexpr (!h){
auto d = v.distance(p);
}
/* If the storage class `Buf` is mutable: */
v = p;
v.get<0>() = p.get<0>();
/* Conversion from geographic coordinates to Euclidean coordinates
* in an earth-centered earth-fixed coordinate system, and vice versa: */
funktionsklempner::geo::Point3D xyz = v.to_ECEF();
v = GP::from_ECEF(xyz, ellps);
}
}
Caveats
Some known caveats to keep in mind when using Funktionsklempner:
- This uses a simplified grammar.
- There is some effort in this code base to handle comments. However, this effort is designed empirically rather than after a standard. You might run into complications with more 'unusual' comment or preprocessor layouts. If you hit such a bug, you may try to file a bug report--but there is no guarantee that all such bugs will be fixed (in which case reduction to the simplified grammar may be the only way to run Funktionsklempner).
Note, furthermore, the terms listed in the LICENSE.
License
Funktionsklempner itself is licensed under the European Public License, v1.2 (EUPL-1.2, see LICENSE file in this directory). The template code for the code generation is licensed under the BSD 3-Clause license (BSD-3-Clause, noted in corresponding source files). The example under the example/ subfolder is equally licensed under the BSD-3-Clause license.
Funktionsklempner vendors a subset of the Boost C++ Libraries, version 1.90.0 in subprojects/packagefiles/boost-1.90.0_units.tar.gz. This code is licensed under the Boost Software License as indicated in the archive. You can find the full Boost library at boost.org.
Changelog
The format is based on Keep a Changelog, and this project adheres to Semantic Versioning.
[1.1.0] - 2026-04-28
Added
- Add geographic point ranges (
funktionsklempner::InputEuclideanPointRangeandfunktionsklempner::OutputEuclideanPointRange) that interpret a sized floating point buffer in combination with a reference ellipsoid as a range of Geographic points (both with and without height). If the buffer is mutable (i.e., Output), assignment from compatibleGeographicPointobjects can be done. Based on the newVectorQuantityViewclass combined with theGeographicPointfunctionality. - Add Euclidean point ranges (
funktionsklempner::InputEuclideanPointRangeandfunktionsklempner::OutputEuclideanPointRange) that interpret a sized floating point buffer as a range of n-dimensional Euclidean points. If the buffer is mutable (i.e., Output), assignment fromEuclideanPointobjects can be done. Based on the newVectorQuantityViewclass combined with theEuclideanPointfunctionality. - Add the
funktionsklempner::geo::VectorQuantityViewclass that can interpret floating point buffers (of n-multiple size) as a sequence of n-dimensional data with axis-dependent units. - Add implementations for points in geographic and n-dimensional Euclidean spaces:
funktionsklempner::geo::GeographicPoint,funktionsklempner::geo::EuclideanPoint<size_t ndim>,funktionsklempner::geo::Point2D, andfunktionsklempner::geo::Point3D. These handle coordinates asboost::units::quantityof the corresponding SI type as well as distance computations - Add GeographicLib as an optional feature dependency. If selected, this enables geodesic distance computations between
funktionsklempner::geo::GeographicPointinstances. - Add iterator classes that interpret ranges over
boost::units::quantityof the corresponding unit (e.g.funktionsklempner::QuantityArray) as multiplexed sequences of points: thefunktionsklempner::geo::EuclideanIteratorand thefunktionsklempner::GeographicIterator. This allows providing point arrays as flattened coordinate arrays of the corresponding unit, and hence enables one to pass point vectors to the C++ side via ctypes.
Changed
- Slight changes to the internal
QuantityViewproxy reference to fulfillstd::indirectly_writableas intended.
[1.0.0] - 2026-03-09
Added
- Start of the changelog.
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file funktionsklempner-1.1.0.tar.gz.
File metadata
- Download URL: funktionsklempner-1.1.0.tar.gz
- Upload date:
- Size: 3.3 MB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.14.4
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
c70ca00d53be2d45f492bedd7537804a90be5873d65d99356f3307ed5fa4f257
|
|
| MD5 |
8fc29c7d4a6456f5769e2f1407fe8161
|
|
| BLAKE2b-256 |
9b38fcf138ab17e3305d154b1f442d1b2eeac4c6d16bd84775af54ba88b9637e
|
File details
Details for the file funktionsklempner-1.1.0-py3-none-any.whl.
File metadata
- Download URL: funktionsklempner-1.1.0-py3-none-any.whl
- Upload date:
- Size: 3.4 MB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.14.4
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
fa51648932b781f7bebc048a69fc2f24adf8c8b1c17f71ce5b44a5ef6555fbf2
|
|
| MD5 |
ba639ec11c4b8dbc22da238b249f8735
|
|
| BLAKE2b-256 |
59bd687a4e4cd1494063a3cc66d122392255d15301b9ddecd3cefbaae3093b4c
|