Metadata-Version: 2.4
Name: afp-sdk
Version: 0.7.0
Summary: Autonomous Futures Protocol Python SDK
Keywords: autonity,web3,trading,crypto,prediction,forecast,markets
License-Expression: MIT
License-File: LICENSE
Classifier: Development Status :: 4 - Beta
Classifier: Programming Language :: Python
Classifier: Programming Language :: Python :: 3.12
Classifier: Programming Language :: Python :: 3.13
Classifier: Programming Language :: Python :: 3.14
Requires-Dist: dag-cbor>=0.3.3
Requires-Dist: decorator>=5.2.1
Requires-Dist: inflection>=0.5.1
Requires-Dist: ipld-car>=0.0.1
Requires-Dist: multiformats>=0.3.1.post4
Requires-Dist: pydantic>=2.10.6
Requires-Dist: requests>=2.32.0
Requires-Dist: rfc8785>=0.1.4
Requires-Dist: siwe>=4.4.0
Requires-Dist: trezor[ethereum]>=0.13.10
Requires-Dist: web3>=7.6.0
Requires-Python: >=3.12
Project-URL: Changes, https://github.com/autonity/afp-sdk/blob/master/CHANGELOG.md
Project-URL: Homepage, https://github.com/autonity/afp-sdk
Project-URL: Issues, https://github.com/autonity/afp-sdk/issues
Project-URL: Source, https://github.com/autonity/afp-sdk
Description-Content-Type: text/markdown

# Autonomous Futures Protocol Python SDK

Decentralized clearing and creation of Forecast Futures on any timeseries.

![CI](https://github.com/autonity/afp-sdk/actions/workflows/ci.yml/badge.svg)

## Installation

This library is published on PyPI as the [afp-sdk](https://pypi.org/project/afp-sdk/)
package. It can be installed in a virtualenv with:

```py
pip install afp-sdk
```

## Documentation

See [afp.autonity.org](https://afp.autonity.org/) for the Autonomous Futures Protocol
documentation, including the Python SDK reference.

## Authentication

The SDK supports 3 methods for authenticating with AutEx and signing blockchain
transactions.

- **Private key:** Use an Autonity account's private key as a hex-string with `0x`
  prefix.
- **Key file:** Use a [Geth / Clef](https://geth.ethereum.org/docs/fundamentals/account-management)
  key file.
- **Trezor device:** Use a [Trezor](https://trezor.io/) hardware wallet. The derivation
  path of an Ethereum account needs to be specified, which can be found in the Account
  Settings in the Trezor Suite.

## Overview

The `afp` package consists of the following:

- `afp` top-level module: High-level API for interacting with the AFP Clearing
  System and the AutEx exchange.
- `afp.bindings` submodule: Low-level API that provides typed Python bindings
  for the Clearing System smart contracts.

## Configuration

By default the SDK communicates with the AFP Clearing System contracts on
Autonity Mainnet, and the AutEx Exchange. Connection parameters can be
overridden via the `afp.AFP` constructor and environment variables; see the
documentation of the `afp.AFP` class for the available parameters.

## Usage

### Preparation

In order to trade in the AFP system, traders need to prepare the following:

- The ID of a product to be traded.
- The address of the product's collateral token.
- An Autonity account for managing the margin account. It needs to hold a
  balance in ATN (for paying gas fee) and in the product's collateral token.
- An Autonity account for signing intents and submitting trades to the
  exchange. The two accounts can be the same.
- The address of an Autonity RPC provider. They can be found on
  [Chainlist](https://chainlist.org/?search=autonity).

We can store those in the following constants (using random example IDs):

```py
import os

PRODUCT_ID = "0x38d502bb683f53ec7c3d7a14b4aa47ac717659e121426131c0189c15bf4b9460"
COLLATERAL_ASSET = "0xD1A1e4035a164cF42228A8aAaBC2c0Ac9e49687B"
PRIVATE_KEY = os.environ["PRIVATE_KEY"]
AUTONITY_RPC_URL = "https://bakerloo.autonity-apis.com"
```

### Configuration

An application instance can be created with the `afp.AFP()` constructor. An instance
is associated with a trading venue and a margin account.

The required constructor arguments are the RPC provider URL and the authenticator of
the blockchain account that manages the margin account.

An "authenticator" is a service that implements the `afp.Authenticator` protocol.
Available options are `afp.PrivateKeyAuthenticator` that reads the private key
from a constructor argument, and `afp.KeyfileAuthenticator` that reads the private
key from an encrypted keyfile.

```py
import afp

app = afp.AFP(
    rpc_url=AUTONITY_RPC_URL,
    authenticator=afp.PrivateKeyAuthenticator(PRIVATE_KEY),
)
```

### Margin Account API

Margin accounts can be managed via the `MarginAccount` session object. It
connects to the specified Autonity RPC provider and communicates with the
Clearing System smart contracts.

```py
margin_account = app.MarginAccount()
```

Collateral can be deposited into the margin account with `margin_account.deposit()`.

```py
from decimal import Decimal

margin_account.deposit(COLLATERAL_ASSET, Decimal("100.00"))
print(margin_account.capital(COLLATERAL_ASSET))
```

### Trading API

Functions of the trading API can be accessed via the `Trading` session object.
It communicates with the AutEx exchange and authenticates on creation with the
intent account's private key. The intent account authenticator is optional, it
defaults to the authenticator set in the `AFP` constructor.

```py
trading = app.Trading()
```

To start trading a product, its parameters shall be retrieved from the server.

```py
product = trading.product(PRODUCT_ID)
```

Intents can be created with `trading.create_intent()`. Intent creation involves
hashing and signing the intent data. (The intent account's address is derived
from the private key specified in the `Trading` constructor.)

```py
from datetime import datetime, timedelta
from decimal import Decimal

intent = trading.create_intent(
    margin_account_id=MARGIN_ACCOUNT_ID,
    product=product,
    side="bid",
    limit_price=Decimal("1.23"),
    quantity=2,
    max_trading_fee_rate=Decimal("0.1"),
    good_until_time=datetime.now() + timedelta(hours=1),
)
```

The intent expressing a limit order can then be submitted to the exchange with
`trading.submit_limit_order()` that returns the created order object.

```py
order = trading.submit_limit_order(intent)
print(order)
```

The exchange then performs various checks to ensure that the order is valid. To
ensure that the order has been accepted, its state can be polled with
`trading.order()`.

```py
order = trading.order(order.id)
print(order.state)
```

Fills of orders submitted by the authenticated intent account can be queried
with `trading.order_fills()`.

```py
fills = trading.order_fills(product_id=PRODUCT_ID)
print(fills)
```

### Product API

Use the `Product` session object to validate a product schema, pin the
specification to IPFS, and register it on-chain.

In order to use the product API for product building, connection parameters of
an IPFS pinning service like [Filebase](https://filebase.com/) should be
included in the `AFP` constructor parameters.

```py
IPFS_API_URL = "https://rpc.filebase.io"
IPFS_API_KEY = os.environ["IPFS_API_KEY"]

app = afp.AFP(
    rpc_url=AUTONITY_RPC_URL,
    authenticator=afp.PrivateKeyAuthenticator(PRIVATE_KEY),
    ipfs_api_url=IPFS_API_URL,
    ipfs_api_key=IPFS_API_KEY,
)

product = app.Product()
```

A JSON product specification can be parsed and validated with
`product.validate_json()`.

```py
with open("product-spec.json") as spec_file:
    specification = product.validate_json(spec_file.read())
```

Alternatively, it can also be validated from a Python dictionary.

```py
spec_dict = {
    "product": {...},
    "outcome_space": {...},
    "outcome_point": {...},
    "oracle_config": {...},
    "oracle_fallback": {...},
}
specification = product.validate(spec_dict)
```

Product specifications are stored at two different places, the `PredictionProductV1`
object is stored on-chain in the Product Registry contract, while the rest of the
product specification is referred to as _extended metadata_ and it is uploaded to IPFS.

The first step therefore is to upload the extended metadata of the product to
IPFS and to pin the root node of the DAG. `product.pin()` returns a modified
copy of the specification that incudes the extended metadata CID.

```py
pinned_specification = product.pin(specification)
```

The product can then be registered with the Product Registry contract via a
blockchain transaction.

```
tx = product.register(
    pinned_specification, initial_builder_stake=Decimal("10")
)
```

See further code examples in the
[examples](https://github.com/autonity/afp-sdk/tree/master/examples/) directory.

## Development

See [DEVELOPMENT.md](https://github.com/autonity/afp-sdk/blob/master/DEVELOPMENT.md)
for developer documentation.
