[](https://badge.fury.io/py/tinybrain)  
# tinybrain
Image pyramid generation specialized for connectomics data types and procedures. If your brain wasn't tiny before, it will be now.  
```python 
import tinybrain 
img = load_3d_em_stack()
# 2x2 and 2x2x2 downsamples are on a fast path.
# e.g. (2,2), (2,2,1), (2,2,1,1), (2,2,2), (2,2,2,1)
img_pyramid = tinybrain.downsample_with_averaging(img, factor=(2,2,1), num_mips=5, sparse=False)
labels = load_3d_labels()
label_pyramid = tinybrain.downsample_segmentation(labels, factor=(2,2,1), num_mips=5, sparse=False)
# We also have a few other types
img_pyramid = tinybrain.downsample_with_min_pooling(image, factor=(2,2,1), num_mips=5)
img_pyramid = tinybrain.downsample_with_max_pooling(image, factor=(2,2,1), num_mips=5)
img_pyramid = tinybrain.downsample_with_striding(image, factor=(2,2,1), num_mips=5)
```
## Installation 
```bash
pip install numpy
pip install tinybrain
```
## Motivation
Image hierarchy generation in connectomics uses a few different techniques for
visualizing data, but predominantly we create image pyramids of uint8 grayscale images using 2x2 average pooling and of uint8 to uint64 segmentation labels using 2x2 mode pooling. When images become very large and people wish to visualize upper mip levels using three axes at once, it becomes desirable to perform 2x2x2 downsamples to maintain isotropy.
It's possible to compute both of these using numpy, however as multiple packages found it useful to copy the downsample functions, it makes sense to formalize these functions into a seperate library located on PyPI.
Given the disparate circumstances that they will be used in, these functions should work 
fast as possible with low memory usage and avoid numerical issues such as integer truncation
while generating multiple mip levels.
## Considerations: downsample_with_averaging 
It's advisable to generate multiple mip levels at once rather than recursively computing
new images as for integer type images, this leads to integer truncation issues. In the common
case of 2x2x1 downsampling, a recursively computed image would lose 0.75 brightness per a 
mip level. Therefore, take advantage of the `num_mips` argument which strikes a balance
that limits integer truncation loss to once every 4 mip levels. This compromise allows
for the use of integer arithmatic and no more memory usage than 2x the input image including
the output downsamples. If you seek to eliminate the loss beyond 4 mip levels, try promoting 
the type before downsampling. 2x2x2x1 downsamples truncate every 8 mip levels.
A C++ high performance path is triggered for 2x2x1x1 and 2x2x2x1 downsample factors on uint8, uint16, float32, 
and float64 data types in Fortran order. Other factors, data types, and orderings are computed using a numpy pathway that is much slower and more memory intensive.
We also include a sparse mode for downsampling 2x2x2 patches, which prevents "ghosting" where one z-slice overlaps a black region on the next slice and becomes semi-transparent after downsampling. We deal with this by neglecting the background pixels from the averaging operation. 
### Example Benchmark 
On a 1024x1024x100 uint8 image I ran the following code. PIL and OpenCV are actually much faster than this benchmark shows because most of the time is spent writing to the numpy array. tinybrain has a large advantage working on 3D and 4D arrays. Of course, this is a very simple benchmark and it may be possible to tune each of these approaches. On single slices, Pillow was faster than tinybrain.
```python
img = np.load("image.npy")
s = time.time()
downsample_with_averaging(img, (2,2,1))
print("Original ", time.time() - s)
s = time.time()
out = tinybrain.downsample_with_averaging(img, (2,2,1))
print("tinybrain ", time.time() - s)
s = time.time()
out = np.zeros(shape=(512,512,100))
for z in range(img.shape[2]):
  out[:,:,z] = cv2.resize(img[:,:,z], dsize=(512, 512) )
print("OpenCV ", time.time() - s)
s = time.time()
out = np.zeros(shape=(512,512,100))
for z in range(img.shape[2]):
  pilimg = Image.fromarray(img[:,:,z])
  out[:,:,z] = pilimg.resize( (512, 512) )
print("Pillow ", time.time() - s)
# Method     Run Time             Rel. Perf.
# Original   1820 ms +/- 3.73 ms    1.0x
# tinybrain    67 ms +/- 0.40 ms   27.2x 
# OpenCV      469 ms +/- 1.12 ms    3.9x
# Pillow      937 ms +/- 7.63 ms    1.9x
```
Here's the output from `perf.py` on an Apple Silicon 2021 Macbook Pro M1.
Note that the image used was a random 2048x2048x64 array that was a uint8
for average pooling and a uint64 for mode pooling to represent real use cases more fairly. In the table, read it as 2D or 3D downsamples, generating a single or multiple mip levels, with sparse mode enabled or disabled. The speed values are in megavoxels per a second and are the mean of ten runs.
| dwnsmpl  |   mips  |   sparse  |   AVG (MVx/sec)  |   MODE (MVx/sec)  |
|----------|---------|-----------|------------------|-------------------|
|   2x2    |   1     |   N       |   3856.07        |   1057.87         |
|   2x2    |   2     |   N       |   2685.80        |   1062.69         |
|   2x2    |   1     |   Y       |   N/A            |   129.64          |
|   2x2    |   2     |   Y       |   N/A            |   81.62           |
|   2x2x2  |   1     |   N       |   4468.55        |   336.85          |
|   2x2x2  |   2     |   N       |   2867.80        |   298.45          |
|   2x2x2  |   1     |   Y       |   1389.47        |   337.87          |
|   2x2x2  |   2     |   Y       |   1259.58        |   293.84          |
As the downsampling code's performance is data dependent due to branching, I also used [`connectomics.npy`](https://github.com/seung-lab/connected-components-3d/blob/master/benchmarks/connectomics.npy.gz) (512<sup>3</sup> uint32 extended to uint64) to see how that affected performance. This data comes from mouse visual cortex and has many equal adjacent voxels. In this volume, the 2x2x2 non-sparse mode is much faster as the "instant" majority detection can skip examining half the voxels in many cases.
| dwnsmpl  |   mips  |   sparse  |   MODE (MVx/sec)  |
|----------|---------|-----------|-------------------|
|   2x2    |   1     |   N       |   1078.09         |
|   2x2    |   2     |   N       |   1030.90         |
|   2x2    |   1     |   Y       |   146.15          |
|   2x2    |   2     |   Y       |   69.25           |
|   2x2x2  |   1     |   N       |   1966.74         |
|   2x2x2  |   2     |   N       |   1790.60         |
|   2x2x2  |   1     |   Y       |   2041.96         |
|   2x2x2  |   2     |   Y       |   1758.42         |
## Considerations: downsample_segmentation 
The `downsample_segmentation` function performs mode pooling operations provided the downsample factor is a power of two, including in three dimensions. If the factor is a non-power of two, striding is used. The mode pooling, which is usually what you want, is computed recursively. Mode pooling is superior to striding, but the recursive calculation can introduce defects at mip levels higher than 1. This may be improved in the future.  
The way the calculation is actually done uses an ensemble of several different methods. For (2,2,1,1) and (2,2,2,1) downsamples, a Cython fast, low memory path is selected. (2,2,1,1) implements [*countless if*](https://towardsdatascience.com/countless-high-performance-2x-downsampling-of-labeled-images-using-python-and-numpy-e70ad3275589). (2,2,2,1) uses a combination of counting and "instant" majority detection. For (4,4,1) or other 2D powers of two, the [*countless 2d*](https://towardsdatascience.com/countless-high-performance-2x-downsampling-of-labeled-images-using-python-and-numpy-e70ad3275589) algorithm is used. For (4,4,4), (8,8,8) etc, the [*dynamic countless 3d*](https://towardsdatascience.com/countless-3d-vectorized-2x-downsampling-of-labeled-volume-images-using-python-and-numpy-59d686c2f75) algorithm is used. For 2D powers of two, [*stippled countless 2d*](https://medium.com/@willsilversmith/countless-2d-inflated-2x-downsampling-of-labeled-images-holding-zero-values-as-background-4d13a7675f2d) is used if the sparse flag is enabled. For all other configurations, striding is used.  
Countless 2d paths are also fast, but use slightly more memory and time. Countless 3D is okay for (2,2,2) and (4,4,4) but will use time and memory exponential in the product of dimensions. This state of affairs could be improved by implementing a counting based algorithm in Cython/C++ for arbitrary factors that doesn't compute recursively. The countless algorithms were developed before I knew how to write Cython and package libraries. However, C++ implementations of countless are much faster than counting for computing the first 2x2x1 mip level. In particular, an AVX2 SIMD implementation can saturate memory bandwidth.    
Documentation for the countless algorithm family is located here: https://github.com/william-silversmith/countless
            
         
        Raw data
        
            {
    "_id": null,
    "home_page": "https://github.com/seung-lab/tinybrain/",
    "name": "tinybrain",
    "maintainer": null,
    "docs_url": null,
    "requires_python": ">=3.9",
    "maintainer_email": null,
    "keywords": null,
    "author": "William Silversmith",
    "author_email": "ws9@princeton.edu",
    "download_url": "https://files.pythonhosted.org/packages/1b/d9/a4b311e5fe7765c9b160043909ae158acd4deb027029b67caee2326c45a5/tinybrain-1.7.0.tar.gz",
    "platform": null,
    "description": "[](https://badge.fury.io/py/tinybrain)  \n\n# tinybrain\n\nImage pyramid generation specialized for connectomics data types and procedures. If your brain wasn't tiny before, it will be now.  \n\n```python \nimport tinybrain \n\nimg = load_3d_em_stack()\n\n# 2x2 and 2x2x2 downsamples are on a fast path.\n# e.g. (2,2), (2,2,1), (2,2,1,1), (2,2,2), (2,2,2,1)\nimg_pyramid = tinybrain.downsample_with_averaging(img, factor=(2,2,1), num_mips=5, sparse=False)\n\nlabels = load_3d_labels()\nlabel_pyramid = tinybrain.downsample_segmentation(labels, factor=(2,2,1), num_mips=5, sparse=False)\n\n# We also have a few other types\nimg_pyramid = tinybrain.downsample_with_min_pooling(image, factor=(2,2,1), num_mips=5)\nimg_pyramid = tinybrain.downsample_with_max_pooling(image, factor=(2,2,1), num_mips=5)\nimg_pyramid = tinybrain.downsample_with_striding(image, factor=(2,2,1), num_mips=5)\n```\n\n## Installation \n\n```bash\npip install numpy\npip install tinybrain\n```\n\n## Motivation\n\nImage hierarchy generation in connectomics uses a few different techniques for\nvisualizing data, but predominantly we create image pyramids of uint8 grayscale images using 2x2 average pooling and of uint8 to uint64 segmentation labels using 2x2 mode pooling. When images become very large and people wish to visualize upper mip levels using three axes at once, it becomes desirable to perform 2x2x2 downsamples to maintain isotropy.\n\nIt's possible to compute both of these using numpy, however as multiple packages found it useful to copy the downsample functions, it makes sense to formalize these functions into a seperate library located on PyPI.\n\nGiven the disparate circumstances that they will be used in, these functions should work \nfast as possible with low memory usage and avoid numerical issues such as integer truncation\nwhile generating multiple mip levels.\n\n## Considerations: downsample_with_averaging \n\nIt's advisable to generate multiple mip levels at once rather than recursively computing\nnew images as for integer type images, this leads to integer truncation issues. In the common\ncase of 2x2x1 downsampling, a recursively computed image would lose 0.75 brightness per a \nmip level. Therefore, take advantage of the `num_mips` argument which strikes a balance\nthat limits integer truncation loss to once every 4 mip levels. This compromise allows\nfor the use of integer arithmatic and no more memory usage than 2x the input image including\nthe output downsamples. If you seek to eliminate the loss beyond 4 mip levels, try promoting \nthe type before downsampling. 2x2x2x1 downsamples truncate every 8 mip levels.\n\nA C++ high performance path is triggered for 2x2x1x1 and 2x2x2x1 downsample factors on uint8, uint16, float32, \nand float64 data types in Fortran order. Other factors, data types, and orderings are computed using a numpy pathway that is much slower and more memory intensive.\n\nWe also include a sparse mode for downsampling 2x2x2 patches, which prevents \"ghosting\" where one z-slice overlaps a black region on the next slice and becomes semi-transparent after downsampling. We deal with this by neglecting the background pixels from the averaging operation. \n\n### Example Benchmark \n\nOn a 1024x1024x100 uint8 image I ran the following code. PIL and OpenCV are actually much faster than this benchmark shows because most of the time is spent writing to the numpy array. tinybrain has a large advantage working on 3D and 4D arrays. Of course, this is a very simple benchmark and it may be possible to tune each of these approaches. On single slices, Pillow was faster than tinybrain.\n\n```python\nimg = np.load(\"image.npy\")\n\ns = time.time()\ndownsample_with_averaging(img, (2,2,1))\nprint(\"Original \", time.time() - s)\n\ns = time.time()\nout = tinybrain.downsample_with_averaging(img, (2,2,1))\nprint(\"tinybrain \", time.time() - s)\n\ns = time.time()\nout = np.zeros(shape=(512,512,100))\nfor z in range(img.shape[2]):\n  out[:,:,z] = cv2.resize(img[:,:,z], dsize=(512, 512) )\nprint(\"OpenCV \", time.time() - s)\n\ns = time.time()\nout = np.zeros(shape=(512,512,100))\nfor z in range(img.shape[2]):\n  pilimg = Image.fromarray(img[:,:,z])\n  out[:,:,z] = pilimg.resize( (512, 512) )\nprint(\"Pillow \", time.time() - s)\n\n# Method     Run Time             Rel. Perf.\n# Original   1820 ms +/- 3.73 ms    1.0x\n# tinybrain    67 ms +/- 0.40 ms   27.2x \n# OpenCV      469 ms +/- 1.12 ms    3.9x\n# Pillow      937 ms +/- 7.63 ms    1.9x\n```\n\nHere's the output from `perf.py` on an Apple Silicon 2021 Macbook Pro M1.\nNote that the image used was a random 2048x2048x64 array that was a uint8\nfor average pooling and a uint64 for mode pooling to represent real use cases more fairly. In the table, read it as 2D or 3D downsamples, generating a single or multiple mip levels, with sparse mode enabled or disabled. The speed values are in megavoxels per a second and are the mean of ten runs.\n\n\n| dwnsmpl  |   mips  |   sparse  |   AVG (MVx/sec)  |   MODE (MVx/sec)  |\n|----------|---------|-----------|------------------|-------------------|\n|   2x2    |   1     |   N       |   3856.07        |   1057.87         |\n|   2x2    |   2     |   N       |   2685.80        |   1062.69         |\n|   2x2    |   1     |   Y       |   N/A            |   129.64          |\n|   2x2    |   2     |   Y       |   N/A            |   81.62           |\n|   2x2x2  |   1     |   N       |   4468.55        |   336.85          |\n|   2x2x2  |   2     |   N       |   2867.80        |   298.45          |\n|   2x2x2  |   1     |   Y       |   1389.47        |   337.87          |\n|   2x2x2  |   2     |   Y       |   1259.58        |   293.84          |\n\nAs the downsampling code's performance is data dependent due to branching, I also used [`connectomics.npy`](https://github.com/seung-lab/connected-components-3d/blob/master/benchmarks/connectomics.npy.gz) (512<sup>3</sup> uint32 extended to uint64) to see how that affected performance. This data comes from mouse visual cortex and has many equal adjacent voxels. In this volume, the 2x2x2 non-sparse mode is much faster as the \"instant\" majority detection can skip examining half the voxels in many cases.\n\n| dwnsmpl  |   mips  |   sparse  |   MODE (MVx/sec)  |\n|----------|---------|-----------|-------------------|\n|   2x2    |   1     |   N       |   1078.09         |\n|   2x2    |   2     |   N       |   1030.90         |\n|   2x2    |   1     |   Y       |   146.15          |\n|   2x2    |   2     |   Y       |   69.25           |\n|   2x2x2  |   1     |   N       |   1966.74         |\n|   2x2x2  |   2     |   N       |   1790.60         |\n|   2x2x2  |   1     |   Y       |   2041.96         |\n|   2x2x2  |   2     |   Y       |   1758.42         |\n\n\n## Considerations: downsample_segmentation \n\nThe `downsample_segmentation` function performs mode pooling operations provided the downsample factor is a power of two, including in three dimensions. If the factor is a non-power of two, striding is used. The mode pooling, which is usually what you want, is computed recursively. Mode pooling is superior to striding, but the recursive calculation can introduce defects at mip levels higher than 1. This may be improved in the future.  \n\nThe way the calculation is actually done uses an ensemble of several different methods. For (2,2,1,1) and (2,2,2,1) downsamples, a Cython fast, low memory path is selected. (2,2,1,1) implements [*countless if*](https://towardsdatascience.com/countless-high-performance-2x-downsampling-of-labeled-images-using-python-and-numpy-e70ad3275589). (2,2,2,1) uses a combination of counting and \"instant\" majority detection. For (4,4,1) or other 2D powers of two, the [*countless 2d*](https://towardsdatascience.com/countless-high-performance-2x-downsampling-of-labeled-images-using-python-and-numpy-e70ad3275589) algorithm is used. For (4,4,4), (8,8,8) etc, the [*dynamic countless 3d*](https://towardsdatascience.com/countless-3d-vectorized-2x-downsampling-of-labeled-volume-images-using-python-and-numpy-59d686c2f75) algorithm is used. For 2D powers of two, [*stippled countless 2d*](https://medium.com/@willsilversmith/countless-2d-inflated-2x-downsampling-of-labeled-images-holding-zero-values-as-background-4d13a7675f2d) is used if the sparse flag is enabled. For all other configurations, striding is used.  \n\nCountless 2d paths are also fast, but use slightly more memory and time. Countless 3D is okay for (2,2,2) and (4,4,4) but will use time and memory exponential in the product of dimensions. This state of affairs could be improved by implementing a counting based algorithm in Cython/C++ for arbitrary factors that doesn't compute recursively. The countless algorithms were developed before I knew how to write Cython and package libraries. However, C++ implementations of countless are much faster than counting for computing the first 2x2x1 mip level. In particular, an AVX2 SIMD implementation can saturate memory bandwidth.    \n\nDocumentation for the countless algorithm family is located here: https://github.com/william-silversmith/countless\n\n",
    "bugtrack_url": null,
    "license": null,
    "summary": "Image pyramid generation specialized for connectomics data types and procedures.",
    "version": "1.7.0",
    "project_urls": {
        "Homepage": "https://github.com/seung-lab/tinybrain/"
    },
    "split_keywords": [],
    "urls": [
        {
            "comment_text": null,
            "digests": {
                "blake2b_256": "30291bcb14f9b4081e01261ebbf93815d7cbc3c45e7148b1200c74574b5e0fe4",
                "md5": "10e86745e82cf047d4fbf0d475afc6dc",
                "sha256": "a4192003a39a83907d124e68c6db57bf3b9377c840535029a339e6716dc49652"
            },
            "downloads": -1,
            "filename": "tinybrain-1.7.0-cp310-cp310-macosx_10_9_x86_64.whl",
            "has_sig": false,
            "md5_digest": "10e86745e82cf047d4fbf0d475afc6dc",
            "packagetype": "bdist_wheel",
            "python_version": "cp310",
            "requires_python": ">=3.9",
            "size": 553953,
            "upload_time": "2025-02-07T18:33:21",
            "upload_time_iso_8601": "2025-02-07T18:33:21.924037Z",
            "url": "https://files.pythonhosted.org/packages/30/29/1bcb14f9b4081e01261ebbf93815d7cbc3c45e7148b1200c74574b5e0fe4/tinybrain-1.7.0-cp310-cp310-macosx_10_9_x86_64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": null,
            "digests": {
                "blake2b_256": "84afa5b2e5dbe7d6191222705155cf71e7969507b7f805747a3dc369e2da8027",
                "md5": "ac1984dc7ab0569019aaf7b5359b137c",
                "sha256": "d54690add8ba786222ddfdc32c6bf2d67594fd93d2c342a7f047f1b9c19fc19f"
            },
            "downloads": -1,
            "filename": "tinybrain-1.7.0-cp310-cp310-macosx_11_0_arm64.whl",
            "has_sig": false,
            "md5_digest": "ac1984dc7ab0569019aaf7b5359b137c",
            "packagetype": "bdist_wheel",
            "python_version": "cp310",
            "requires_python": ">=3.9",
            "size": 466781,
            "upload_time": "2025-02-07T18:33:23",
            "upload_time_iso_8601": "2025-02-07T18:33:23.501435Z",
            "url": "https://files.pythonhosted.org/packages/84/af/a5b2e5dbe7d6191222705155cf71e7969507b7f805747a3dc369e2da8027/tinybrain-1.7.0-cp310-cp310-macosx_11_0_arm64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": null,
            "digests": {
                "blake2b_256": "2376662cc2bcdf9d16201ab8917829e258f9f2cc7442e759d665e58843404884",
                "md5": "b606ddd9e237d4d30150cf0b36607522",
                "sha256": "50f68d55f01ce215ac91ec842221f63eb9fdbc6298993d8a9f67a62e8f8e4c80"
            },
            "downloads": -1,
            "filename": "tinybrain-1.7.0-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl",
            "has_sig": false,
            "md5_digest": "b606ddd9e237d4d30150cf0b36607522",
            "packagetype": "bdist_wheel",
            "python_version": "cp310",
            "requires_python": ">=3.9",
            "size": 3586400,
            "upload_time": "2025-02-07T18:33:26",
            "upload_time_iso_8601": "2025-02-07T18:33:26.939739Z",
            "url": "https://files.pythonhosted.org/packages/23/76/662cc2bcdf9d16201ab8917829e258f9f2cc7442e759d665e58843404884/tinybrain-1.7.0-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": null,
            "digests": {
                "blake2b_256": "a8ea2e664d5e719632d97fa4d07cf2998df7eec66f41a0ecb691f645b893e120",
                "md5": "2b016b8c7b842d40d705d0258c1c5dc7",
                "sha256": "1ca94f845f0b2c47a2f00abb4cdb6b3956ceb247f81977f3342f353468407348"
            },
            "downloads": -1,
            "filename": "tinybrain-1.7.0-cp310-cp310-win_amd64.whl",
            "has_sig": false,
            "md5_digest": "2b016b8c7b842d40d705d0258c1c5dc7",
            "packagetype": "bdist_wheel",
            "python_version": "cp310",
            "requires_python": ">=3.9",
            "size": 382437,
            "upload_time": "2025-02-07T18:33:28",
            "upload_time_iso_8601": "2025-02-07T18:33:28.388317Z",
            "url": "https://files.pythonhosted.org/packages/a8/ea/2e664d5e719632d97fa4d07cf2998df7eec66f41a0ecb691f645b893e120/tinybrain-1.7.0-cp310-cp310-win_amd64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": null,
            "digests": {
                "blake2b_256": "67d7f6c675ee8e0a10d2f5a347efc0a64e6abbcae924ce8e456e8ad62d67a80b",
                "md5": "eacc254e07779ce2ec1f019246baf13f",
                "sha256": "784036751ac9e9e0b33b8aaa8955701221f406b065b64878e98352ded5620936"
            },
            "downloads": -1,
            "filename": "tinybrain-1.7.0-cp311-cp311-macosx_10_9_x86_64.whl",
            "has_sig": false,
            "md5_digest": "eacc254e07779ce2ec1f019246baf13f",
            "packagetype": "bdist_wheel",
            "python_version": "cp311",
            "requires_python": ">=3.9",
            "size": 555407,
            "upload_time": "2025-02-07T18:33:29",
            "upload_time_iso_8601": "2025-02-07T18:33:29.673494Z",
            "url": "https://files.pythonhosted.org/packages/67/d7/f6c675ee8e0a10d2f5a347efc0a64e6abbcae924ce8e456e8ad62d67a80b/tinybrain-1.7.0-cp311-cp311-macosx_10_9_x86_64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": null,
            "digests": {
                "blake2b_256": "e0be6b49595ae3aed16487a8724d1f9875965b217e66274b84ec5195f0339cc6",
                "md5": "73a6f85d3cbe1a1e0d6f10fe495aa72d",
                "sha256": "f2494cb536f6c9ab9a8df421f3a7ec3d674092574bc5b38281be3f38be205958"
            },
            "downloads": -1,
            "filename": "tinybrain-1.7.0-cp311-cp311-macosx_11_0_arm64.whl",
            "has_sig": false,
            "md5_digest": "73a6f85d3cbe1a1e0d6f10fe495aa72d",
            "packagetype": "bdist_wheel",
            "python_version": "cp311",
            "requires_python": ">=3.9",
            "size": 467264,
            "upload_time": "2025-02-07T18:33:32",
            "upload_time_iso_8601": "2025-02-07T18:33:32.230212Z",
            "url": "https://files.pythonhosted.org/packages/e0/be/6b49595ae3aed16487a8724d1f9875965b217e66274b84ec5195f0339cc6/tinybrain-1.7.0-cp311-cp311-macosx_11_0_arm64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": null,
            "digests": {
                "blake2b_256": "090996c39a3b5b5f7b45eff71a023cab26e067b279299a1cbab566d4dcfda71b",
                "md5": "0ccbbad773f9e0de9ed04e7888a29345",
                "sha256": "a318f16585c1687fe9dbe03cb6a84948c15bf4df68622fe8abbe0ae52c9eeff0"
            },
            "downloads": -1,
            "filename": "tinybrain-1.7.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl",
            "has_sig": false,
            "md5_digest": "0ccbbad773f9e0de9ed04e7888a29345",
            "packagetype": "bdist_wheel",
            "python_version": "cp311",
            "requires_python": ">=3.9",
            "size": 3744416,
            "upload_time": "2025-02-07T18:33:34",
            "upload_time_iso_8601": "2025-02-07T18:33:34.333260Z",
            "url": "https://files.pythonhosted.org/packages/09/09/96c39a3b5b5f7b45eff71a023cab26e067b279299a1cbab566d4dcfda71b/tinybrain-1.7.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": null,
            "digests": {
                "blake2b_256": "caac357e04210681bec262899fa1938d22e5cf85ae755a2b41e072435f935e41",
                "md5": "7b9b2cf43194fbc93d0c5733681c6106",
                "sha256": "083ce9b01c4b7e86318da2b573f2310bd345aa4a97f15c3e8f35b4b9ed5f1d50"
            },
            "downloads": -1,
            "filename": "tinybrain-1.7.0-cp311-cp311-win_amd64.whl",
            "has_sig": false,
            "md5_digest": "7b9b2cf43194fbc93d0c5733681c6106",
            "packagetype": "bdist_wheel",
            "python_version": "cp311",
            "requires_python": ">=3.9",
            "size": 382794,
            "upload_time": "2025-02-07T18:33:37",
            "upload_time_iso_8601": "2025-02-07T18:33:37.511053Z",
            "url": "https://files.pythonhosted.org/packages/ca/ac/357e04210681bec262899fa1938d22e5cf85ae755a2b41e072435f935e41/tinybrain-1.7.0-cp311-cp311-win_amd64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": null,
            "digests": {
                "blake2b_256": "8bb9df611d52d7c064da7311524bb1d880b0db84ca5f0532dad3f80141c66b57",
                "md5": "fd46b68713098a38a6a8104c1e63e5e5",
                "sha256": "84598b4a60e12370e8c405f56a4d536a4711874ed1abdb9d95d798480fc48932"
            },
            "downloads": -1,
            "filename": "tinybrain-1.7.0-cp312-cp312-macosx_10_13_x86_64.whl",
            "has_sig": false,
            "md5_digest": "fd46b68713098a38a6a8104c1e63e5e5",
            "packagetype": "bdist_wheel",
            "python_version": "cp312",
            "requires_python": ">=3.9",
            "size": 522008,
            "upload_time": "2025-02-07T18:33:38",
            "upload_time_iso_8601": "2025-02-07T18:33:38.970421Z",
            "url": "https://files.pythonhosted.org/packages/8b/b9/df611d52d7c064da7311524bb1d880b0db84ca5f0532dad3f80141c66b57/tinybrain-1.7.0-cp312-cp312-macosx_10_13_x86_64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": null,
            "digests": {
                "blake2b_256": "df7c3c4434d50a2c4094ccf25ba7af9ee670e722f8194ea94976a9e569de64dc",
                "md5": "8b9916932a0159cf236d9a72b24d54e0",
                "sha256": "6269fad69af8db0815f95fcff22c5daf119353dc097ce16135694f6c58683d3e"
            },
            "downloads": -1,
            "filename": "tinybrain-1.7.0-cp312-cp312-macosx_11_0_arm64.whl",
            "has_sig": false,
            "md5_digest": "8b9916932a0159cf236d9a72b24d54e0",
            "packagetype": "bdist_wheel",
            "python_version": "cp312",
            "requires_python": ">=3.9",
            "size": 455087,
            "upload_time": "2025-02-07T18:33:40",
            "upload_time_iso_8601": "2025-02-07T18:33:40.329913Z",
            "url": "https://files.pythonhosted.org/packages/df/7c/3c4434d50a2c4094ccf25ba7af9ee670e722f8194ea94976a9e569de64dc/tinybrain-1.7.0-cp312-cp312-macosx_11_0_arm64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": null,
            "digests": {
                "blake2b_256": "9b701a0036a13b0c9467d27ef1b08a2562a5ed23e53aaced63ecff2f845db7a4",
                "md5": "9e2efef23b78a1f17b4a808f4d7a4dfd",
                "sha256": "83b0256e12713ca2ecccba635559ed0fc43ccb7f6974e429a04a728002f59f31"
            },
            "downloads": -1,
            "filename": "tinybrain-1.7.0-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl",
            "has_sig": false,
            "md5_digest": "9e2efef23b78a1f17b4a808f4d7a4dfd",
            "packagetype": "bdist_wheel",
            "python_version": "cp312",
            "requires_python": ">=3.9",
            "size": 3676686,
            "upload_time": "2025-02-07T18:33:42",
            "upload_time_iso_8601": "2025-02-07T18:33:42.142144Z",
            "url": "https://files.pythonhosted.org/packages/9b/70/1a0036a13b0c9467d27ef1b08a2562a5ed23e53aaced63ecff2f845db7a4/tinybrain-1.7.0-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": null,
            "digests": {
                "blake2b_256": "980b1603f6187700ec18b537ba4dbc16f4539f552a9be9ad24d97396276a316b",
                "md5": "bc80ef7184d9837842ef3436d4cf6a6a",
                "sha256": "5a1c36dced6fd6a9e1487f07fea49618ee7b8fbbfd1dc59b625f4c06c50a88a5"
            },
            "downloads": -1,
            "filename": "tinybrain-1.7.0-cp312-cp312-win_amd64.whl",
            "has_sig": false,
            "md5_digest": "bc80ef7184d9837842ef3436d4cf6a6a",
            "packagetype": "bdist_wheel",
            "python_version": "cp312",
            "requires_python": ">=3.9",
            "size": 356527,
            "upload_time": "2025-02-07T18:33:43",
            "upload_time_iso_8601": "2025-02-07T18:33:43.563597Z",
            "url": "https://files.pythonhosted.org/packages/98/0b/1603f6187700ec18b537ba4dbc16f4539f552a9be9ad24d97396276a316b/tinybrain-1.7.0-cp312-cp312-win_amd64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": null,
            "digests": {
                "blake2b_256": "f0fb0db2802ef44a4b8d800be8117e44c7afbce65d1888bd83a2a7c7359ccd80",
                "md5": "6cc4c4eaf309da3d249ebba420d50308",
                "sha256": "aa5a668a468bd6821c6e9900d8b9a098868de049a4b64d3a1d3f336bffbe96b2"
            },
            "downloads": -1,
            "filename": "tinybrain-1.7.0-cp313-cp313-macosx_10_13_x86_64.whl",
            "has_sig": false,
            "md5_digest": "6cc4c4eaf309da3d249ebba420d50308",
            "packagetype": "bdist_wheel",
            "python_version": "cp313",
            "requires_python": ">=3.9",
            "size": 517292,
            "upload_time": "2025-02-07T18:33:44",
            "upload_time_iso_8601": "2025-02-07T18:33:44.864400Z",
            "url": "https://files.pythonhosted.org/packages/f0/fb/0db2802ef44a4b8d800be8117e44c7afbce65d1888bd83a2a7c7359ccd80/tinybrain-1.7.0-cp313-cp313-macosx_10_13_x86_64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": null,
            "digests": {
                "blake2b_256": "d36d47288acca9a118b5d1b29205e11203ced0afec0609afdeba255489d2ad68",
                "md5": "28a630cc5be1cb53180f7ef4d955ccaa",
                "sha256": "07fcbb91c18f918b8f1c517836ed6a066f48e2e556dfe90e0deeb79b00325ff2"
            },
            "downloads": -1,
            "filename": "tinybrain-1.7.0-cp313-cp313-macosx_11_0_arm64.whl",
            "has_sig": false,
            "md5_digest": "28a630cc5be1cb53180f7ef4d955ccaa",
            "packagetype": "bdist_wheel",
            "python_version": "cp313",
            "requires_python": ">=3.9",
            "size": 446981,
            "upload_time": "2025-02-07T18:33:50",
            "upload_time_iso_8601": "2025-02-07T18:33:50.323549Z",
            "url": "https://files.pythonhosted.org/packages/d3/6d/47288acca9a118b5d1b29205e11203ced0afec0609afdeba255489d2ad68/tinybrain-1.7.0-cp313-cp313-macosx_11_0_arm64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": null,
            "digests": {
                "blake2b_256": "5583b317540cc1ed57259c6deaafd66d4a8de3a444b38edd89ab9c2e609fae4c",
                "md5": "4d06616352c36d937387eec8d1397127",
                "sha256": "e18223f303dad96800927e5abaf71ca8918567af12456e2fc42f2bd372abccb4"
            },
            "downloads": -1,
            "filename": "tinybrain-1.7.0-cp313-cp313-manylinux_2_17_x86_64.manylinux2014_x86_64.whl",
            "has_sig": false,
            "md5_digest": "4d06616352c36d937387eec8d1397127",
            "packagetype": "bdist_wheel",
            "python_version": "cp313",
            "requires_python": ">=3.9",
            "size": 3653530,
            "upload_time": "2025-02-07T18:33:51",
            "upload_time_iso_8601": "2025-02-07T18:33:51.969515Z",
            "url": "https://files.pythonhosted.org/packages/55/83/b317540cc1ed57259c6deaafd66d4a8de3a444b38edd89ab9c2e609fae4c/tinybrain-1.7.0-cp313-cp313-manylinux_2_17_x86_64.manylinux2014_x86_64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": null,
            "digests": {
                "blake2b_256": "5e4eeeb98e5218eff7551735d6ff0af9396fd16bfde02239de78aaa9bc19cf93",
                "md5": "7de84c5b7f10c4e47f31a60d30d3877b",
                "sha256": "41258f94f76c5b43613cb18bfe439b72d0a892a131ed19226e49fc2fee2e1881"
            },
            "downloads": -1,
            "filename": "tinybrain-1.7.0-cp313-cp313-win_amd64.whl",
            "has_sig": false,
            "md5_digest": "7de84c5b7f10c4e47f31a60d30d3877b",
            "packagetype": "bdist_wheel",
            "python_version": "cp313",
            "requires_python": ">=3.9",
            "size": 356206,
            "upload_time": "2025-02-07T18:33:53",
            "upload_time_iso_8601": "2025-02-07T18:33:53.480687Z",
            "url": "https://files.pythonhosted.org/packages/5e/4e/eeb98e5218eff7551735d6ff0af9396fd16bfde02239de78aaa9bc19cf93/tinybrain-1.7.0-cp313-cp313-win_amd64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": null,
            "digests": {
                "blake2b_256": "87b55c33c079da86963fd0a7ab0b178a8e6955930200643a6b9e621b26449a88",
                "md5": "24fca6e0b578f36b10ef5e13725f3df5",
                "sha256": "e88c00825dda693fde8f382e41d7b4026cf0703d5a04775f170000e7dd6c7028"
            },
            "downloads": -1,
            "filename": "tinybrain-1.7.0-cp39-cp39-macosx_10_9_x86_64.whl",
            "has_sig": false,
            "md5_digest": "24fca6e0b578f36b10ef5e13725f3df5",
            "packagetype": "bdist_wheel",
            "python_version": "cp39",
            "requires_python": ">=3.9",
            "size": 554821,
            "upload_time": "2025-02-07T18:33:54",
            "upload_time_iso_8601": "2025-02-07T18:33:54.926084Z",
            "url": "https://files.pythonhosted.org/packages/87/b5/5c33c079da86963fd0a7ab0b178a8e6955930200643a6b9e621b26449a88/tinybrain-1.7.0-cp39-cp39-macosx_10_9_x86_64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": null,
            "digests": {
                "blake2b_256": "7c0ba5b98c8d9b5bfd65e00d2a72458dc2eaace2097413c5d117540d08e65c33",
                "md5": "7e977877ca73f323acd5cee400437ac7",
                "sha256": "20da6c11915be05eb78f8b81b5ffca5afc7c11afb38fdd2b8e0cf8fb65563625"
            },
            "downloads": -1,
            "filename": "tinybrain-1.7.0-cp39-cp39-macosx_11_0_arm64.whl",
            "has_sig": false,
            "md5_digest": "7e977877ca73f323acd5cee400437ac7",
            "packagetype": "bdist_wheel",
            "python_version": "cp39",
            "requires_python": ">=3.9",
            "size": 467350,
            "upload_time": "2025-02-07T18:33:56",
            "upload_time_iso_8601": "2025-02-07T18:33:56.364136Z",
            "url": "https://files.pythonhosted.org/packages/7c/0b/a5b98c8d9b5bfd65e00d2a72458dc2eaace2097413c5d117540d08e65c33/tinybrain-1.7.0-cp39-cp39-macosx_11_0_arm64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": null,
            "digests": {
                "blake2b_256": "d0c2192a20ce9002875e940af97dc0131f0282ea198346ed6220a6341c0dccde",
                "md5": "c923da1732133e6c002f2d71e537adcb",
                "sha256": "ff637cb360046e28c345aea843d247a5659a70894e59a3428a447dea195d9b09"
            },
            "downloads": -1,
            "filename": "tinybrain-1.7.0-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl",
            "has_sig": false,
            "md5_digest": "c923da1732133e6c002f2d71e537adcb",
            "packagetype": "bdist_wheel",
            "python_version": "cp39",
            "requires_python": ">=3.9",
            "size": 3582345,
            "upload_time": "2025-02-07T18:34:00",
            "upload_time_iso_8601": "2025-02-07T18:34:00.443860Z",
            "url": "https://files.pythonhosted.org/packages/d0/c2/192a20ce9002875e940af97dc0131f0282ea198346ed6220a6341c0dccde/tinybrain-1.7.0-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": null,
            "digests": {
                "blake2b_256": "a2db66416a0bcfb67b0a696fceeca9741cd6f06e6bb928ba98d4afb63e334be6",
                "md5": "e702b93735fe37854cfc60184be1a0fd",
                "sha256": "6e0e82a2bb46ac9157692d8168df7d5bd04a44d41a01b7f8cb25a6dfd139f8ed"
            },
            "downloads": -1,
            "filename": "tinybrain-1.7.0-cp39-cp39-win_amd64.whl",
            "has_sig": false,
            "md5_digest": "e702b93735fe37854cfc60184be1a0fd",
            "packagetype": "bdist_wheel",
            "python_version": "cp39",
            "requires_python": ">=3.9",
            "size": 382901,
            "upload_time": "2025-02-07T18:34:02",
            "upload_time_iso_8601": "2025-02-07T18:34:02.970634Z",
            "url": "https://files.pythonhosted.org/packages/a2/db/66416a0bcfb67b0a696fceeca9741cd6f06e6bb928ba98d4afb63e334be6/tinybrain-1.7.0-cp39-cp39-win_amd64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": null,
            "digests": {
                "blake2b_256": "1bd9a4b311e5fe7765c9b160043909ae158acd4deb027029b67caee2326c45a5",
                "md5": "5d90112cd32f9292c79a32425c25d7b9",
                "sha256": "31937e73e7d87727a9a67cb1f9d8a76f4d0dcda47318cf40331f6800c1ee8e89"
            },
            "downloads": -1,
            "filename": "tinybrain-1.7.0.tar.gz",
            "has_sig": false,
            "md5_digest": "5d90112cd32f9292c79a32425c25d7b9",
            "packagetype": "sdist",
            "python_version": "source",
            "requires_python": ">=3.9",
            "size": 43272,
            "upload_time": "2025-02-07T18:34:04",
            "upload_time_iso_8601": "2025-02-07T18:34:04.219760Z",
            "url": "https://files.pythonhosted.org/packages/1b/d9/a4b311e5fe7765c9b160043909ae158acd4deb027029b67caee2326c45a5/tinybrain-1.7.0.tar.gz",
            "yanked": false,
            "yanked_reason": null
        }
    ],
    "upload_time": "2025-02-07 18:34:04",
    "github": true,
    "gitlab": false,
    "bitbucket": false,
    "codeberg": false,
    "github_user": "seung-lab",
    "github_project": "tinybrain",
    "travis_ci": false,
    "coveralls": false,
    "github_actions": true,
    "requirements": [
        {
            "name": "numpy",
            "specs": []
        }
    ],
    "tox": true,
    "lcname": "tinybrain"
}