v3io


Namev3io JSON
Version 0.6.9 PyPI version JSON
download
home_pagehttps://github.com/v3io/v3io-py
SummarySDK for v3io
upload_time2024-08-25 10:42:18
maintainerNone
docs_urlNone
authorEran Duchan
requires_pythonNone
licenseMIT
keywords
VCS
bugtrack_url
requirements requests future orjson
Travis-CI No Travis.
coveralls test coverage No coveralls.
            # Python SDK for V3IO

Python (2.7 and 3.5+) client for the Iguazio Data Science Platform (the "platform"). Designed to allow fast access to the data layer and basic access to the control layer. This library can be used in Nuclio functions, Jupyter notebooks, local Python IDEs; anywhere with a Python interpreter and access to a platform. 

# Installing
Simply get with `pip` (locking to a specific version is recommended, as always):
```
pip install v3io
```

# Dataplane client
With the dataplane client you can manipulate data in the platform's multi-model data layer, including:
* Objects
* Key-values (NoSQL)
* Streams
* Containers

Under the hood, the client connects through the platform's web API (https://www.iguazio.com/docs/latest-release/data-layer/reference/web-apis/) and wraps each low level API with an interface. Calls are blocking, but you can use the batching interface to send multiple requests in parallel for greater performance. 

## Creating a client
Create a dataplane client, passing in the web API endpoint and your access key:

```python
import v3io.dataplane


v3io_client = v3io.dataplane.Client(endpoint='https://v3io-webapi:8081', access_key='some_access_key')
```

> Note: In some environments (like Jupyter notebooks) you do not need to pass the endpoint and access key, as they will automatically be inferred from the environment

### Making requests
The client supports a handful of low level APIs, each receiving different arguments but all returning a `Response` object. A `Response` object holds 4 fields:
* `status_code`: The HTTP status code returned
* `output`: An object containing the parsed response (each API returns a different object)
* `headers` and `body`: The raw headers and body of the response

You would normally only access the `output` field unless an API was called that returns raw data like `object.get` (in which case `body` holds the response). Consult the reference for each API call to see how to handle its `Response` object. In the example below, we perform a simple request to get the containers available in our tenant, print the returned status code and containers:

```python
# list the contents of a container
response = v3io_client.container.list('users', '/')

# print the status code. outputs:
# 
#  Status code: 200
#
print(f'Status code: {response.status_code}')
```

We can also get help information about the parameters this API call receives:
```python
help(v3io_client.container.list)
```

### Handling errors
By default, making a request will raise an exception if any non-200 status code is returned. We can override this default behavior in two ways. 

The first is to simply never raise an exception and handle the status manually:
```python
# list the contents of a container and never raise an exception
response = v3io_client.container.list('users', '/', raise_for_status=v3io.dataplane.RaiseForStatus.never)

# do anything we want with the status code
# some_logic(response.status_code)
```

The second is to indicate which status codes are acceptable:
```python
# list the contents of a container and raise if the status code is not 200 or 204
response = v3io_client.container.list('users', '/', raise_for_status=[200, 204])
```

### Creating batches
To get the highest possible throughput, we can send many requests towards the data layer and wait for all the responses to arrive (rather than send each request and wait for the response). The SDK supports this through batching. Any API call can be made through the client's built in `batch` object. The API call receives the exact same arguments it would normally receive (except for `raise_for_status`), and does not block until the response arrives. To wait for all pending responses, call `wait()` on the `batch` object:

```python
# do 16 writes in parallel
for idx in range(16):

    # returns immediately
    v3io_client.batch.object.put(container='bigdata',
                                 path=f'/object{idx}',
                                 body=f'object-{idx}')

# wait for all writes to complete
v3io_client.batch.wait()
```

The looped `object.put` interface above will send 16 `put object` requests to the data layer in parallel. When `wait` is called, it will block until either all responses arrive (in which case it will return a `Responses` object, containing the `responses` of each call) or an error occurs - in which case an exception is thrown. You can pass `raise_for_status` to `wait`, and it behaves as explained above.

> Note: The `batch` object is stateful, so you can only create one batch at a time. However, you can create multiple parallel batches yourself through the client's `create_batch()` interface

## Examples

### Accessing objects

Put data in an object, get it back and then delete the object:

```python
# put contents to some object
v3io_client.object.put(container='users',
                       path='/my-object',
                       body='hello, there')

# get the object
response = v3io_client.object.get(container='users', path='/my-object')

# print the contents. outputs:
#
#   hello, there
#
print(response.body.decode('utf-8'))

# delete the object
v3io_client.object.delete(container='users', path='/my-object')
```

For a detailed example, refer to the [v3io-objects notebook](examples/v3io-objects.ipynb). 

### Accessing key-values (NoSQL)
Create a table, update a record and run a query.

```python
items = {
    'bob': {'age': 42, 'feature': 'mustache'},
    'linda': {'age': 41, 'feature': 'singing'},
    'louise': {'age': 9, 'feature': 'bunny ears'},
}

# add the records to the table
for item_key, item_attributes in items.items():
    v3io_client.kv.put(container='users', table_path='/bobs-burgers', key=item_key, attributes=item_attributes)

# adds two fields (height, quip) to the louise record
v3io_client.kv.update(container='users',
                      table_path='/bobs-burgers',
                      key='louise',   
                      attributes={
                          'height': 130,
                          'quip': 'i can smell fear on you'
                      })

# get a record by key, specifying specific arguments
response = v3io_client.kv.get(container='users',
                              table_path='/bobs-burgers',
                              key='louise',
                              attribute_names=['__size', 'age', 'quip', 'height'])


# print the item from the response. outputs:
#
#   {'__size': 0.0, 'quip': 'i can smell fear on you', 'height': 130.0}
#
print(response.output.item)

# create a query, and use an items cursor to iterate the results
items_cursor = v3io_client.kv.new_cursor(container='users',
                                         table_path='/bobs-burgers',
                                         attribute_names=['age', 'feature'],
                                         filter_expression='age > 15')

# print the output
for item in items_cursor.all():
    print(item)
```

For a detailed example, refer to the [examples/v3io-kv notebook](examples/v3io-kv.ipynb). 

### Accessing streams
Creates a stream with several partitions, writes records to it, reads the records and deletes the stream:

```python
# create a stream w/8 shards
v3io_client.stream.create(container='users',
                          stream_path='/my-test-stream',
                          shard_count=8)

# write 4 records - 3 with explicitly specifying the shard and 1 using hashing
records = [
    {'shard_id': 1, 'data': 'first shard record #1'},
    {'shard_id': 1, 'data': 'first shard record #2'},
    {'shard_id': 2, 'data': 'second shard record #1'},
    {'data': 'some shard record #1'}
]

v3io_client.stream.put_records(container='users',
                               stream_path='/my-test-stream',
                               records=records)

# seek to the beginning of the shard of #1 so we know where to read from 
response = v3io_client.stream.seek(container='users',
                                   stream_path='/my-test-stream',
                                   shard_id=1,
                                   seek_type='EARLIEST')

# get records from the shard (should receive 2)
response = v3io_client.stream.get_records(container='users',
                                          stream_path='/my-test-stream',
                                          shard_id=1,
                                          location=response.output.location)

# print the records. outputs:
#
#   first shard record #1
#   first shard record #2
#
for record in response.output.records:
    print(record.data.decode('utf-8'))

# delete the stream
v3io_client.stream.delete(container='users', stream_path='/my-test-stream')
```

For a detailed example, refer to the [v3io-streams notebook](examples/v3io-streams.ipynb). 

## Support for asyncio (experimental)

All synchronous APIs are available as `async` interfaces through the `aio` module. The differences between the sync and async API is as follows:
1. You must initialize a different client (`v3io.aio.dataplane.Client`) from `v3io.aio.dataplane`
2. All interfaces should be called with `await`
3. `v3io.aio.dataplane.RaiseForStatus.never` should be used over `v3io.dataplane.RaiseForStatus.never` (although they are the same)
4. The batching functionality doesn't exist, as you can achieve the same through standard asyncio practices

Note: For the time being, `aiohttp` must be provided externally - it is not a v3io-py dependency. This will be fixed in future versions.

```python
import v3io.aio.dataplane

v3io_client = v3io.aio.dataplane.Client(endpoint='https://v3io-webapi:8081', access_key='some_access_key')

# put contents to some object
await v3io_client.object.put(container='users',
                             path='/my-object',
                             body='hello, there')

# get the object
response = await v3io_client.object.get(container='users', path='/my-object')

# print the contents. outputs:
#
#   hello, there
#
print(response.body.decode('utf-8'))

# delete the object
await v3io_client.object.delete(container='users', path='/my-object')
```


## Support for unit tests (experimental)

To facilitate unit tests, v3io-py can pass along all requests to the user rather than communicate over the network with the dataplane layer. This is done by passing the `Client` object a `verifier` transport. This transport receives a list of callables that will be called in the order of requests the `Client` generates. The user can then verify the content of the request and either return a response or raise an exception. With this mechanism, users can test snippets of their code for various positive and negative behvior.

In the below example, two requests are sent and verified. Using `MagicMock` for outputs/responses is recommended:

```python
def test_verifier_transport(self):
    container_name = 'some_container'

    #
    # Register a set of verifiers
    #

    def _verify_object_get(request):

        # verify some stuff from the request
        self.assertEqual(request.container, container_name)
        self.assertEqual(request.path, os.path.join(os.sep, container_name, 'some/path'))

        # return a mocked response
        return unittest.mock.MagicMock(status_code=200,
                                       body='some body')

    def _verify_kv_get(request):

        # verify some stuff from the request
        self.assertEqual(request.container, container_name)
        self.assertEqual(request.path, os.path.join(os.sep, container_name, 'some/table/path/some_item_key'))

        # prepare and output mock
        output = unittest.mock.MagicMock(item={
            'some_key': 'some_value'
        })

        # prepare a response mock
        return unittest.mock.MagicMock(output=output)

    # create a verifier transport. pass it a set of request verifiers
    verifier_transport = v3io.dataplane.transport.verifier.Transport(request_verifiers=[
        _verify_object_get,
        _verify_kv_get,
    ])

    #
    # Run some flow. This can be a call to the actual code you want to test. By specfiying all
    # the verifying behavior beforehand, the actual code under test remains untouched
    #

    # create a client with a verifier transport
    client = v3io.dataplane.Client(transport_kind=verifier_transport)

    # do an object get
    response = client.object.get(container='some_container', path='some/path')

    # verify that we got some body from the verifier
    self.assertEqual(response.body, 'some body')
    self.assertEqual(response.status_code, 200)

    # do an item get
    response = client.kv.get(container=container_name,
                             table_path='some/table/path',
                             key='some_item_key')

    # verify that we got a proper response
    self.assertEqual(response.output.item['some_key'], 'some_value')
```

### Custom transports

If the `verifier` transport is inadequate for your use case, a custom transport can be provided. It must conform to the following minimal interface:

```python
class Transport(v3io.dataplane.transport.abstract.Transport):

    def __init__(self):
        super().__init__(None, '', 0, None, 'DEBUG')

    # receives a request and returns a response
    def wait_response(self, request, raise_for_status=None):
        pass
```

            

Raw data

            {
    "_id": null,
    "home_page": "https://github.com/v3io/v3io-py",
    "name": "v3io",
    "maintainer": null,
    "docs_url": null,
    "requires_python": null,
    "maintainer_email": null,
    "keywords": null,
    "author": "Eran Duchan",
    "author_email": "erand@iguazio.com",
    "download_url": "https://files.pythonhosted.org/packages/d1/55/df5b38bb21d893af1d1d0d076604d667c346a7b8f6203b3dbc0e273b4c0e/v3io-0.6.9.tar.gz",
    "platform": null,
    "description": "# Python SDK for V3IO\n\nPython (2.7 and 3.5+) client for the Iguazio Data Science Platform (the \"platform\"). Designed to allow fast access to the data layer and basic access to the control layer. This library can be used in Nuclio functions, Jupyter notebooks, local Python IDEs; anywhere with a Python interpreter and access to a platform. \n\n# Installing\nSimply get with `pip` (locking to a specific version is recommended, as always):\n```\npip install v3io\n```\n\n# Dataplane client\nWith the dataplane client you can manipulate data in the platform's multi-model data layer, including:\n* Objects\n* Key-values (NoSQL)\n* Streams\n* Containers\n\nUnder the hood, the client connects through the platform's web API (https://www.iguazio.com/docs/latest-release/data-layer/reference/web-apis/) and wraps each low level API with an interface. Calls are blocking, but you can use the batching interface to send multiple requests in parallel for greater performance. \n\n## Creating a client\nCreate a dataplane client, passing in the web API endpoint and your access key:\n\n```python\nimport v3io.dataplane\n\n\nv3io_client = v3io.dataplane.Client(endpoint='https://v3io-webapi:8081', access_key='some_access_key')\n```\n\n> Note: In some environments (like Jupyter notebooks) you do not need to pass the endpoint and access key, as they will automatically be inferred from the environment\n\n### Making requests\nThe client supports a handful of low level APIs, each receiving different arguments but all returning a `Response` object. A `Response` object holds 4 fields:\n* `status_code`: The HTTP status code returned\n* `output`: An object containing the parsed response (each API returns a different object)\n* `headers` and `body`: The raw headers and body of the response\n\nYou would normally only access the `output` field unless an API was called that returns raw data like `object.get` (in which case `body` holds the response). Consult the reference for each API call to see how to handle its `Response` object. In the example below, we perform a simple request to get the containers available in our tenant, print the returned status code and containers:\n\n```python\n# list the contents of a container\nresponse = v3io_client.container.list('users', '/')\n\n# print the status code. outputs:\n# \n#  Status code: 200\n#\nprint(f'Status code: {response.status_code}')\n```\n\nWe can also get help information about the parameters this API call receives:\n```python\nhelp(v3io_client.container.list)\n```\n\n### Handling errors\nBy default, making a request will raise an exception if any non-200 status code is returned. We can override this default behavior in two ways. \n\nThe first is to simply never raise an exception and handle the status manually:\n```python\n# list the contents of a container and never raise an exception\nresponse = v3io_client.container.list('users', '/', raise_for_status=v3io.dataplane.RaiseForStatus.never)\n\n# do anything we want with the status code\n# some_logic(response.status_code)\n```\n\nThe second is to indicate which status codes are acceptable:\n```python\n# list the contents of a container and raise if the status code is not 200 or 204\nresponse = v3io_client.container.list('users', '/', raise_for_status=[200, 204])\n```\n\n### Creating batches\nTo get the highest possible throughput, we can send many requests towards the data layer and wait for all the responses to arrive (rather than send each request and wait for the response). The SDK supports this through batching. Any API call can be made through the client's built in `batch` object. The API call receives the exact same arguments it would normally receive (except for `raise_for_status`), and does not block until the response arrives. To wait for all pending responses, call `wait()` on the `batch` object:\n\n```python\n# do 16 writes in parallel\nfor idx in range(16):\n\n    # returns immediately\n    v3io_client.batch.object.put(container='bigdata',\n                                 path=f'/object{idx}',\n                                 body=f'object-{idx}')\n\n# wait for all writes to complete\nv3io_client.batch.wait()\n```\n\nThe looped `object.put` interface above will send 16 `put object` requests to the data layer in parallel. When `wait` is called, it will block until either all responses arrive (in which case it will return a `Responses` object, containing the `responses` of each call) or an error occurs - in which case an exception is thrown. You can pass `raise_for_status` to `wait`, and it behaves as explained above.\n\n> Note: The `batch` object is stateful, so you can only create one batch at a time. However, you can create multiple parallel batches yourself through the client's `create_batch()` interface\n\n## Examples\n\n### Accessing objects\n\nPut data in an object, get it back and then delete the object:\n\n```python\n# put contents to some object\nv3io_client.object.put(container='users',\n                       path='/my-object',\n                       body='hello, there')\n\n# get the object\nresponse = v3io_client.object.get(container='users', path='/my-object')\n\n# print the contents. outputs:\n#\n#   hello, there\n#\nprint(response.body.decode('utf-8'))\n\n# delete the object\nv3io_client.object.delete(container='users', path='/my-object')\n```\n\nFor a detailed example, refer to the [v3io-objects notebook](examples/v3io-objects.ipynb). \n\n### Accessing key-values (NoSQL)\nCreate a table, update a record and run a query.\n\n```python\nitems = {\n    'bob': {'age': 42, 'feature': 'mustache'},\n    'linda': {'age': 41, 'feature': 'singing'},\n    'louise': {'age': 9, 'feature': 'bunny ears'},\n}\n\n# add the records to the table\nfor item_key, item_attributes in items.items():\n    v3io_client.kv.put(container='users', table_path='/bobs-burgers', key=item_key, attributes=item_attributes)\n\n# adds two fields (height, quip) to the louise record\nv3io_client.kv.update(container='users',\n                      table_path='/bobs-burgers',\n                      key='louise',   \n                      attributes={\n                          'height': 130,\n                          'quip': 'i can smell fear on you'\n                      })\n\n# get a record by key, specifying specific arguments\nresponse = v3io_client.kv.get(container='users',\n                              table_path='/bobs-burgers',\n                              key='louise',\n                              attribute_names=['__size', 'age', 'quip', 'height'])\n\n\n# print the item from the response. outputs:\n#\n#   {'__size': 0.0, 'quip': 'i can smell fear on you', 'height': 130.0}\n#\nprint(response.output.item)\n\n# create a query, and use an items cursor to iterate the results\nitems_cursor = v3io_client.kv.new_cursor(container='users',\n                                         table_path='/bobs-burgers',\n                                         attribute_names=['age', 'feature'],\n                                         filter_expression='age > 15')\n\n# print the output\nfor item in items_cursor.all():\n    print(item)\n```\n\nFor a detailed example, refer to the [examples/v3io-kv notebook](examples/v3io-kv.ipynb). \n\n### Accessing streams\nCreates a stream with several partitions, writes records to it, reads the records and deletes the stream:\n\n```python\n# create a stream w/8 shards\nv3io_client.stream.create(container='users',\n                          stream_path='/my-test-stream',\n                          shard_count=8)\n\n# write 4 records - 3 with explicitly specifying the shard and 1 using hashing\nrecords = [\n    {'shard_id': 1, 'data': 'first shard record #1'},\n    {'shard_id': 1, 'data': 'first shard record #2'},\n    {'shard_id': 2, 'data': 'second shard record #1'},\n    {'data': 'some shard record #1'}\n]\n\nv3io_client.stream.put_records(container='users',\n                               stream_path='/my-test-stream',\n                               records=records)\n\n# seek to the beginning of the shard of #1 so we know where to read from \nresponse = v3io_client.stream.seek(container='users',\n                                   stream_path='/my-test-stream',\n                                   shard_id=1,\n                                   seek_type='EARLIEST')\n\n# get records from the shard (should receive 2)\nresponse = v3io_client.stream.get_records(container='users',\n                                          stream_path='/my-test-stream',\n                                          shard_id=1,\n                                          location=response.output.location)\n\n# print the records. outputs:\n#\n#   first shard record #1\n#   first shard record #2\n#\nfor record in response.output.records:\n    print(record.data.decode('utf-8'))\n\n# delete the stream\nv3io_client.stream.delete(container='users', stream_path='/my-test-stream')\n```\n\nFor a detailed example, refer to the [v3io-streams notebook](examples/v3io-streams.ipynb). \n\n## Support for asyncio (experimental)\n\nAll synchronous APIs are available as `async` interfaces through the `aio` module. The differences between the sync and async API is as follows:\n1. You must initialize a different client (`v3io.aio.dataplane.Client`) from `v3io.aio.dataplane`\n2. All interfaces should be called with `await`\n3. `v3io.aio.dataplane.RaiseForStatus.never` should be used over `v3io.dataplane.RaiseForStatus.never` (although they are the same)\n4. The batching functionality doesn't exist, as you can achieve the same through standard asyncio practices\n\nNote: For the time being, `aiohttp` must be provided externally - it is not a v3io-py dependency. This will be fixed in future versions.\n\n```python\nimport v3io.aio.dataplane\n\nv3io_client = v3io.aio.dataplane.Client(endpoint='https://v3io-webapi:8081', access_key='some_access_key')\n\n# put contents to some object\nawait v3io_client.object.put(container='users',\n                             path='/my-object',\n                             body='hello, there')\n\n# get the object\nresponse = await v3io_client.object.get(container='users', path='/my-object')\n\n# print the contents. outputs:\n#\n#   hello, there\n#\nprint(response.body.decode('utf-8'))\n\n# delete the object\nawait v3io_client.object.delete(container='users', path='/my-object')\n```\n\n\n## Support for unit tests (experimental)\n\nTo facilitate unit tests, v3io-py can pass along all requests to the user rather than communicate over the network with the dataplane layer. This is done by passing the `Client` object a `verifier` transport. This transport receives a list of callables that will be called in the order of requests the `Client` generates. The user can then verify the content of the request and either return a response or raise an exception. With this mechanism, users can test snippets of their code for various positive and negative behvior.\n\nIn the below example, two requests are sent and verified. Using `MagicMock` for outputs/responses is recommended:\n\n```python\ndef test_verifier_transport(self):\n    container_name = 'some_container'\n\n    #\n    # Register a set of verifiers\n    #\n\n    def _verify_object_get(request):\n\n        # verify some stuff from the request\n        self.assertEqual(request.container, container_name)\n        self.assertEqual(request.path, os.path.join(os.sep, container_name, 'some/path'))\n\n        # return a mocked response\n        return unittest.mock.MagicMock(status_code=200,\n                                       body='some body')\n\n    def _verify_kv_get(request):\n\n        # verify some stuff from the request\n        self.assertEqual(request.container, container_name)\n        self.assertEqual(request.path, os.path.join(os.sep, container_name, 'some/table/path/some_item_key'))\n\n        # prepare and output mock\n        output = unittest.mock.MagicMock(item={\n            'some_key': 'some_value'\n        })\n\n        # prepare a response mock\n        return unittest.mock.MagicMock(output=output)\n\n    # create a verifier transport. pass it a set of request verifiers\n    verifier_transport = v3io.dataplane.transport.verifier.Transport(request_verifiers=[\n        _verify_object_get,\n        _verify_kv_get,\n    ])\n\n    #\n    # Run some flow. This can be a call to the actual code you want to test. By specfiying all\n    # the verifying behavior beforehand, the actual code under test remains untouched\n    #\n\n    # create a client with a verifier transport\n    client = v3io.dataplane.Client(transport_kind=verifier_transport)\n\n    # do an object get\n    response = client.object.get(container='some_container', path='some/path')\n\n    # verify that we got some body from the verifier\n    self.assertEqual(response.body, 'some body')\n    self.assertEqual(response.status_code, 200)\n\n    # do an item get\n    response = client.kv.get(container=container_name,\n                             table_path='some/table/path',\n                             key='some_item_key')\n\n    # verify that we got a proper response\n    self.assertEqual(response.output.item['some_key'], 'some_value')\n```\n\n### Custom transports\n\nIf the `verifier` transport is inadequate for your use case, a custom transport can be provided. It must conform to the following minimal interface:\n\n```python\nclass Transport(v3io.dataplane.transport.abstract.Transport):\n\n    def __init__(self):\n        super().__init__(None, '', 0, None, 'DEBUG')\n\n    # receives a request and returns a response\n    def wait_response(self, request, raise_for_status=None):\n        pass\n```\n",
    "bugtrack_url": null,
    "license": "MIT",
    "summary": "SDK for v3io",
    "version": "0.6.9",
    "project_urls": {
        "Homepage": "https://github.com/v3io/v3io-py"
    },
    "split_keywords": [],
    "urls": [
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "96e4b87206ca19593b8136e33c3e2cc02251285f13c358397012f89221b286ba",
                "md5": "2a601e8b8360afc26d209ed540e152d7",
                "sha256": "6e256fba5d9f312fc53791bf596a0eca4a1b07feb03918d081c1377df9fdcbf3"
            },
            "downloads": -1,
            "filename": "v3io-0.6.9-py3-none-any.whl",
            "has_sig": false,
            "md5_digest": "2a601e8b8360afc26d209ed540e152d7",
            "packagetype": "bdist_wheel",
            "python_version": "py3",
            "requires_python": null,
            "size": 64820,
            "upload_time": "2024-08-25T10:42:17",
            "upload_time_iso_8601": "2024-08-25T10:42:17.121681Z",
            "url": "https://files.pythonhosted.org/packages/96/e4/b87206ca19593b8136e33c3e2cc02251285f13c358397012f89221b286ba/v3io-0.6.9-py3-none-any.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "d155df5b38bb21d893af1d1d0d076604d667c346a7b8f6203b3dbc0e273b4c0e",
                "md5": "f0c2348f81cc6a584aedf39e6264aaf6",
                "sha256": "f368172249b107c702b13521da06a0b6698e2c7a751c9bd7aaa92b8e54c54e10"
            },
            "downloads": -1,
            "filename": "v3io-0.6.9.tar.gz",
            "has_sig": false,
            "md5_digest": "f0c2348f81cc6a584aedf39e6264aaf6",
            "packagetype": "sdist",
            "python_version": "source",
            "requires_python": null,
            "size": 55764,
            "upload_time": "2024-08-25T10:42:18",
            "upload_time_iso_8601": "2024-08-25T10:42:18.586517Z",
            "url": "https://files.pythonhosted.org/packages/d1/55/df5b38bb21d893af1d1d0d076604d667c346a7b8f6203b3dbc0e273b4c0e/v3io-0.6.9.tar.gz",
            "yanked": false,
            "yanked_reason": null
        }
    ],
    "upload_time": "2024-08-25 10:42:18",
    "github": true,
    "gitlab": false,
    "bitbucket": false,
    "codeberg": false,
    "github_user": "v3io",
    "github_project": "v3io-py",
    "travis_ci": false,
    "coveralls": false,
    "github_actions": true,
    "requirements": [
        {
            "name": "requests",
            "specs": [
                [
                    ">=",
                    "2.19.1"
                ]
            ]
        },
        {
            "name": "future",
            "specs": [
                [
                    ">=",
                    "0.18.2"
                ]
            ]
        },
        {
            "name": "orjson",
            "specs": [
                [
                    ">=",
                    "3.9.15"
                ],
                [
                    "<",
                    "4"
                ]
            ]
        }
    ],
    "lcname": "v3io"
}
        
Elapsed time: 0.52890s