atproto


Nameatproto JSON
Version 0.0.56 PyPI version JSON
download
home_pagehttps://github.com/MarshalX/atproto
SummaryThe AT Protocol SDK
upload_time2024-12-05 17:47:46
maintainerNone
docs_urlNone
authorIlya (Marshal)
requires_python<3.14,>=3.8
licenseMIT
keywords library sdk codegen xrpc xrpc-client atprotocol atproto lexicon parser schema bluesky bluesky-api at uri atp nsid did cid
VCS
bugtrack_url
requirements No requirements were recorded.
Travis-CI No Travis.
coveralls test coverage
            <p align="center">
    <a href="https://github.com/MarshalX/atproto">
        <img alt="Logo of atproto SDK for Python by Midjourney'" src="https://github.com/MarshalX/atproto/raw/main/.github/images/logo.png">
    </a>
    <br>
    <b>Autogenerated from lexicons, well type hinted, documented, sync and async SDK for Python</b>
    <br>
    <a href="https://github.com/MarshalX/atproto/tree/main/examples">
        Examples
    </a>
    •
    <a href="https://atproto.blue">
        Documentation
    </a>
    •
    <a href="https://discord.gg/PCyVJXU9jN">
        Discord Bluesky API
    </a>
</p>

## The AT Protocol SDK

> ⚠️ Under construction. Until the 1.0.0 release compatibility between versions is not guaranteed. 

Code snippet:

```python
from atproto import Client, client_utils


def main():
    client = Client()
    profile = client.login('my-handle', 'my-password')
    print('Welcome,', profile.display_name)
    
    text = client_utils.TextBuilder().text('Hello World from ').link('Python SDK', 'https://atproto.blue')
    post = client.send_post(text)
    client.like(post.uri, post.cid)


if __name__ == '__main__':
    main()

```

<details>
  <summary>Code snippet of async version</summary>

```python
import asyncio

from atproto import AsyncClient, client_utils


async def main():
    client = AsyncClient()
    profile = await client.login('my-handle', 'my-password')
    print('Welcome,', profile.display_name)

    text = client_utils.TextBuilder().text('Hello World from ').link('Python SDK', 'https://atproto.blue')
    post = await client.send_post(text)
    await client.like(post.uri, post.cid)

    
if __name__ == '__main__':
    # use run() for a higher Python version
    asyncio.get_event_loop().run_until_complete(main())

```
</details>

💬 [Direct Messages (Chats)](https://atproto.blue/en/latest/dm.html)

🍿 [Example project with custom feed generator](https://github.com/MarshalX/bluesky-feed-generator)

🔥 [Firehose data streaming is available](https://atproto.blue/en/latest/atproto_firehose/index.html)

🌐 [Identity resolvers for DID and Handle](https://atproto.blue/en/latest/atproto_identity/index.html)

### Introduction

This SDK attempts to implement everything that provides ATProto. There is support for Lexicon Schemes, XRPC clients, Firehose, Identity, DID keys, signatures, and more. All models, queries, and procedures are generated automatically. The main focus is on the lexicons of atproto.com and bsky.app, but it doesn't have a vendor lock on it. Feel free to use the code generator for your own lexicon schemes. SDK also provides utilities to work with CID, NSID, AT URI Schemes, DAG-CBOR, CAR files, DID Documents and more.

### Requirements

- Python 3.8 or higher.

### Installing

``` bash
pip install atproto
```

### Quick start

First of all, you need to create the instance of the XRPC Client. To do so, you have two major options: asynchronous, and synchronous. The difference only in import and how you call the methods. If you are not familiar with async, use sync instead.

For sync:

```python
from atproto import Client

client = Client()
# By default, it uses the server of bsky.app. To change this behavior, pass the base api URL to constructor
# Client('https://example.com')
```

For async:

```python
from atproto import AsyncClient

client = AsyncClient()
# By default, it uses the server of bsky.app. To change this behavior, pass the base api URL to constructor
# AsyncClient('https://example.com')
```

In the snippets below, only the sync version will be presented.

Right after the creation of the Client instance, you probably want to access the full API and perform actions by profile. To achieve this, you should log in to the network using your handle and password. The password could be app-specific.

```python
from atproto import Client

client = Client()
client.login('my-username', 'my-password')
```

You are awesome! Now you feel to pick any high-level method that you want and perform it!

Code to send post:

```python
from atproto import Client

client = Client()
client.login('my-username', 'my-password')
client.send_post(text='Hello World!')
```

Useful links to continue:

- [List of all methods with documentation](https://atproto.readthedocs.io/en/latest/atproto_client/index.html).
- [Examples of using the methods](https://github.com/MarshalX/atproto/tree/main/examples).

### SDK structure

The SDK is built upon the following components:

| Package            | Description                                                                 |
|--------------------|-----------------------------------------------------------------------------|
| `atproto`          | Import shortcuts to other packages.                                         |
| `atproto_cli`      | CLI tool to generate code.                                                  |
| `atproto_client`   | XRPC Client, data models, and utils like rich text helper.                  |
| `atproto_codegen`  | Code generator of models, clients, and namespaces.                          |
| `atproto_core`     | Tools to work with NSID, AT URI Schemes, CID, CAR files, and DID Documents. |
| `atproto_crypto`   | Crypto utils like multibase, signature verification, work with DID keys.    |
| `atproto_firehose` | Firehose (data streaming) client and models.                                |
| `atproto_identity` | Identity resolvers for DID, Handle, AT Protocol data, signing keys.         |
| `atproto_lexicon`  | Lexicon parser.                                                             |
| `atproto_server`   | Server-side utils like JWT.                                                 |

I highly recommend you to use the `atproto` package to import everything that you need. 
It contains shortcuts to all other packages.

### Documentation

The documentation is live at [atproto.blue](https://atproto.blue/).

### Getting help

You can get help in several ways:
- Report bugs, request new features by [creating an issue](https://github.com/MarshalX/atproto/issues/new).
- Ask questions by [starting a discussion](https://github.com/MarshalX/atproto/discussions/new).
- Ask questions in [Discord server](https://discord.gg/PCyVJXU9jN).

### Advanced usage

I'll be honest. The high-level Client that was shown in the "Quick Start" section is not a real ATProto API. This is syntax sugar built upon the real XRPC methods! The high-level methods are not cover the full need of developers. To be able to do anything that you want, you should know to work with low-level API. Let's dive into it!

The basics:
- Namespaces – classes that group sub-namespaces and the XRPC queries and procedures. Built upon NSID ATProto semantic.
- Model – dataclasses for input, output, and params of the methods from namespaces. Models describe Record and all other types in the Lexicon Schemes.

#### Namespaces

The client contains references to the root of all namespaces. It's `com` and `app` for now.

```python
from atproto import Client

Client().com
Client().app
```

To dive deeper, you can navigate using hints from your IDE. Thanks to well-type hinted SDK, it's much easier.

```python
from atproto import Client

Client().com.atproto.server.create_session(...)
Client().com.atproto.sync.get_blob(...)
Client().app.bsky.feed.get_likes(...)
Client().app.bsky.graph.get_follows(...)
```

The endpoint of the path is always the method that you want to call. The method presents a query or procedure in XRPC. You should not care about it much. The only thing you need to know is that the procedures required data objects. Queries could be called with or without params.

#### Records

In some sub-namespaces, you can find records. Such record classes provide a syntax sugar not defined in the lexicon scheme. This sugar provides a more convenient way to work with repository operations. Such as creating a record, deleting a record, and so on.

Here are some available records of Bluesky records:

```python
from atproto import Client

Client().app.bsky.feed.post
Client().app.bsky.feed.like
Client().app.bsky.graph.follow
Client().app.bsky.graph.block
Client().app.bsky.actor.profile
# ... more
```

Usage example with the `post` record:

```python
from atproto import AtUri, Client, models

client = Client()
client.login('my-username', 'my-password')

posts = client.app.bsky.feed.post.list(client.me.did, limit=10)
for uri, post in posts.records.items():
    print(uri, post.text)

post = client.app.bsky.feed.post.get(client.me.did, AtUri.from_str(uri).rkey)
print(post.value.text)

post_record = models.AppBskyFeedPost.Record(text='test record namespaces', created_at=client.get_current_time_iso())
new_post = client.app.bsky.feed.post.create(client.me.did, post_record)
print(new_post)

deleted_post = client.app.bsky.feed.post.delete(client.me.did, AtUri.from_str(new_post.uri).rkey)
print(deleted_post)
```

Please note that not all repository operations are covered by these syntax sugars. You can always use the low-level methods to perform any desired action. One such action is updating a record.

#### Models

To deal with methods, we need to deal with models! Models are available in the `models` module and have NSID-based aliases. Let's take a look at it.

```python
from atproto import models

models.ComAtprotoIdentityResolveHandle
models.AppBskyFeedPost
models.AppBskyActorGetProfile
# 90+ more...
```

The model classes in the "models" aliases could be:

- Data model
- Params model
- Response model
- Sugar response model
- Record model
- Type model

The only thing you need to know is how to create instances of models. Not with all models, you will work as model-creator. For example, SDK will create Response models for you.

There are a few ways how to create the instance of a model:

- Dict-based
- Class-based

The instances of data and params models should be passed as arguments to the methods that were described above.

Dict-based:

```python
from atproto import Client

client = Client()
client.login('my-username', 'my-password')
# The params model will be created automatically internally for you!
print(client.com.atproto.identity.resolve_handle({'handle': 'marshal.dev'}))
```

Class-based:

```python
from atproto import Client, models

client = Client()
client.login('my-username', 'my-password')
params = models.ComAtprotoIdentityResolveHandle.Params(handle='marshal.dev')
print(client.com.atproto.identity.resolve_handle(params))
```

Tip: look at typehint of the method to figure out the name and the path to the input/data model!

Pro Tip: use IDE autocompletion to find necessary models! Just start typing the method name right after the dot (`models.{type method name in camel case`).

Models could be nested as hell. Be ready for it!

This is how we can send a post with the image using low-level XRPC Client:

```python
from atproto import Client, models

client = Client()
client.login('my-username', 'my-password')

with open('cat.jpg', 'rb') as f:
    img_data = f.read()

    upload = client.upload_blob(img_data)
    images = [models.AppBskyEmbedImages.Image(alt='Img alt', image=upload.blob)]
    embed = models.AppBskyEmbedImages.Main(images=images)

    client.com.atproto.repo.create_record(
        models.ComAtprotoRepoCreateRecord.Data(
            repo=client.me.did,
            collection=models.ids.AppBskyFeedPost,
            record=models.AppBskyFeedPost.Record(
                created_at=client.get_current_time_iso(), text='Text of the post', embed=embed
            ),
        )
    )

    # of course, you can use the syntax sugar here instead
    post = models.AppBskyFeedPost.Record(text='Text of the post', embed=embed, created_at=client.get_current_time_iso())
    client.app.bsky.feed.post.create(client.me.did, post)
    # or even high-level client
    client.send_image(text='Text of the post', image=img_data, image_alt='Img alt')
    # these three methods are equivalent
```

I hope you are not scared. May the Force be with you. Good luck!

### Change log

The full change log is available in [CHANGES.md](https://github.com/MarshalX/atproto/blob/main/CHANGES.md).

### Contributing

Contributions of all sizes are welcome. The contribution guidelines will be presented later.

### License

MIT


            

Raw data

            {
    "_id": null,
    "home_page": "https://github.com/MarshalX/atproto",
    "name": "atproto",
    "maintainer": null,
    "docs_url": null,
    "requires_python": "<3.14,>=3.8",
    "maintainer_email": null,
    "keywords": "library, sdk, codegen, xrpc, xrpc-client, atprotocol, atproto, lexicon, parser, schema, bluesky, bluesky-api, at, uri, atp, nsid, did, cid",
    "author": "Ilya (Marshal)",
    "author_email": "ilya@marshal.dev",
    "download_url": "https://files.pythonhosted.org/packages/45/ad/195c036a01b5857c9b49b5bf39dd978d075ec97c0c6132d95040b2a723da/atproto-0.0.56.tar.gz",
    "platform": null,
    "description": "<p align=\"center\">\n    <a href=\"https://github.com/MarshalX/atproto\">\n        <img alt=\"Logo of atproto SDK for Python by Midjourney'\" src=\"https://github.com/MarshalX/atproto/raw/main/.github/images/logo.png\">\n    </a>\n    <br>\n    <b>Autogenerated from lexicons, well type hinted, documented, sync and async SDK for Python</b>\n    <br>\n    <a href=\"https://github.com/MarshalX/atproto/tree/main/examples\">\n        Examples\n    </a>\n    \u2022\n    <a href=\"https://atproto.blue\">\n        Documentation\n    </a>\n    \u2022\n    <a href=\"https://discord.gg/PCyVJXU9jN\">\n        Discord Bluesky API\n    </a>\n</p>\n\n## The AT Protocol SDK\n\n> \u26a0\ufe0f Under construction. Until the 1.0.0 release compatibility between versions is not guaranteed. \n\nCode snippet:\n\n```python\nfrom atproto import Client, client_utils\n\n\ndef main():\n    client = Client()\n    profile = client.login('my-handle', 'my-password')\n    print('Welcome,', profile.display_name)\n    \n    text = client_utils.TextBuilder().text('Hello World from ').link('Python SDK', 'https://atproto.blue')\n    post = client.send_post(text)\n    client.like(post.uri, post.cid)\n\n\nif __name__ == '__main__':\n    main()\n\n```\n\n<details>\n  <summary>Code snippet of async version</summary>\n\n```python\nimport asyncio\n\nfrom atproto import AsyncClient, client_utils\n\n\nasync def main():\n    client = AsyncClient()\n    profile = await client.login('my-handle', 'my-password')\n    print('Welcome,', profile.display_name)\n\n    text = client_utils.TextBuilder().text('Hello World from ').link('Python SDK', 'https://atproto.blue')\n    post = await client.send_post(text)\n    await client.like(post.uri, post.cid)\n\n    \nif __name__ == '__main__':\n    # use run() for a higher Python version\n    asyncio.get_event_loop().run_until_complete(main())\n\n```\n</details>\n\n\ud83d\udcac [Direct Messages (Chats)](https://atproto.blue/en/latest/dm.html)\n\n\ud83c\udf7f [Example project with custom feed generator](https://github.com/MarshalX/bluesky-feed-generator)\n\n\ud83d\udd25 [Firehose data streaming is available](https://atproto.blue/en/latest/atproto_firehose/index.html)\n\n\ud83c\udf10 [Identity resolvers for DID and Handle](https://atproto.blue/en/latest/atproto_identity/index.html)\n\n### Introduction\n\nThis SDK attempts to implement everything that provides ATProto. There is support for Lexicon Schemes, XRPC clients, Firehose, Identity, DID keys, signatures, and more. All models, queries, and procedures are generated automatically. The main focus is on the lexicons of atproto.com and bsky.app, but it doesn't have a vendor lock on it. Feel free to use the code generator for your own lexicon schemes. SDK also provides utilities to work with CID, NSID, AT URI Schemes, DAG-CBOR, CAR files, DID Documents and more.\n\n### Requirements\n\n- Python 3.8 or higher.\n\n### Installing\n\n``` bash\npip install atproto\n```\n\n### Quick start\n\nFirst of all, you need to create the instance of the XRPC Client. To do so, you have two major options: asynchronous, and synchronous. The difference only in import and how you call the methods. If you are not familiar with async, use sync instead.\n\nFor sync:\n\n```python\nfrom atproto import Client\n\nclient = Client()\n# By default, it uses the server of bsky.app. To change this behavior, pass the base api URL to constructor\n# Client('https://example.com')\n```\n\nFor async:\n\n```python\nfrom atproto import AsyncClient\n\nclient = AsyncClient()\n# By default, it uses the server of bsky.app. To change this behavior, pass the base api URL to constructor\n# AsyncClient('https://example.com')\n```\n\nIn the snippets below, only the sync version will be presented.\n\nRight after the creation of the Client instance, you probably want to access the full API and perform actions by profile. To achieve this, you should log in to the network using your handle and password. The password could be app-specific.\n\n```python\nfrom atproto import Client\n\nclient = Client()\nclient.login('my-username', 'my-password')\n```\n\nYou are awesome! Now you feel to pick any high-level method that you want and perform it!\n\nCode to send post:\n\n```python\nfrom atproto import Client\n\nclient = Client()\nclient.login('my-username', 'my-password')\nclient.send_post(text='Hello World!')\n```\n\nUseful links to continue:\n\n- [List of all methods with documentation](https://atproto.readthedocs.io/en/latest/atproto_client/index.html).\n- [Examples of using the methods](https://github.com/MarshalX/atproto/tree/main/examples).\n\n### SDK structure\n\nThe SDK is built upon the following components:\n\n| Package            | Description                                                                 |\n|--------------------|-----------------------------------------------------------------------------|\n| `atproto`          | Import shortcuts to other packages.                                         |\n| `atproto_cli`      | CLI tool to generate code.                                                  |\n| `atproto_client`   | XRPC Client, data models, and utils like rich text helper.                  |\n| `atproto_codegen`  | Code generator of models, clients, and namespaces.                          |\n| `atproto_core`     | Tools to work with NSID, AT URI Schemes, CID, CAR files, and DID Documents. |\n| `atproto_crypto`   | Crypto utils like multibase, signature verification, work with DID keys.    |\n| `atproto_firehose` | Firehose (data streaming) client and models.                                |\n| `atproto_identity` | Identity resolvers for DID, Handle, AT Protocol data, signing keys.         |\n| `atproto_lexicon`  | Lexicon parser.                                                             |\n| `atproto_server`   | Server-side utils like JWT.                                                 |\n\nI highly recommend you to use the `atproto` package to import everything that you need. \nIt contains shortcuts to all other packages.\n\n### Documentation\n\nThe documentation is live at [atproto.blue](https://atproto.blue/).\n\n### Getting help\n\nYou can get help in several ways:\n- Report bugs, request new features by [creating an issue](https://github.com/MarshalX/atproto/issues/new).\n- Ask questions by [starting a discussion](https://github.com/MarshalX/atproto/discussions/new).\n- Ask questions in [Discord server](https://discord.gg/PCyVJXU9jN).\n\n### Advanced usage\n\nI'll be honest. The high-level Client that was shown in the \"Quick Start\" section is not a real ATProto API. This is syntax sugar built upon the real XRPC methods! The high-level methods are not cover the full need of developers. To be able to do anything that you want, you should know to work with low-level API. Let's dive into it!\n\nThe basics:\n- Namespaces \u2013 classes that group sub-namespaces and the XRPC queries and procedures. Built upon NSID ATProto semantic.\n- Model \u2013 dataclasses for input, output, and params of the methods from namespaces. Models describe Record and all other types in the Lexicon Schemes.\n\n#### Namespaces\n\nThe client contains references to the root of all namespaces. It's `com` and `app` for now.\n\n```python\nfrom atproto import Client\n\nClient().com\nClient().app\n```\n\nTo dive deeper, you can navigate using hints from your IDE. Thanks to well-type hinted SDK, it's much easier.\n\n```python\nfrom atproto import Client\n\nClient().com.atproto.server.create_session(...)\nClient().com.atproto.sync.get_blob(...)\nClient().app.bsky.feed.get_likes(...)\nClient().app.bsky.graph.get_follows(...)\n```\n\nThe endpoint of the path is always the method that you want to call. The method presents a query or procedure in XRPC. You should not care about it much. The only thing you need to know is that the procedures required data objects. Queries could be called with or without params.\n\n#### Records\n\nIn some sub-namespaces, you can find records. Such record classes provide a syntax sugar not defined in the lexicon scheme. This sugar provides a more convenient way to work with repository operations. Such as creating a record, deleting a record, and so on.\n\nHere are some available records of Bluesky records:\n\n```python\nfrom atproto import Client\n\nClient().app.bsky.feed.post\nClient().app.bsky.feed.like\nClient().app.bsky.graph.follow\nClient().app.bsky.graph.block\nClient().app.bsky.actor.profile\n# ... more\n```\n\nUsage example with the `post` record:\n\n```python\nfrom atproto import AtUri, Client, models\n\nclient = Client()\nclient.login('my-username', 'my-password')\n\nposts = client.app.bsky.feed.post.list(client.me.did, limit=10)\nfor uri, post in posts.records.items():\n    print(uri, post.text)\n\npost = client.app.bsky.feed.post.get(client.me.did, AtUri.from_str(uri).rkey)\nprint(post.value.text)\n\npost_record = models.AppBskyFeedPost.Record(text='test record namespaces', created_at=client.get_current_time_iso())\nnew_post = client.app.bsky.feed.post.create(client.me.did, post_record)\nprint(new_post)\n\ndeleted_post = client.app.bsky.feed.post.delete(client.me.did, AtUri.from_str(new_post.uri).rkey)\nprint(deleted_post)\n```\n\nPlease note that not all repository operations are covered by these syntax sugars. You can always use the low-level methods to perform any desired action. One such action is updating a record.\n\n#### Models\n\nTo deal with methods, we need to deal with models! Models are available in the `models` module and have NSID-based aliases. Let's take a look at it.\n\n```python\nfrom atproto import models\n\nmodels.ComAtprotoIdentityResolveHandle\nmodels.AppBskyFeedPost\nmodels.AppBskyActorGetProfile\n# 90+ more...\n```\n\nThe model classes in the \"models\" aliases could be:\n\n- Data model\n- Params model\n- Response model\n- Sugar response model\n- Record model\n- Type model\n\nThe only thing you need to know is how to create instances of models. Not with all models, you will work as model-creator. For example, SDK will create Response models for you.\n\nThere are a few ways how to create the instance of a model:\n\n- Dict-based\n- Class-based\n\nThe instances of data and params models should be passed as arguments to the methods that were described above.\n\nDict-based:\n\n```python\nfrom atproto import Client\n\nclient = Client()\nclient.login('my-username', 'my-password')\n# The params model will be created automatically internally for you!\nprint(client.com.atproto.identity.resolve_handle({'handle': 'marshal.dev'}))\n```\n\nClass-based:\n\n```python\nfrom atproto import Client, models\n\nclient = Client()\nclient.login('my-username', 'my-password')\nparams = models.ComAtprotoIdentityResolveHandle.Params(handle='marshal.dev')\nprint(client.com.atproto.identity.resolve_handle(params))\n```\n\nTip: look at typehint of the method to figure out the name and the path to the input/data model!\n\nPro Tip: use IDE autocompletion to find necessary models! Just start typing the method name right after the dot (`models.{type method name in camel case`).\n\nModels could be nested as hell. Be ready for it!\n\nThis is how we can send a post with the image using low-level XRPC Client:\n\n```python\nfrom atproto import Client, models\n\nclient = Client()\nclient.login('my-username', 'my-password')\n\nwith open('cat.jpg', 'rb') as f:\n    img_data = f.read()\n\n    upload = client.upload_blob(img_data)\n    images = [models.AppBskyEmbedImages.Image(alt='Img alt', image=upload.blob)]\n    embed = models.AppBskyEmbedImages.Main(images=images)\n\n    client.com.atproto.repo.create_record(\n        models.ComAtprotoRepoCreateRecord.Data(\n            repo=client.me.did,\n            collection=models.ids.AppBskyFeedPost,\n            record=models.AppBskyFeedPost.Record(\n                created_at=client.get_current_time_iso(), text='Text of the post', embed=embed\n            ),\n        )\n    )\n\n    # of course, you can use the syntax sugar here instead\n    post = models.AppBskyFeedPost.Record(text='Text of the post', embed=embed, created_at=client.get_current_time_iso())\n    client.app.bsky.feed.post.create(client.me.did, post)\n    # or even high-level client\n    client.send_image(text='Text of the post', image=img_data, image_alt='Img alt')\n    # these three methods are equivalent\n```\n\nI hope you are not scared. May the Force be with you. Good luck!\n\n### Change log\n\nThe full change log is available in [CHANGES.md](https://github.com/MarshalX/atproto/blob/main/CHANGES.md).\n\n### Contributing\n\nContributions of all sizes are welcome. The contribution guidelines will be presented later.\n\n### License\n\nMIT\n\n",
    "bugtrack_url": null,
    "license": "MIT",
    "summary": "The AT Protocol SDK",
    "version": "0.0.56",
    "project_urls": {
        "Author": "https://github.com/MarshalX",
        "Changes": "https://github.com/MarshalX/atproto/blob/main/CHANGES.md",
        "Documentation": "https://atproto.blue",
        "Homepage": "https://github.com/MarshalX/atproto",
        "Repository": "https://github.com/MarshalX/atproto",
        "Tracker": "https://github.com/MarshalX/atproto/issues"
    },
    "split_keywords": [
        "library",
        " sdk",
        " codegen",
        " xrpc",
        " xrpc-client",
        " atprotocol",
        " atproto",
        " lexicon",
        " parser",
        " schema",
        " bluesky",
        " bluesky-api",
        " at",
        " uri",
        " atp",
        " nsid",
        " did",
        " cid"
    ],
    "urls": [
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "b4e062f833f12e4751863c6ad5c8c3a754fccdf81f747e2aeea5b9ee4f777af9",
                "md5": "bbe379ae960c25c78d3637d7ab19c1ce",
                "sha256": "b49f93b6fc4f09ebaf3f65bd96f4080c51841db0d591ad577df620cf48930a7c"
            },
            "downloads": -1,
            "filename": "atproto-0.0.56-py3-none-any.whl",
            "has_sig": false,
            "md5_digest": "bbe379ae960c25c78d3637d7ab19c1ce",
            "packagetype": "bdist_wheel",
            "python_version": "py3",
            "requires_python": "<3.14,>=3.8",
            "size": 337090,
            "upload_time": "2024-12-05T17:47:44",
            "upload_time_iso_8601": "2024-12-05T17:47:44.834562Z",
            "url": "https://files.pythonhosted.org/packages/b4/e0/62f833f12e4751863c6ad5c8c3a754fccdf81f747e2aeea5b9ee4f777af9/atproto-0.0.56-py3-none-any.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "45ad195c036a01b5857c9b49b5bf39dd978d075ec97c0c6132d95040b2a723da",
                "md5": "7cc9a5fc286cec0d6545db6d90a0ebcc",
                "sha256": "8f273dad6bf27e878ca98d58c41c85a0da275eb17f23a394057edee8220c6776"
            },
            "downloads": -1,
            "filename": "atproto-0.0.56.tar.gz",
            "has_sig": false,
            "md5_digest": "7cc9a5fc286cec0d6545db6d90a0ebcc",
            "packagetype": "sdist",
            "python_version": "source",
            "requires_python": "<3.14,>=3.8",
            "size": 166550,
            "upload_time": "2024-12-05T17:47:46",
            "upload_time_iso_8601": "2024-12-05T17:47:46.469505Z",
            "url": "https://files.pythonhosted.org/packages/45/ad/195c036a01b5857c9b49b5bf39dd978d075ec97c0c6132d95040b2a723da/atproto-0.0.56.tar.gz",
            "yanked": false,
            "yanked_reason": null
        }
    ],
    "upload_time": "2024-12-05 17:47:46",
    "github": true,
    "gitlab": false,
    "bitbucket": false,
    "codeberg": false,
    "github_user": "MarshalX",
    "github_project": "atproto",
    "travis_ci": false,
    "coveralls": true,
    "github_actions": true,
    "lcname": "atproto"
}
        
Elapsed time: 0.38214s