cloudpathlib


Namecloudpathlib JSON
Version 0.18.1 PyPI version JSON
download
home_page
Summarypathlib-style classes for cloud storage services.
upload_time2024-02-26 22:59:10
maintainer
docs_urlNone
author
requires_python>=3.7
license
keywords pathlib cloud storage s3 azure blob storage google cloud storage
VCS
bugtrack_url
requirements No requirements were recorded.
Travis-CI No Travis.
coveralls test coverage No coveralls.
            ![](https://raw.githubusercontent.com/drivendataorg/cloudpathlib/master/docs/docs/logo.svg)

<h1></h1>

[![Docs Status](https://img.shields.io/badge/docs-stable-informational)](https://cloudpathlib.drivendata.org/)
[![PyPI](https://img.shields.io/pypi/v/cloudpathlib.svg)](https://pypi.org/project/cloudpathlib/)
[![conda-forge](https://img.shields.io/conda/vn/conda-forge/cloudpathlib.svg)](https://anaconda.org/conda-forge/cloudpathlib)
[![conda-forge feedstock](https://img.shields.io/badge/conda--forge-feedstock-yellowgreen)](https://github.com/conda-forge/cloudpathlib-feedstock)
[![tests](https://github.com/drivendataorg/cloudpathlib/workflows/tests/badge.svg?branch=master)](https://github.com/drivendataorg/cloudpathlib/actions?query=workflow%3Atests+branch%3Amaster)
[![codecov](https://codecov.io/gh/drivendataorg/cloudpathlib/branch/master/graph/badge.svg)](https://codecov.io/gh/drivendataorg/cloudpathlib)

> Our goal is to be the meringue of file management libraries: the subtle sweetness of `pathlib` working in harmony with the ethereal lightness of the cloud.

A Python library with classes that mimic `pathlib.Path`'s interface for URIs from different cloud storage services.

```python
with CloudPath("s3://bucket/filename.txt").open("w+") as f:
    f.write("Send my changes to the cloud!")
```

## Why use cloudpathlib?

 - **Familiar**: If you know how to interact with `Path`, you know how to interact with `CloudPath`. All of the cloud-relevant `Path` methods are implemented.
 - **Supported clouds**: AWS S3, Google Cloud Storage, and Azure Blob Storage are implemented. FTP is on the way.
 - **Extensible**: The base classes do most of the work generically, so implementing two small classes `MyPath` and `MyClient` is all you need to add support for a new cloud storage service.
 - **Read/write support**: Reading just works. Using the `write_text`, `write_bytes` or `.open('w')` methods will all upload your changes to cloud storage without any additional file management as a developer.
 - **Seamless caching**: Files are downloaded locally only when necessary. You can also easily pass a persistent cache folder so that across processes and sessions you only re-download what is necessary.
 - **Tested**: Comprehensive test suite and code coverage.
 - **Testability**: Local filesystem implementations that can be used to easily mock cloud storage in your unit tests.


## Installation

`cloudpathlib` depends on the cloud services' SDKs (e.g., `boto3`, `google-cloud-storage`, `azure-storage-blob`) to communicate with their respective storage service. If you try to use cloud paths for a cloud service for which you don't have dependencies installed, `cloudpathlib` will error and let you know what you need to install.

To install a cloud service's SDK dependency when installing `cloudpathlib`, you need to specify it using pip's ["extras"](https://packaging.python.org/tutorials/installing-packages/#installing-setuptools-extras) specification. For example:

```bash
pip install cloudpathlib[s3,gs,azure]
```

With some shells, you may need to use quotes:

```bash
pip install "cloudpathlib[s3,gs,azure]"
```

Currently supported cloud storage services are: `azure`, `gs`, `s3`. You can also use `all` to install all available services' dependencies.

If you do not specify any extras or separately install any cloud SDKs, you will only be able to develop with the base classes for rolling your own cloud path class.

### conda

`cloudpathlib` is also available using `conda` from conda-forge. Note that to install the necessary cloud service SDK dependency, you should include the appropriate suffix in the package name. For example:

```bash
conda install cloudpathlib-s3 -c conda-forge
```

If no suffix is used, only the base classes will be usable. See the [conda-forge/cloudpathlib-feedstock](https://github.com/conda-forge/cloudpathlib-feedstock) for all installation options.

### Development version

You can get latest development version from GitHub:

```bash
pip install https://github.com/drivendataorg/cloudpathlib.git#egg=cloudpathlib[all]
```

Note that you similarly need to specify cloud service dependencies, such as `all` in the above example command.

## Quick usage

Here's an example to get the gist of using the package. By default, `cloudpathlib` authenticates with the environment variables supported by each respective cloud service SDK. For more details and advanced authentication options, see the ["Authentication"](https://cloudpathlib.drivendata.org/stable/authentication/) documentation.

```python
from cloudpathlib import CloudPath

# dispatches to S3Path based on prefix
root_dir = CloudPath("s3://drivendata-public-assets/")
root_dir
#> S3Path('s3://drivendata-public-assets/')

# there's only one file, but globbing works in nested folder
for f in root_dir.glob('**/*.txt'):
    text_data = f.read_text()
    print(f)
    print(text_data)
#> s3://drivendata-public-assets/odsc-west-2019/DATA_DICTIONARY.txt
#> Eviction Lab Data Dictionary
#>
#> Additional information in our FAQ evictionlab.org/help-faq/
#> Full methodology evictionlab.org/methods/
#>
#> ... (additional text output truncated)

# use / to join paths (and, in this case, create a new file)
new_file_copy = root_dir / "nested_dir/copy_file.txt"
new_file_copy
#> S3Path('s3://drivendata-public-assets/nested_dir/copy_file.txt')

# show things work and the file does not exist yet
new_file_copy.exists()
#> False

# writing text data to the new file in the cloud
new_file_copy.write_text(text_data)
#> 6933

# file now listed
list(root_dir.glob('**/*.txt'))
#> [S3Path('s3://drivendata-public-assets/nested_dir/copy_file.txt'),
#>  S3Path('s3://drivendata-public-assets/odsc-west-2019/DATA_DICTIONARY.txt')]

# but, we can remove it
new_file_copy.unlink()

# no longer there
list(root_dir.glob('**/*.txt'))
#> [S3Path('s3://drivendata-public-assets/odsc-west-2019/DATA_DICTIONARY.txt')]
```

## Supported methods and properties

Most methods and properties from `pathlib.Path` are supported except for the ones that don't make sense in a cloud context. There are a few additional methods or properties that relate to specific cloud services or specifically for cloud paths.

| Methods + properties   | `AzureBlobPath`   | `S3Path`   | `GSPath`   |
|:-----------------------|:------------------|:-----------|:-----------|
| `absolute`             | ✅                | ✅         | ✅         |
| `anchor`               | ✅                | ✅         | ✅         |
| `as_uri`               | ✅                | ✅         | ✅         |
| `drive`                | ✅                | ✅         | ✅         |
| `exists`               | ✅                | ✅         | ✅         |
| `glob`                 | ✅                | ✅         | ✅         |
| `is_absolute`          | ✅                | ✅         | ✅         |
| `is_dir`               | ✅                | ✅         | ✅         |
| `is_file`              | ✅                | ✅         | ✅         |
| `is_relative_to`       | ✅                | ✅         | ✅         |
| `iterdir`              | ✅                | ✅         | ✅         |
| `joinpath`             | ✅                | ✅         | ✅         |
| `match`                | ✅                | ✅         | ✅         |
| `mkdir`                | ✅                | ✅         | ✅         |
| `name`                 | ✅                | ✅         | ✅         |
| `open`                 | ✅                | ✅         | ✅         |
| `parent`               | ✅                | ✅         | ✅         |
| `parents`              | ✅                | ✅         | ✅         |
| `parts`                | ✅                | ✅         | ✅         |
| `read_bytes`           | ✅                | ✅         | ✅         |
| `read_text`            | ✅                | ✅         | ✅         |
| `relative_to`          | ✅                | ✅         | ✅         |
| `rename`               | ✅                | ✅         | ✅         |
| `replace`              | ✅                | ✅         | ✅         |
| `resolve`              | ✅                | ✅         | ✅         |
| `rglob`                | ✅                | ✅         | ✅         |
| `rmdir`                | ✅                | ✅         | ✅         |
| `samefile`             | ✅                | ✅         | ✅         |
| `stat`                 | ✅                | ✅         | ✅         |
| `stem`                 | ✅                | ✅         | ✅         |
| `suffix`               | ✅                | ✅         | ✅         |
| `suffixes`             | ✅                | ✅         | ✅         |
| `touch`                | ✅                | ✅         | ✅         |
| `unlink`               | ✅                | ✅         | ✅         |
| `with_name`            | ✅                | ✅         | ✅         |
| `with_stem`            | ✅                | ✅         | ✅         |
| `with_suffix`          | ✅                | ✅         | ✅         |
| `write_bytes`          | ✅                | ✅         | ✅         |
| `write_text`           | ✅                | ✅         | ✅         |
| `as_posix`             | ❌                | ❌         | ❌         |
| `chmod`                | ❌                | ❌         | ❌         |
| `cwd`                  | ❌                | ❌         | ❌         |
| `expanduser`           | ❌                | ❌         | ❌         |
| `group`                | ❌                | ❌         | ❌         |
| `hardlink_to`          | ❌                | ❌         | ❌         |
| `home`                 | ❌                | ❌         | ❌         |
| `is_block_device`      | ❌                | ❌         | ❌         |
| `is_char_device`       | ❌                | ❌         | ❌         |
| `is_fifo`              | ❌                | ❌         | ❌         |
| `is_mount`             | ❌                | ❌         | ❌         |
| `is_reserved`          | ❌                | ❌         | ❌         |
| `is_socket`            | ❌                | ❌         | ❌         |
| `is_symlink`           | ❌                | ❌         | ❌         |
| `lchmod`               | ❌                | ❌         | ❌         |
| `link_to`              | ❌                | ❌         | ❌         |
| `lstat`                | ❌                | ❌         | ❌         |
| `owner`                | ❌                | ❌         | ❌         |
| `readlink`             | ❌                | ❌         | ❌         |
| `root`                 | ❌                | ❌         | ❌         |
| `symlink_to`           | ❌                | ❌         | ❌         |
| `as_url`               | ✅                | ✅         | ✅         |
| `clear_cache`          | ✅                | ✅         | ✅         |
| `cloud_prefix`         | ✅                | ✅         | ✅         |
| `copy`                 | ✅                | ✅         | ✅         |
| `copytree`             | ✅                | ✅         | ✅         |
| `download_to`          | ✅                | ✅         | ✅         |
| `etag`                 | ✅                | ✅         | ✅         |
| `fspath`               | ✅                | ✅         | ✅         |
| `is_junction`          | ✅                | ✅         | ✅         |
| `is_valid_cloudpath`   | ✅                | ✅         | ✅         |
| `rmtree`               | ✅                | ✅         | ✅         |
| `upload_from`          | ✅                | ✅         | ✅         |
| `validate`             | ✅                | ✅         | ✅         |
| `walk`                 | ✅                | ✅         | ✅         |
| `with_segments`        | ✅                | ✅         | ✅         |
| `blob`                 | ✅                | ❌         | ✅         |
| `bucket`               | ❌                | ✅         | ✅         |
| `container`            | ✅                | ❌         | ❌         |
| `key`                  | ❌                | ✅         | ❌         |
| `md5`                  | ✅                | ❌         | ❌         |

----

<sup>Icon made by <a href="https://www.flaticon.com/authors/srip" title="srip">srip</a> from <a href="https://www.flaticon.com/" title="Flaticon">www.flaticon.com</a>.</sup>
<br /><sup>Sample code block generated using the [reprexpy package](https://github.com/crew102/reprexpy).</sup>


            

Raw data

            {
    "_id": null,
    "home_page": "",
    "name": "cloudpathlib",
    "maintainer": "",
    "docs_url": null,
    "requires_python": ">=3.7",
    "maintainer_email": "",
    "keywords": "pathlib,cloud storage,s3,azure blob storage,google cloud storage",
    "author": "",
    "author_email": "DrivenData <info@drivendata.org>",
    "download_url": "https://files.pythonhosted.org/packages/ea/2d/4031e0e18739281d8ed5448c549590a8382b1b801ced83222dc1793d84b2/cloudpathlib-0.18.1.tar.gz",
    "platform": null,
    "description": "![](https://raw.githubusercontent.com/drivendataorg/cloudpathlib/master/docs/docs/logo.svg)\n\n<h1></h1>\n\n[![Docs Status](https://img.shields.io/badge/docs-stable-informational)](https://cloudpathlib.drivendata.org/)\n[![PyPI](https://img.shields.io/pypi/v/cloudpathlib.svg)](https://pypi.org/project/cloudpathlib/)\n[![conda-forge](https://img.shields.io/conda/vn/conda-forge/cloudpathlib.svg)](https://anaconda.org/conda-forge/cloudpathlib)\n[![conda-forge feedstock](https://img.shields.io/badge/conda--forge-feedstock-yellowgreen)](https://github.com/conda-forge/cloudpathlib-feedstock)\n[![tests](https://github.com/drivendataorg/cloudpathlib/workflows/tests/badge.svg?branch=master)](https://github.com/drivendataorg/cloudpathlib/actions?query=workflow%3Atests+branch%3Amaster)\n[![codecov](https://codecov.io/gh/drivendataorg/cloudpathlib/branch/master/graph/badge.svg)](https://codecov.io/gh/drivendataorg/cloudpathlib)\n\n> Our goal is to be the meringue of file management libraries: the subtle sweetness of `pathlib` working in harmony with the ethereal lightness of the cloud.\n\nA Python library with classes that mimic `pathlib.Path`'s interface for URIs from different cloud storage services.\n\n```python\nwith CloudPath(\"s3://bucket/filename.txt\").open(\"w+\") as f:\n    f.write(\"Send my changes to the cloud!\")\n```\n\n## Why use cloudpathlib?\n\n - **Familiar**: If you know how to interact with `Path`, you know how to interact with `CloudPath`. All of the cloud-relevant `Path` methods are implemented.\n - **Supported clouds**: AWS S3, Google Cloud Storage, and Azure Blob Storage are implemented. FTP is on the way.\n - **Extensible**: The base classes do most of the work generically, so implementing two small classes `MyPath` and `MyClient` is all you need to add support for a new cloud storage service.\n - **Read/write support**: Reading just works. Using the `write_text`, `write_bytes` or `.open('w')` methods will all upload your changes to cloud storage without any additional file management as a developer.\n - **Seamless caching**: Files are downloaded locally only when necessary. You can also easily pass a persistent cache folder so that across processes and sessions you only re-download what is necessary.\n - **Tested**: Comprehensive test suite and code coverage.\n - **Testability**: Local filesystem implementations that can be used to easily mock cloud storage in your unit tests.\n\n\n## Installation\n\n`cloudpathlib` depends on the cloud services' SDKs (e.g., `boto3`, `google-cloud-storage`, `azure-storage-blob`) to communicate with their respective storage service. If you try to use cloud paths for a cloud service for which you don't have dependencies installed, `cloudpathlib` will error and let you know what you need to install.\n\nTo install a cloud service's SDK dependency when installing `cloudpathlib`, you need to specify it using pip's [\"extras\"](https://packaging.python.org/tutorials/installing-packages/#installing-setuptools-extras) specification. For example:\n\n```bash\npip install cloudpathlib[s3,gs,azure]\n```\n\nWith some shells, you may need to use quotes:\n\n```bash\npip install \"cloudpathlib[s3,gs,azure]\"\n```\n\nCurrently supported cloud storage services are: `azure`, `gs`, `s3`. You can also use `all` to install all available services' dependencies.\n\nIf you do not specify any extras or separately install any cloud SDKs, you will only be able to develop with the base classes for rolling your own cloud path class.\n\n### conda\n\n`cloudpathlib` is also available using `conda` from conda-forge. Note that to install the necessary cloud service SDK dependency, you should include the appropriate suffix in the package name. For example:\n\n```bash\nconda install cloudpathlib-s3 -c conda-forge\n```\n\nIf no suffix is used, only the base classes will be usable. See the [conda-forge/cloudpathlib-feedstock](https://github.com/conda-forge/cloudpathlib-feedstock) for all installation options.\n\n### Development version\n\nYou can get latest development version from GitHub:\n\n```bash\npip install https://github.com/drivendataorg/cloudpathlib.git#egg=cloudpathlib[all]\n```\n\nNote that you similarly need to specify cloud service dependencies, such as `all` in the above example command.\n\n## Quick usage\n\nHere's an example to get the gist of using the package. By default, `cloudpathlib` authenticates with the environment variables supported by each respective cloud service SDK. For more details and advanced authentication options, see the [\"Authentication\"](https://cloudpathlib.drivendata.org/stable/authentication/) documentation.\n\n```python\nfrom cloudpathlib import CloudPath\n\n# dispatches to S3Path based on prefix\nroot_dir = CloudPath(\"s3://drivendata-public-assets/\")\nroot_dir\n#> S3Path('s3://drivendata-public-assets/')\n\n# there's only one file, but globbing works in nested folder\nfor f in root_dir.glob('**/*.txt'):\n    text_data = f.read_text()\n    print(f)\n    print(text_data)\n#> s3://drivendata-public-assets/odsc-west-2019/DATA_DICTIONARY.txt\n#> Eviction Lab Data Dictionary\n#>\n#> Additional information in our FAQ evictionlab.org/help-faq/\n#> Full methodology evictionlab.org/methods/\n#>\n#> ... (additional text output truncated)\n\n# use / to join paths (and, in this case, create a new file)\nnew_file_copy = root_dir / \"nested_dir/copy_file.txt\"\nnew_file_copy\n#> S3Path('s3://drivendata-public-assets/nested_dir/copy_file.txt')\n\n# show things work and the file does not exist yet\nnew_file_copy.exists()\n#> False\n\n# writing text data to the new file in the cloud\nnew_file_copy.write_text(text_data)\n#> 6933\n\n# file now listed\nlist(root_dir.glob('**/*.txt'))\n#> [S3Path('s3://drivendata-public-assets/nested_dir/copy_file.txt'),\n#>  S3Path('s3://drivendata-public-assets/odsc-west-2019/DATA_DICTIONARY.txt')]\n\n# but, we can remove it\nnew_file_copy.unlink()\n\n# no longer there\nlist(root_dir.glob('**/*.txt'))\n#> [S3Path('s3://drivendata-public-assets/odsc-west-2019/DATA_DICTIONARY.txt')]\n```\n\n## Supported methods and properties\n\nMost methods and properties from `pathlib.Path` are supported except for the ones that don't make sense in a cloud context. There are a few additional methods or properties that relate to specific cloud services or specifically for cloud paths.\n\n| Methods + properties   | `AzureBlobPath`   | `S3Path`   | `GSPath`   |\n|:-----------------------|:------------------|:-----------|:-----------|\n| `absolute`             | \u2705                | \u2705         | \u2705         |\n| `anchor`               | \u2705                | \u2705         | \u2705         |\n| `as_uri`               | \u2705                | \u2705         | \u2705         |\n| `drive`                | \u2705                | \u2705         | \u2705         |\n| `exists`               | \u2705                | \u2705         | \u2705         |\n| `glob`                 | \u2705                | \u2705         | \u2705         |\n| `is_absolute`          | \u2705                | \u2705         | \u2705         |\n| `is_dir`               | \u2705                | \u2705         | \u2705         |\n| `is_file`              | \u2705                | \u2705         | \u2705         |\n| `is_relative_to`       | \u2705                | \u2705         | \u2705         |\n| `iterdir`              | \u2705                | \u2705         | \u2705         |\n| `joinpath`             | \u2705                | \u2705         | \u2705         |\n| `match`                | \u2705                | \u2705         | \u2705         |\n| `mkdir`                | \u2705                | \u2705         | \u2705         |\n| `name`                 | \u2705                | \u2705         | \u2705         |\n| `open`                 | \u2705                | \u2705         | \u2705         |\n| `parent`               | \u2705                | \u2705         | \u2705         |\n| `parents`              | \u2705                | \u2705         | \u2705         |\n| `parts`                | \u2705                | \u2705         | \u2705         |\n| `read_bytes`           | \u2705                | \u2705         | \u2705         |\n| `read_text`            | \u2705                | \u2705         | \u2705         |\n| `relative_to`          | \u2705                | \u2705         | \u2705         |\n| `rename`               | \u2705                | \u2705         | \u2705         |\n| `replace`              | \u2705                | \u2705         | \u2705         |\n| `resolve`              | \u2705                | \u2705         | \u2705         |\n| `rglob`                | \u2705                | \u2705         | \u2705         |\n| `rmdir`                | \u2705                | \u2705         | \u2705         |\n| `samefile`             | \u2705                | \u2705         | \u2705         |\n| `stat`                 | \u2705                | \u2705         | \u2705         |\n| `stem`                 | \u2705                | \u2705         | \u2705         |\n| `suffix`               | \u2705                | \u2705         | \u2705         |\n| `suffixes`             | \u2705                | \u2705         | \u2705         |\n| `touch`                | \u2705                | \u2705         | \u2705         |\n| `unlink`               | \u2705                | \u2705         | \u2705         |\n| `with_name`            | \u2705                | \u2705         | \u2705         |\n| `with_stem`            | \u2705                | \u2705         | \u2705         |\n| `with_suffix`          | \u2705                | \u2705         | \u2705         |\n| `write_bytes`          | \u2705                | \u2705         | \u2705         |\n| `write_text`           | \u2705                | \u2705         | \u2705         |\n| `as_posix`             | \u274c                | \u274c         | \u274c         |\n| `chmod`                | \u274c                | \u274c         | \u274c         |\n| `cwd`                  | \u274c                | \u274c         | \u274c         |\n| `expanduser`           | \u274c                | \u274c         | \u274c         |\n| `group`                | \u274c                | \u274c         | \u274c         |\n| `hardlink_to`          | \u274c                | \u274c         | \u274c         |\n| `home`                 | \u274c                | \u274c         | \u274c         |\n| `is_block_device`      | \u274c                | \u274c         | \u274c         |\n| `is_char_device`       | \u274c                | \u274c         | \u274c         |\n| `is_fifo`              | \u274c                | \u274c         | \u274c         |\n| `is_mount`             | \u274c                | \u274c         | \u274c         |\n| `is_reserved`          | \u274c                | \u274c         | \u274c         |\n| `is_socket`            | \u274c                | \u274c         | \u274c         |\n| `is_symlink`           | \u274c                | \u274c         | \u274c         |\n| `lchmod`               | \u274c                | \u274c         | \u274c         |\n| `link_to`              | \u274c                | \u274c         | \u274c         |\n| `lstat`                | \u274c                | \u274c         | \u274c         |\n| `owner`                | \u274c                | \u274c         | \u274c         |\n| `readlink`             | \u274c                | \u274c         | \u274c         |\n| `root`                 | \u274c                | \u274c         | \u274c         |\n| `symlink_to`           | \u274c                | \u274c         | \u274c         |\n| `as_url`               | \u2705                | \u2705         | \u2705         |\n| `clear_cache`          | \u2705                | \u2705         | \u2705         |\n| `cloud_prefix`         | \u2705                | \u2705         | \u2705         |\n| `copy`                 | \u2705                | \u2705         | \u2705         |\n| `copytree`             | \u2705                | \u2705         | \u2705         |\n| `download_to`          | \u2705                | \u2705         | \u2705         |\n| `etag`                 | \u2705                | \u2705         | \u2705         |\n| `fspath`               | \u2705                | \u2705         | \u2705         |\n| `is_junction`          | \u2705                | \u2705         | \u2705         |\n| `is_valid_cloudpath`   | \u2705                | \u2705         | \u2705         |\n| `rmtree`               | \u2705                | \u2705         | \u2705         |\n| `upload_from`          | \u2705                | \u2705         | \u2705         |\n| `validate`             | \u2705                | \u2705         | \u2705         |\n| `walk`                 | \u2705                | \u2705         | \u2705         |\n| `with_segments`        | \u2705                | \u2705         | \u2705         |\n| `blob`                 | \u2705                | \u274c         | \u2705         |\n| `bucket`               | \u274c                | \u2705         | \u2705         |\n| `container`            | \u2705                | \u274c         | \u274c         |\n| `key`                  | \u274c                | \u2705         | \u274c         |\n| `md5`                  | \u2705                | \u274c         | \u274c         |\n\n----\n\n<sup>Icon made by <a href=\"https://www.flaticon.com/authors/srip\" title=\"srip\">srip</a> from <a href=\"https://www.flaticon.com/\" title=\"Flaticon\">www.flaticon.com</a>.</sup>\n<br /><sup>Sample code block generated using the [reprexpy package](https://github.com/crew102/reprexpy).</sup>\n\n",
    "bugtrack_url": null,
    "license": "",
    "summary": "pathlib-style classes for cloud storage services.",
    "version": "0.18.1",
    "project_urls": {
        "Bug Tracker": "https://github.com/drivendataorg/cloudpathlib/issues",
        "Changelog": "https://cloudpathlib.drivendata.org/stable/changelog/",
        "Documentation": "https://cloudpathlib.drivendata.org/",
        "Repository": "https://github.com/drivendataorg/cloudpathlib"
    },
    "split_keywords": [
        "pathlib",
        "cloud storage",
        "s3",
        "azure blob storage",
        "google cloud storage"
    ],
    "urls": [
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "bcbad8f2c0151585519759135550574385dd7a223abbc6b6c06dab7ada565773",
                "md5": "9ff28c61753ba5e5bf9d23eca9ba8ed8",
                "sha256": "20efd5d772c75df91bb2ac52e053be53fd9000f5e9755fd92375a2a9fe6005e0"
            },
            "downloads": -1,
            "filename": "cloudpathlib-0.18.1-py3-none-any.whl",
            "has_sig": false,
            "md5_digest": "9ff28c61753ba5e5bf9d23eca9ba8ed8",
            "packagetype": "bdist_wheel",
            "python_version": "py3",
            "requires_python": ">=3.7",
            "size": 47335,
            "upload_time": "2024-02-26T22:59:08",
            "upload_time_iso_8601": "2024-02-26T22:59:08.833918Z",
            "url": "https://files.pythonhosted.org/packages/bc/ba/d8f2c0151585519759135550574385dd7a223abbc6b6c06dab7ada565773/cloudpathlib-0.18.1-py3-none-any.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "ea2d4031e0e18739281d8ed5448c549590a8382b1b801ced83222dc1793d84b2",
                "md5": "8812d456eb2163aa9fe09e98372987e1",
                "sha256": "ffd22f324bfbf9c3f2bc1bec6e8372cb372a0feef17c7f2b48030cd6810ea859"
            },
            "downloads": -1,
            "filename": "cloudpathlib-0.18.1.tar.gz",
            "has_sig": false,
            "md5_digest": "8812d456eb2163aa9fe09e98372987e1",
            "packagetype": "sdist",
            "python_version": "source",
            "requires_python": ">=3.7",
            "size": 39871,
            "upload_time": "2024-02-26T22:59:10",
            "upload_time_iso_8601": "2024-02-26T22:59:10.884225Z",
            "url": "https://files.pythonhosted.org/packages/ea/2d/4031e0e18739281d8ed5448c549590a8382b1b801ced83222dc1793d84b2/cloudpathlib-0.18.1.tar.gz",
            "yanked": false,
            "yanked_reason": null
        }
    ],
    "upload_time": "2024-02-26 22:59:10",
    "github": true,
    "gitlab": false,
    "bitbucket": false,
    "codeberg": false,
    "github_user": "drivendataorg",
    "github_project": "cloudpathlib",
    "travis_ci": false,
    "coveralls": false,
    "github_actions": true,
    "lcname": "cloudpathlib"
}
        
Elapsed time: 0.23489s