RoughPy


NameRoughPy JSON
Version 0.1.1 PyPI version JSON
download
home_page
Summary
upload_time2024-01-29 10:50:25
maintainer
docs_urlNone
author
requires_python
licenseBSD-3-Clause
keywords data streams rough-paths signatures
VCS
bugtrack_url
requirements No requirements were recorded.
Travis-CI No Travis.
coveralls test coverage No coveralls.
            <center>
  <img src="https://raw.githubusercontent.com/datasig-ac-uk/RoughPy/main/branding/logo/logo_square_white.jpg">
</center><br>

# RoughPy
RoughPy is a package for working with streaming data as rough paths, and working with algebraic objects such as free tensors, shuffle tensors, and elements of the free Lie algebra.

This library is currently in an alpha stage, and as such many features are still incomplete or not fully implemented. Please bear this in mind when looking at the source code.


## Installation
RoughPy can be installed from PyPI using `pip` on Windows, Linux, and MacOS (Intel based Mac only, sorry not Apple Silicon support yet). Simply run
```
pip install roughpy
```
to get the latest version.

Alternatively, the wheel files can be downloaded from the [Releases](https://github.com/datasig-ac-uk/RoughPy/releases) page.

### Installing from source
RoughPy can be installed from source, although this is not the recommended way to install.
The build system requires [vcpkg](https://github.com/Microsoft/vcpkg) in order to obtain the necessary dependencies (except for MKL on x86 platforms, which is installed via pip).
You will need to make sure that vcpkg is available on your system before attempting to build RoughPy.
The following commands should be sufficient to set up the environment for building RoughPy:
```bash
git clone https://github.com/Microsoft/vcpkg.git tools/vcpkg
tools/vcpkg/bootstrap-vcpkg.sh
export CMAKE_TOOLCHAIN_FILE=$(pwd)/tools/vcpkg/scripts/buildsystems/vcpkg.cmake
```
With this environment variable set, most of the dependencies will be installed automatically during the build process.

On MacOS with Apple Silicon you will need to install libomp (for example using Homebrew `brew install libomp`).
This is not necessary on Intel based MacOS where the Intel iomp5 can be used instead. 
The build system will use `brew --prefix libomp` to try to locate this library.
(The actual `brew` executable can be customised by setting the `ROUGHPY_BREW_EXECUTABLE` CMake variable 
or environment variable.)

You should now be able to pip install either using the PyPI source distribution (using the `--no-binary :roughpy:` 
flag), or directly from GitHub (recommended):
```bash
pip install git+https://github.com/datasig-ac-uk/RoughPy.git
```
It will take some time to build.

## Intervals in RoughPy
RoughPy is very careful in how it works with intervals.
One design goal is that it should be able to handle jumps in the underlying signal that occur at particular times, including the beginning or end of the interval, and still guarantee that if you combine the signature over adjacent interval, you always get the signature over the entire interval.
This implies that there has to be a decision about whether data at the exact beginning or exact end of the interval is included.
The convention in RoughPy are that we use clopen intervals, and that data at beginning of the interval is seen, and data at the end of the interval is seen in the next interval.
A second design goal is that the code should be efficient, and so the internal representation of a stream involves caching the signature over dyadic intervals of different resolutions.
Recovering the signature over any interval using the cache has logarithmic complexity (using at most 2n tensor multiplications, when n is the internal resolution of the stream).
Resolution refers to the length of the finest granularity at which we will store information about the underlying data.
Any event occurs within one of these finest granularity intervals, multiple events occur within the same interval resolve to a more complex log-signature which correctly reflects the time sequence of the events within this grain of time.
However, no query of the stream is allowed to see finer resolution than the internal resolution of the stream, it is only allowed to access the information over intervals that are a union of these finest resolution granular intervals.
For this reason, a query over any interval is replaced by a query is replaced by a query over an interval whose endpoints have been shifted to be consistent with the granular resolution, obtained by rounding these points to the contained end-point of the unique clopen granular interval containing this point.
In particular, if both the left-hand and right-hand ends of the interval are contained in the clopen granular interval, we round the interval to the empty interval. 
Specifying a resolution of 32 or 64 equates to using integer arithmetic.

## Usage
Following the NumPy (and related) convention, we import RoughPy under the alias `rp` as follows:
```python
import roughpy as rp
```
The main object(s) that you will interact with are `Stream` objects or the family of factory classes such as `LieIncrementStream`. For example, we can create a `LieIncrementStream` using the following commands:
```python
import numpy as np
stream = rp.LieIncrementStream.from_increments(np.array([[0, 1, 2], [3, 4, 5]], dtype=np.float64), depth=2)
```
This will create a stream whose (hidden) underlying data are the two increments `[0., 1., 2.]` and `[3., 4., 5.]`, and whose algebra elements are truncated at maximum depth 2.
To compute the log signature over an interval we use the `log_signature` method on the stream, for example
```python
interval = rp.RealInterval(0., 1.)
lsig = stream.log_signature(interval)
```
Printing this new object `lsig` should give the following result
```
{ 1(2) 2(3) }
```
which is the first increment from the underlying data. (By default, the increments are assumed to occur at parameter values equal to their row index in the provided data.)

Similarly, the signature can be computed using the `signature` method on the stream object:
```python
sig = stream.signature(interval)
```
Notice that the `lsig` and `sig` objects have types `Lie` and `FreeTensor`, respectively. They behave exactly as you would expect elements of these algebras to behave. Moreover, they will (usually) be convertible directly to a NumPy array (or TensorFlow, PyTorch, JAX tensor type in the future) of the underlying data, so you can interact with them as if they were simple arrays.

We can also construct streams by providing the raw data of Lie increments with higher order terms by specifying the width using the same constructor above.
For example, if we take width 2 and depth 2 then the elements of a Lie element will have keys (1, 2, [1,2]).
So if we provide the following data, we construct a stream whose underlying Lie increments are width 2, depth 2 
```
stream = rp.LieIncrementStream.from_increments(np.array([[1, 2, 0.5], [0.2, -0.1, 0.2]], dtype=np.float64), width=2, depth=2)
print(stream.log_signature(rp.RealInterval(0, 0.5), 2)) # returns the first increment
# { 1(1), 2(2), 0.5([1,2]) }
>>> print(stream.log_signature(rp.RealInterval(0, 1.5), 2)) # Campbell-Baker-Hausdorff product of both increments
{ 1.2(1) 1.9(2) 0.45([1,2]) }
```


## Support
If you have a specific problem, the best way to record this is to open an issue on GitHub. We welcome any feedback or bug reports.

## Contributing 
In the future, we will welcome pull requests to implement new features, fix bugs, add documentation or examples, or add tests to the project.
However, at present, we do not have robust CI pipelines set up to rigorously test incoming changes, and therefor will not be accepting pull requests made from outside the current team.


## Contributors
The full list of contributors is listed in `THANKS` alongside this readme. The people mentioned in this document constitute `The RoughPy Developers`.

## License
RoughPy is licensed under a BSD-3-Clause license. This was chosen specifically to match the license of NumPy.
# Changelog

Version 0.1.1:
  - Fixed type promotions in scalar arithmetic - left hand types are now promoted when appropriate.
  - Added "tensor_functions" module for implementing additional functions on tensors. Currently only Log is implemented.
  - Fixed a few bugs in the build system


Version 0.1.0:
  - Added framework for integrating device support and redesigned scalars module to accommodate the changes.
  - Made changes to type deduction in constructors to avoid exceptions when providing lists of python ints/floats.
  - Changed the implementation of __array__ for algebra types. A copy is always made, and the size of the array is always
    equal to the dimension of the chosen width/depth composition.
  - Changed the behaviour when no resolution is given. RoughPy now uses a heuristic to compute a suitable resolution for computing signatures if none is given.


Version 0.0.8:
  - Disabled linking to BLAS/LAPACK to reduce compile times whilst under development.
  - Greatly expanded the serialization support internally.
  - Many classes can now be pickled/unpickled for transport across process boundaries.
    (Note that this is not yet a safe means of storing and transmitting stream data.)
  - Overlay triplets are now forced for all builds. This should improve reproducibility.
  - Restructured the algebra module to improve build times.
  - Added CMake detection of headers to help builds with non-compliant compilers.
  - Fixed an error in the construction of `PyLieKey` in `PyLieKeyIterator`. [#40](https://github.com/datasig-ac-uk/RoughPy/issues/40)


Version 0.0.7:
  - Overhaul the (internal) ScalarType API:
    . the overloads of convert_copy have been removed in favour of the variant that takes two ScalarPointers;
    . the single versions of add(_inplace) and friends have been replaced with more flexible add_into;
      batch compute methods and friends;
    . replaced single value uminus with uminus into with similar signature to to add_into and friends;
    . removed single value copy method;
  - Added constructor for ScalarPointer from type_id and pointer.
  - Implementations of ScalarType methods that are essentially the same for all types are implemented
    in a common implementation layer.
  - Added threading support in platform
  - add_into and friends have threading support if available and enabled.
  - Added default implementation of type_id_of so that non-specialized types
  look for a ScalarType object.
  - Greatly simplified the design of ScalarMatrix - it now only supports full,
  dense matrices.
  - Redesigned the interface between the Scalar linear algebra and
  MKL/BLAS+LAPACK.
  - Added function to query ring characteristics of a ScalarType - currently
  unused.
  - Added KeyScalarStream for constructing streams from array-like data more easily.
  - Implemented the `from_type_details` function for scalar types. This fixes a bug when constructing objects using the dlpack protocol.
  - Overhaul constructor for `LieIncrementStream`s from increment data to reduce
   number of copies (if possible) and to handle non-contiguous or oddly shaped
   data correctly.
  - Change implementation of `LieIncrementStream` to allow adding the parameter
  channel during construction.
  - Change implementation of `TickStream` to allow adding parameter channel
  during construction.




Version 0.0.6:
  - Externally sourced streams (sound-file streams) now support setting channel
  types/schema in factory function.
  - Added option for customising file name macro in exception throws.
  - Made some improvements to the internal interface of ScalarType to allow more
   efficient implementations of vectorised operations.
  - Added fast "is_zero" method to Python algebra objects.

Version 0.0.5:
  - Added free functions for performing free-tensor, shuffle, half-shuffle
  multiplication between pairs of tensors (of either kind).
  - Added free function for applying the adjoint of left free tensor
  multiplication to arbitrary tensors.
  - Improved exception usage, messages now include filename, lineno, and
  function name to help locate c++ exceptions passed through to Python.
  - Basis objects in Python are now iterable.
  - Added `split_n` and `to_index` methods to Tensor key.

Version 0.0.4:
  - Overhauled the RPY_CHECK macro so it now gives much better contextual
  information.
  - Readme updated to reflect PyPI installation with wheels.
  - Antipode is now implemented in libalgebra_lite, and exposed to Python for
  Free tensors.
  - Streams now carry a support interval, outside of which the signature will be
   return trivial.
  - Build requirements fixed for non x86 platforms.
  - Expanded coverage of pytype stub file.

Version 0.0.3:
  - Added datetime interval support.
  - Integrated schema reparametrisation into stream signature methods.
  - Fixed bug in names of scalar types, names now display correctly.
  - Fixed bfloat16 given wrong scalar type code.
  - Added half precision float and bfloat16 to py module
  - Added real partition class and Python interface.
  - Implemented simplify method on Streams.
  - Added polynomial coefficients
  - started an examples folder

Version 0.0.2-alpha:
  - Added datetime and timedelta object support to tick data parsing.
  - Expanded build system to include more versions of Python.
  - Stabilised compilation on all three platforms.
  - Fixed numerous bugs in the build system.

Version 0.0.1-alpha:
  - First alpha release.

            

Raw data

            {
    "_id": null,
    "home_page": "",
    "name": "RoughPy",
    "maintainer": "",
    "docs_url": null,
    "requires_python": "",
    "maintainer_email": "",
    "keywords": "data streams rough-paths signatures",
    "author": "",
    "author_email": "",
    "download_url": "https://files.pythonhosted.org/packages/c7/b1/69750218722e7d93084172c5393abe68b90350ffa39b8b47ae9647aeea08/roughpy-0.1.1.tar.gz",
    "platform": null,
    "description": "<center>\n  <img src=\"https://raw.githubusercontent.com/datasig-ac-uk/RoughPy/main/branding/logo/logo_square_white.jpg\">\n</center><br>\n\n# RoughPy\nRoughPy is a package for working with streaming data as rough paths, and working with algebraic objects such as free tensors, shuffle tensors, and elements of the free Lie algebra.\n\nThis library is currently in an alpha stage, and as such many features are still incomplete or not fully implemented. Please bear this in mind when looking at the source code.\n\n\n## Installation\nRoughPy can be installed from PyPI using `pip` on Windows, Linux, and MacOS (Intel based Mac only, sorry not Apple Silicon support yet). Simply run\n```\npip install roughpy\n```\nto get the latest version.\n\nAlternatively, the wheel files can be downloaded from the [Releases](https://github.com/datasig-ac-uk/RoughPy/releases) page.\n\n### Installing from source\nRoughPy can be installed from source, although this is not the recommended way to install.\nThe build system requires [vcpkg](https://github.com/Microsoft/vcpkg) in order to obtain the necessary dependencies (except for MKL on x86 platforms, which is installed via pip).\nYou will need to make sure that vcpkg is available on your system before attempting to build RoughPy.\nThe following commands should be sufficient to set up the environment for building RoughPy:\n```bash\ngit clone https://github.com/Microsoft/vcpkg.git tools/vcpkg\ntools/vcpkg/bootstrap-vcpkg.sh\nexport CMAKE_TOOLCHAIN_FILE=$(pwd)/tools/vcpkg/scripts/buildsystems/vcpkg.cmake\n```\nWith this environment variable set, most of the dependencies will be installed automatically during the build process.\n\nOn MacOS with Apple Silicon you will need to install libomp (for example using Homebrew `brew install libomp`).\nThis is not necessary on Intel based MacOS where the Intel iomp5 can be used instead. \nThe build system will use `brew --prefix libomp` to try to locate this library.\n(The actual `brew` executable can be customised by setting the `ROUGHPY_BREW_EXECUTABLE` CMake variable \nor environment variable.)\n\nYou should now be able to pip install either using the PyPI source distribution (using the `--no-binary :roughpy:` \nflag), or directly from GitHub (recommended):\n```bash\npip install git+https://github.com/datasig-ac-uk/RoughPy.git\n```\nIt will take some time to build.\n\n## Intervals in RoughPy\nRoughPy is very careful in how it works with intervals.\nOne design goal is that it should be able to handle jumps in the underlying signal that occur at particular times, including the beginning or end of the interval, and still guarantee that if you combine the signature over adjacent interval, you always get the signature over the entire interval.\nThis implies that there has to be a decision about whether data at the exact beginning or exact end of the interval is included.\nThe convention in RoughPy are that we use clopen intervals, and that data at beginning of the interval is seen, and data at the end of the interval is seen in the next interval.\nA second design goal is that the code should be efficient, and so the internal representation of a stream involves caching the signature over dyadic intervals of different resolutions.\nRecovering the signature over any interval using the cache has logarithmic complexity (using at most 2n tensor multiplications, when n is the internal resolution of the stream).\nResolution refers to the length of the finest granularity at which we will store information about the underlying data.\nAny event occurs within one of these finest granularity intervals, multiple events occur within the same interval resolve to a more complex log-signature which correctly reflects the time sequence of the events within this grain of time.\nHowever, no query of the stream is allowed to see finer resolution than the internal resolution of the stream, it is only allowed to access the information over intervals that are a union of these finest resolution granular intervals.\nFor this reason, a query over any interval is replaced by a query is replaced by a query over an interval whose endpoints have been shifted to be consistent with the granular resolution, obtained by rounding these points to the contained end-point of the unique clopen granular interval containing this point.\nIn particular, if both the left-hand and right-hand ends of the interval are contained in the clopen granular interval, we round the interval to the empty interval. \nSpecifying a resolution of 32 or 64 equates to using integer arithmetic.\n\n## Usage\nFollowing the NumPy (and related) convention, we import RoughPy under the alias `rp` as follows:\n```python\nimport roughpy as rp\n```\nThe main object(s) that you will interact with are `Stream` objects or the family of factory classes such as `LieIncrementStream`. For example, we can create a `LieIncrementStream` using the following commands:\n```python\nimport numpy as np\nstream = rp.LieIncrementStream.from_increments(np.array([[0, 1, 2], [3, 4, 5]], dtype=np.float64), depth=2)\n```\nThis will create a stream whose (hidden) underlying data are the two increments `[0., 1., 2.]` and `[3., 4., 5.]`, and whose algebra elements are truncated at maximum depth 2.\nTo compute the log signature over an interval we use the `log_signature` method on the stream, for example\n```python\ninterval = rp.RealInterval(0., 1.)\nlsig = stream.log_signature(interval)\n```\nPrinting this new object `lsig` should give the following result\n```\n{ 1(2) 2(3) }\n```\nwhich is the first increment from the underlying data. (By default, the increments are assumed to occur at parameter values equal to their row index in the provided data.)\n\nSimilarly, the signature can be computed using the `signature` method on the stream object:\n```python\nsig = stream.signature(interval)\n```\nNotice that the `lsig` and `sig` objects have types `Lie` and `FreeTensor`, respectively. They behave exactly as you would expect elements of these algebras to behave. Moreover, they will (usually) be convertible directly to a NumPy array (or TensorFlow, PyTorch, JAX tensor type in the future) of the underlying data, so you can interact with them as if they were simple arrays.\n\nWe can also construct streams by providing the raw data of Lie increments with higher order terms by specifying the width using the same constructor above.\nFor example, if we take width 2 and depth 2 then the elements of a Lie element will have keys (1, 2, [1,2]).\nSo if we provide the following data, we construct a stream whose underlying Lie increments are width 2, depth 2 \n```\nstream = rp.LieIncrementStream.from_increments(np.array([[1, 2, 0.5], [0.2, -0.1, 0.2]], dtype=np.float64), width=2, depth=2)\nprint(stream.log_signature(rp.RealInterval(0, 0.5), 2)) # returns the first increment\n# { 1(1), 2(2), 0.5([1,2]) }\n>>> print(stream.log_signature(rp.RealInterval(0, 1.5), 2)) # Campbell-Baker-Hausdorff product of both increments\n{ 1.2(1) 1.9(2) 0.45([1,2]) }\n```\n\n\n## Support\nIf you have a specific problem, the best way to record this is to open an issue on GitHub. We welcome any feedback or bug reports.\n\n## Contributing \nIn the future, we will welcome pull requests to implement new features, fix bugs, add documentation or examples, or add tests to the project.\nHowever, at present, we do not have robust CI pipelines set up to rigorously test incoming changes, and therefor will not be accepting pull requests made from outside the current team.\n\n\n## Contributors\nThe full list of contributors is listed in `THANKS` alongside this readme. The people mentioned in this document constitute `The RoughPy Developers`.\n\n## License\nRoughPy is licensed under a BSD-3-Clause license. This was chosen specifically to match the license of NumPy.\n# Changelog\n\nVersion 0.1.1:\n  - Fixed type promotions in scalar arithmetic - left hand types are now promoted when appropriate.\n  - Added \"tensor_functions\" module for implementing additional functions on tensors. Currently only Log is implemented.\n  - Fixed a few bugs in the build system\n\n\nVersion 0.1.0:\n  - Added framework for integrating device support and redesigned scalars module to accommodate the changes.\n  - Made changes to type deduction in constructors to avoid exceptions when providing lists of python ints/floats.\n  - Changed the implementation of __array__ for algebra types. A copy is always made, and the size of the array is always\n    equal to the dimension of the chosen width/depth composition.\n  - Changed the behaviour when no resolution is given. RoughPy now uses a heuristic to compute a suitable resolution for computing signatures if none is given.\n\n\nVersion 0.0.8:\n  - Disabled linking to BLAS/LAPACK to reduce compile times whilst under development.\n  - Greatly expanded the serialization support internally.\n  - Many classes can now be pickled/unpickled for transport across process boundaries.\n    (Note that this is not yet a safe means of storing and transmitting stream data.)\n  - Overlay triplets are now forced for all builds. This should improve reproducibility.\n  - Restructured the algebra module to improve build times.\n  - Added CMake detection of headers to help builds with non-compliant compilers.\n  - Fixed an error in the construction of `PyLieKey` in `PyLieKeyIterator`. [#40](https://github.com/datasig-ac-uk/RoughPy/issues/40)\n\n\nVersion 0.0.7:\n  - Overhaul the (internal) ScalarType API:\n    . the overloads of convert_copy have been removed in favour of the variant that takes two ScalarPointers;\n    . the single versions of add(_inplace) and friends have been replaced with more flexible add_into;\n      batch compute methods and friends;\n    . replaced single value uminus with uminus into with similar signature to to add_into and friends;\n    . removed single value copy method;\n  - Added constructor for ScalarPointer from type_id and pointer.\n  - Implementations of ScalarType methods that are essentially the same for all types are implemented\n    in a common implementation layer.\n  - Added threading support in platform\n  - add_into and friends have threading support if available and enabled.\n  - Added default implementation of type_id_of so that non-specialized types\n  look for a ScalarType object.\n  - Greatly simplified the design of ScalarMatrix - it now only supports full,\n  dense matrices.\n  - Redesigned the interface between the Scalar linear algebra and\n  MKL/BLAS+LAPACK.\n  - Added function to query ring characteristics of a ScalarType - currently\n  unused.\n  - Added KeyScalarStream for constructing streams from array-like data more easily.\n  - Implemented the `from_type_details` function for scalar types. This fixes a bug when constructing objects using the dlpack protocol.\n  - Overhaul constructor for `LieIncrementStream`s from increment data to reduce\n   number of copies (if possible) and to handle non-contiguous or oddly shaped\n   data correctly.\n  - Change implementation of `LieIncrementStream` to allow adding the parameter\n  channel during construction.\n  - Change implementation of `TickStream` to allow adding parameter channel\n  during construction.\n\n\n\n\nVersion 0.0.6:\n  - Externally sourced streams (sound-file streams) now support setting channel\n  types/schema in factory function.\n  - Added option for customising file name macro in exception throws.\n  - Made some improvements to the internal interface of ScalarType to allow more\n   efficient implementations of vectorised operations.\n  - Added fast \"is_zero\" method to Python algebra objects.\n\nVersion 0.0.5:\n  - Added free functions for performing free-tensor, shuffle, half-shuffle\n  multiplication between pairs of tensors (of either kind).\n  - Added free function for applying the adjoint of left free tensor\n  multiplication to arbitrary tensors.\n  - Improved exception usage, messages now include filename, lineno, and\n  function name to help locate c++ exceptions passed through to Python.\n  - Basis objects in Python are now iterable.\n  - Added `split_n` and `to_index` methods to Tensor key.\n\nVersion 0.0.4:\n  - Overhauled the RPY_CHECK macro so it now gives much better contextual\n  information.\n  - Readme updated to reflect PyPI installation with wheels.\n  - Antipode is now implemented in libalgebra_lite, and exposed to Python for\n  Free tensors.\n  - Streams now carry a support interval, outside of which the signature will be\n   return trivial.\n  - Build requirements fixed for non x86 platforms.\n  - Expanded coverage of pytype stub file.\n\nVersion 0.0.3:\n  - Added datetime interval support.\n  - Integrated schema reparametrisation into stream signature methods.\n  - Fixed bug in names of scalar types, names now display correctly.\n  - Fixed bfloat16 given wrong scalar type code.\n  - Added half precision float and bfloat16 to py module\n  - Added real partition class and Python interface.\n  - Implemented simplify method on Streams.\n  - Added polynomial coefficients\n  - started an examples folder\n\nVersion 0.0.2-alpha:\n  - Added datetime and timedelta object support to tick data parsing.\n  - Expanded build system to include more versions of Python.\n  - Stabilised compilation on all three platforms.\n  - Fixed numerous bugs in the build system.\n\nVersion 0.0.1-alpha:\n  - First alpha release.\n",
    "bugtrack_url": null,
    "license": "BSD-3-Clause",
    "summary": "",
    "version": "0.1.1",
    "project_urls": null,
    "split_keywords": [
        "data",
        "streams",
        "rough-paths",
        "signatures"
    ],
    "urls": [
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "1c21d88ce9ac2c720fd5b73ac1db1de4b2c1eeea8949ca46ad81ad0066bb485d",
                "md5": "85bd22460ef4521c68df3ccc02c7a607",
                "sha256": "b3c4d8cd56de8ab1b76ceb772c34b931830048d6437fbe735071d95be542e5eb"
            },
            "downloads": -1,
            "filename": "roughpy-0.1.1-cp310-cp310-macosx_11_0_x86_64.whl",
            "has_sig": false,
            "md5_digest": "85bd22460ef4521c68df3ccc02c7a607",
            "packagetype": "bdist_wheel",
            "python_version": "cp310",
            "requires_python": null,
            "size": 6333676,
            "upload_time": "2024-01-29T10:49:49",
            "upload_time_iso_8601": "2024-01-29T10:49:49.491610Z",
            "url": "https://files.pythonhosted.org/packages/1c/21/d88ce9ac2c720fd5b73ac1db1de4b2c1eeea8949ca46ad81ad0066bb485d/roughpy-0.1.1-cp310-cp310-macosx_11_0_x86_64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "8b07ff89b4c8aeb9ea2920993dd9b2e98a1c40b2f864c9401a1e1b644bcca84f",
                "md5": "40bb76b78fe8c040b182893e6539f4c3",
                "sha256": "5adacadbd69879c65abc907a0d013ff65e2301d422d7864c5d1a13cb55a5fde9"
            },
            "downloads": -1,
            "filename": "roughpy-0.1.1-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl",
            "has_sig": false,
            "md5_digest": "40bb76b78fe8c040b182893e6539f4c3",
            "packagetype": "bdist_wheel",
            "python_version": "cp310",
            "requires_python": null,
            "size": 7882233,
            "upload_time": "2024-01-29T10:49:52",
            "upload_time_iso_8601": "2024-01-29T10:49:52.889761Z",
            "url": "https://files.pythonhosted.org/packages/8b/07/ff89b4c8aeb9ea2920993dd9b2e98a1c40b2f864c9401a1e1b644bcca84f/roughpy-0.1.1-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "64d35371b1c4e6550557ae0067af20932ab3241d6ec981d8dcec842291598381",
                "md5": "1bb75618420f626aad1e8c4aa58a8e8d",
                "sha256": "b0925962cb055dad2979dadf80fd7110e486ac75b2aa5f6b930f1e2b0292b1e5"
            },
            "downloads": -1,
            "filename": "roughpy-0.1.1-cp310-cp310-win_amd64.whl",
            "has_sig": false,
            "md5_digest": "1bb75618420f626aad1e8c4aa58a8e8d",
            "packagetype": "bdist_wheel",
            "python_version": "cp310",
            "requires_python": null,
            "size": 2929928,
            "upload_time": "2024-01-29T10:49:55",
            "upload_time_iso_8601": "2024-01-29T10:49:55.702254Z",
            "url": "https://files.pythonhosted.org/packages/64/d3/5371b1c4e6550557ae0067af20932ab3241d6ec981d8dcec842291598381/roughpy-0.1.1-cp310-cp310-win_amd64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "2e107934bb695517fecefa29e9cb9fe217b1b7dfeb0f1c6aa23fad21aa198f26",
                "md5": "7833aab52d729b519e8abc9f95e8147f",
                "sha256": "27033e71bcbd5bde4e39bc1bf5d7baa69ca19e45a8ba4b7ed4872536d93c7148"
            },
            "downloads": -1,
            "filename": "roughpy-0.1.1-cp311-cp311-macosx_11_0_x86_64.whl",
            "has_sig": false,
            "md5_digest": "7833aab52d729b519e8abc9f95e8147f",
            "packagetype": "bdist_wheel",
            "python_version": "cp311",
            "requires_python": null,
            "size": 6335176,
            "upload_time": "2024-01-29T10:49:58",
            "upload_time_iso_8601": "2024-01-29T10:49:58.534749Z",
            "url": "https://files.pythonhosted.org/packages/2e/10/7934bb695517fecefa29e9cb9fe217b1b7dfeb0f1c6aa23fad21aa198f26/roughpy-0.1.1-cp311-cp311-macosx_11_0_x86_64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "6571ef5f1a30adbd4b603e2ee4ef4d2e1e0bb05004ac8769a32c7d591ce34395",
                "md5": "5e4b1490f9abc4d2e1bfb7bac523b955",
                "sha256": "4c3c246a0afe0878fe0ead85faf577728b050ed5a03a545e33d5ec4d4fd1eae1"
            },
            "downloads": -1,
            "filename": "roughpy-0.1.1-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl",
            "has_sig": false,
            "md5_digest": "5e4b1490f9abc4d2e1bfb7bac523b955",
            "packagetype": "bdist_wheel",
            "python_version": "cp311",
            "requires_python": null,
            "size": 7887029,
            "upload_time": "2024-01-29T10:50:02",
            "upload_time_iso_8601": "2024-01-29T10:50:02.099474Z",
            "url": "https://files.pythonhosted.org/packages/65/71/ef5f1a30adbd4b603e2ee4ef4d2e1e0bb05004ac8769a32c7d591ce34395/roughpy-0.1.1-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "3bd9d64971909ac5ab8c02b9c88e76f016db5e7e21fd5098d6efd308d683ce37",
                "md5": "8d7f8f5b990b8793725a8b6690e54971",
                "sha256": "2cde120fba6f611e962d84e1af960d2554b993c0a3f3e31776285e9af6f3837a"
            },
            "downloads": -1,
            "filename": "roughpy-0.1.1-cp311-cp311-win_amd64.whl",
            "has_sig": false,
            "md5_digest": "8d7f8f5b990b8793725a8b6690e54971",
            "packagetype": "bdist_wheel",
            "python_version": "cp311",
            "requires_python": null,
            "size": 2930551,
            "upload_time": "2024-01-29T10:50:04",
            "upload_time_iso_8601": "2024-01-29T10:50:04.432115Z",
            "url": "https://files.pythonhosted.org/packages/3b/d9/d64971909ac5ab8c02b9c88e76f016db5e7e21fd5098d6efd308d683ce37/roughpy-0.1.1-cp311-cp311-win_amd64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "c709d13a249b45d46e684c5d99bef7bc41ce60046c88d327c7e9886c890ac21c",
                "md5": "90b2d4a0dbf8925e270ce76b9a89e6d7",
                "sha256": "7cbce8db9c6ae0bdaccd0ca47018e635df7242b1426eabd74801f327e0a1d7d1"
            },
            "downloads": -1,
            "filename": "roughpy-0.1.1-cp312-cp312-macosx_11_0_x86_64.whl",
            "has_sig": false,
            "md5_digest": "90b2d4a0dbf8925e270ce76b9a89e6d7",
            "packagetype": "bdist_wheel",
            "python_version": "cp312",
            "requires_python": null,
            "size": 6345501,
            "upload_time": "2024-01-29T10:50:06",
            "upload_time_iso_8601": "2024-01-29T10:50:06.986892Z",
            "url": "https://files.pythonhosted.org/packages/c7/09/d13a249b45d46e684c5d99bef7bc41ce60046c88d327c7e9886c890ac21c/roughpy-0.1.1-cp312-cp312-macosx_11_0_x86_64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "fe219a0a7896ce1ce07e83cfad83ca8f5843ce0288cc68d0e6d1d72cd5ca9db8",
                "md5": "ce51ca4e84e9611771313105254b9ca1",
                "sha256": "c4cec4e49f1f9fabbd757cdbff44c0e996390dd28d65e3fbfe2bb720952cc0fd"
            },
            "downloads": -1,
            "filename": "roughpy-0.1.1-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl",
            "has_sig": false,
            "md5_digest": "ce51ca4e84e9611771313105254b9ca1",
            "packagetype": "bdist_wheel",
            "python_version": "cp312",
            "requires_python": null,
            "size": 7874368,
            "upload_time": "2024-01-29T10:50:10",
            "upload_time_iso_8601": "2024-01-29T10:50:10.655493Z",
            "url": "https://files.pythonhosted.org/packages/fe/21/9a0a7896ce1ce07e83cfad83ca8f5843ce0288cc68d0e6d1d72cd5ca9db8/roughpy-0.1.1-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "287cae8f9c3addd2f1290489ce3f63a8be048802d41ee17b6423b337ea72129b",
                "md5": "65be93762bf24f2f98c100fb6b974233",
                "sha256": "77a22a6e1ed5fafc4f89eda6437e7021e6eb3f4185daa2dccb77d62472faf114"
            },
            "downloads": -1,
            "filename": "roughpy-0.1.1-cp312-cp312-win_amd64.whl",
            "has_sig": false,
            "md5_digest": "65be93762bf24f2f98c100fb6b974233",
            "packagetype": "bdist_wheel",
            "python_version": "cp312",
            "requires_python": null,
            "size": 2931933,
            "upload_time": "2024-01-29T10:50:12",
            "upload_time_iso_8601": "2024-01-29T10:50:12.906141Z",
            "url": "https://files.pythonhosted.org/packages/28/7c/ae8f9c3addd2f1290489ce3f63a8be048802d41ee17b6423b337ea72129b/roughpy-0.1.1-cp312-cp312-win_amd64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "6aeb76224a6f784ce9dbba9bb23e1da7ba7c0a1a432ea82062787667092f2eba",
                "md5": "c26cacb05bfb57c60bd162ca465c62e0",
                "sha256": "2475fabab37230ca2cb3fc06d7d12e5a28b96100ec45a6cc19843d55fd0ae97f"
            },
            "downloads": -1,
            "filename": "roughpy-0.1.1-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl",
            "has_sig": false,
            "md5_digest": "c26cacb05bfb57c60bd162ca465c62e0",
            "packagetype": "bdist_wheel",
            "python_version": "cp38",
            "requires_python": null,
            "size": 7882558,
            "upload_time": "2024-01-29T10:50:16",
            "upload_time_iso_8601": "2024-01-29T10:50:16.111929Z",
            "url": "https://files.pythonhosted.org/packages/6a/eb/76224a6f784ce9dbba9bb23e1da7ba7c0a1a432ea82062787667092f2eba/roughpy-0.1.1-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "fa690ff832ce6f1c1215d81e2f1e18dcb393689834d46d1739b454d71a076d4e",
                "md5": "1d465240f0eaedb9bcf30dbd1172bd6c",
                "sha256": "c2ab96131df7be544c809b79b54b9cc32e4c636753cfbe9efa8dbf2c5c8b0061"
            },
            "downloads": -1,
            "filename": "roughpy-0.1.1-cp38-cp38-win_amd64.whl",
            "has_sig": false,
            "md5_digest": "1d465240f0eaedb9bcf30dbd1172bd6c",
            "packagetype": "bdist_wheel",
            "python_version": "cp38",
            "requires_python": null,
            "size": 2957108,
            "upload_time": "2024-01-29T10:50:18",
            "upload_time_iso_8601": "2024-01-29T10:50:18.373496Z",
            "url": "https://files.pythonhosted.org/packages/fa/69/0ff832ce6f1c1215d81e2f1e18dcb393689834d46d1739b454d71a076d4e/roughpy-0.1.1-cp38-cp38-win_amd64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "0f6bf33b243330007fb42bb50b81c11cbd32139282fb2d94b5e801d761d329df",
                "md5": "cdda37a53cddfa3548806fe2c8883de3",
                "sha256": "2873fff4991d0a8f2655e5bc604ec7867f4e245bd68c07b4189c48bc0f08e2e7"
            },
            "downloads": -1,
            "filename": "roughpy-0.1.1-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl",
            "has_sig": false,
            "md5_digest": "cdda37a53cddfa3548806fe2c8883de3",
            "packagetype": "bdist_wheel",
            "python_version": "cp39",
            "requires_python": null,
            "size": 7884609,
            "upload_time": "2024-01-29T10:50:21",
            "upload_time_iso_8601": "2024-01-29T10:50:21.384699Z",
            "url": "https://files.pythonhosted.org/packages/0f/6b/f33b243330007fb42bb50b81c11cbd32139282fb2d94b5e801d761d329df/roughpy-0.1.1-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "579dbecfb5fdda700f5641a0494f3e7ebda82402081ba783241db0ed90d5357b",
                "md5": "d245385176ce1c6a0cc090e149860a95",
                "sha256": "924ef6d2e296198099d60eb12250b4ab7c8312cf9bb70f266b93b57d892f65f5"
            },
            "downloads": -1,
            "filename": "roughpy-0.1.1-cp39-cp39-win_amd64.whl",
            "has_sig": false,
            "md5_digest": "d245385176ce1c6a0cc090e149860a95",
            "packagetype": "bdist_wheel",
            "python_version": "cp39",
            "requires_python": null,
            "size": 2911039,
            "upload_time": "2024-01-29T10:50:23",
            "upload_time_iso_8601": "2024-01-29T10:50:23.516107Z",
            "url": "https://files.pythonhosted.org/packages/57/9d/becfb5fdda700f5641a0494f3e7ebda82402081ba783241db0ed90d5357b/roughpy-0.1.1-cp39-cp39-win_amd64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "c7b169750218722e7d93084172c5393abe68b90350ffa39b8b47ae9647aeea08",
                "md5": "251c1c5456beec04d5542ef27209db54",
                "sha256": "df6d0e2e33ff2ccbce5f3cf93c40b575e8215861c2bf3792c4ded98d7c7e64d7"
            },
            "downloads": -1,
            "filename": "roughpy-0.1.1.tar.gz",
            "has_sig": false,
            "md5_digest": "251c1c5456beec04d5542ef27209db54",
            "packagetype": "sdist",
            "python_version": "source",
            "requires_python": null,
            "size": 986734,
            "upload_time": "2024-01-29T10:50:25",
            "upload_time_iso_8601": "2024-01-29T10:50:25.821135Z",
            "url": "https://files.pythonhosted.org/packages/c7/b1/69750218722e7d93084172c5393abe68b90350ffa39b8b47ae9647aeea08/roughpy-0.1.1.tar.gz",
            "yanked": false,
            "yanked_reason": null
        }
    ],
    "upload_time": "2024-01-29 10:50:25",
    "github": false,
    "gitlab": false,
    "bitbucket": false,
    "codeberg": false,
    "lcname": "roughpy"
}
        
Elapsed time: 0.16874s