hubble-exchange


Namehubble-exchange JSON
Version 0.8.8 PyPI version JSON
download
home_page
SummaryOfficial python SDK for Hubble Exchange
upload_time2023-11-30 13:17:20
maintainer
docs_urlNone
author
requires_python>=3.7
licenseMIT License Copyright (c) 2023 Hubble Exchange Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
keywords hubble exchange orderbook perpetual futures dex sdk python avalanche ethereum web3 crypto
VCS
bugtrack_url
requirements No requirements were recorded.
Travis-CI No Travis.
coveralls test coverage No coveralls.
            # Python SDK for Hubble Exchange

[Hubble Exchange](https://hubble.exchange) is a Layer 1 Blockchain for a Decentralised Perps OrderBook
<br>[Twitter](https://twitter.com/HubbleExchange)


## Installation

The simplest way is to install the package from PyPI:
```shell
pip install hubble-exchange
```

## Example usage:

All read/write functions are async
<br>Requires HUBBLE_RPC, HUBBLE_WS_RPC, HUBBLE_ENV, PRIVATE_KEY environment variable to be set
```shell
export HUBBLE_RPC=https://candy-hubblenet-rpc.hubble.exchange/ext/bc/iKMFgo49o4X3Pd3UWUkmPwjKom3xZz3Vo6Y1kkwL2Ce6DZaPm/rpc
export HUBBLE_WS_RPC=wss://candy-hubblenet-rpc.hubble.exchange/ext/bc/iKMFgo49o4X3Pd3UWUkmPwjKom3xZz3Vo6Y1kkwL2Ce6DZaPm/ws
export HUBBLE_ENV=hubblenext
export PRIVATE_KEY=<private key>
export HUBBLE_INDEXER_API_URL=""
```

```python
import os
from hubble_exchange import HubbleClient, OrderBookDepthResponse, LimitOrder, IOCOrder


async def main():

    async def callback(response):
        print(f"Received response: {response}")
        return response

    client = HubbleClient(os.getenv("PRIVATE_KEY"))

    # get a dict of all market ids and names - for example {0: "ETH-Perp", 1: "AVAX-Perp"}
    markets = await client.get_markets()

    # place limit orders
    limit_orders = []
    limit_orders.append(LimitOrder.new(3, 1, 1.2, False, False)) # market = 3, qty = 1, price = 1.2, reduce_only = False, post_only = False
    limit_orders.append(LimitOrder.new(0, 0.1, 1800, False, True)) # market = 0, qty = 0.1, price = 1800, reduce_only = False, post_only = True
    # placed_orders list will contain the order ids for the orders placed
    placed_orders = await client.place_limit_orders(limit_orders, True, callback)

    # place ioc orders
    ioc_orders = []
    ioc_orders.append(IOCOrder.new(3, 1, 1.2, False, 3)) # market = 3, qty = 1, price = 1.2, reduce_only = False, expiry = 3 seconds
    ioc_orders.append(IOCOrder.new(0, 0.1, 1800, False, 3)) # market = 0, qty = 0.1, price = 1800, reduce_only = False, expiry = 3 seconds
    # placed_orders list will contain the order ids for the orders placed
    placed_orders = await client.place_ioc_orders(ioc_orders, True, callback)

    # get limit order details - only works for open limit orders
    order_details = await client.get_limit_order_details(order.id, callback)

    # get status for limit orders - works for all order statuses
    order_status = await client.get_limit_order_status(order.id, callback)
    
    # cancel an order
    await client.cancel_limit_orders([order], True, callback)

    # order can also be cancelled by order id
    await client.cancel_order_by_id(order.id, True, callback)

    # get current order book for market = 1
    order_book = await client.get_order_book(1, callback)

    # get current margin and positions(uses the address for which private key is set)
    positions = await client.get_margin_and_positions(callback)

    # get order fills
    order_fills = await client.get_order_fills()

    # get all open orders
    open_orders = await client.get_open_orders(None, callback)
    # get open orders for market_id = 0
    open_orders = await client.get_open_orders(0, callback)

    # get user trades in all markets in a time range
    await client.get_trades(None, 1691669740, 1691583340, callback)
    # get user trades in market_id = 3 in a time range
    await client.get_trades(3, 1691669740, 1691583340, callback)

    # subscribe to order book updates for market = 0; receives a new message every second(only for those prices where the quantity has changed)
    async def on_message(ws, message):
        print(f"Received orderbook update: {message}")

    asyncio.run(client.subscribe_to_order_book_depth(0, callback=on_message))
```

---

### GetPositionsResponse description
- margin: trader's margin
- reservedMargin: trader's reserved margin
- positions: List[Position]

### Position description
- market: market id
- openNotional: open notional
- size: size of the position
- unrealisedFunding: unrealised funding payment for this position
- liquidationThreshold: max amount of size that can be liquidated at once
- notionalPosition: notional position size
- unrealisedProfit: unrealised profit on this position
- marginFraction: margin/notional position
- liquidationPrice: the asset price at which this position might get liquidated
- markPrice: latest price of this market

### OrderStatusResponse description
- executedQty: executed quantity
- orderId: order id
- origQty: total order quantity
- price: order price
- reduceOnly: whether the order is reduce-only or not
- postOnly: whether the order is post-only or not
- positionSide: (LONG|SHORT)
- status: order status (NEW|FILLED|CANCELED|REJECTED|PARTIALLY_FILLED)
- symbol: market id
- time: time when the order was placed in unix format
- type: LIMIT_ORDER
- updateTime: time when the order was last updated in unix format
- salt: order salt

## Orderbook depth feed
Orderbook depth feed can be subscribed to using the `subscribe_to_order_book_depth` method. It emits the change in orderbook depth every second. It only emits the prices where the quantity has changed, but with absolute quantities(not the change).

```python
import os
from hubble_exchange import HubbleClient

async def main():
    client = HubbleClient(os.getenv("PRIVATE_KEY"))
    # subscribe to market id 0
    await client.subscribe_to_order_book_depth(0, callback=callback)
```

### OrderBookDepthUpdateResponse description
- T: timestamp
- symbol: market id
- bids: list of [price, quantity] for bids
- asks: list of [price, quantity] for asks


## Trader feed

All order updates related to a particular trader can be subscribed to using the `subscribe_to_trader_updates` method.
It can be subscribed in 2 confirmation modes - head block or accepted block. Events received in head block mode are not finalised and can be reverted. When an event is removed from the chain, the client will receive a `removed=True` event. Events received in accepted block mode are finalised and will alwats have `removed=False`.

```python
import os
from hubble_exchange import HubbleClient, ConfirmationMode

async def main():
    client = HubbleClient(os.getenv("PRIVATE_KEY"))
    await client.subscribe_to_trader_updates(ConfirmationMode.accepted, callback)
```


### TraderFeedUpdate description

- Trader: address of the trader
- OrderId: order id
- OrderType: order type - "limit" order or "ioc" (market order)
- Removed: whether the event is being removed or not
- EventName: name of the contract event (OrderAccepted|OrderRejected|OrderMatched|OrderCancelAccepted|OrderCancelRejected)
- Args: args is a dynamic field and it contains information about the event. For example, OrderPlaced event contains order object, OrderMatched event contains fillAmount and price
- BlockNumber: block number in which the transaction was included
- BlockStatus: (head|accepted) whether the block is accepted or only a preferred block(head block)
- Timestamp: timestamp of the block in unix format
- TransactionHash: transaction hash

#### EventName description
- OrderAccepted - Order was successfully placed
- OrderRejected - Order was rejected
- OrderMatched - Order was matched
- OrderCancelAccepted - Order was successfully cancelled(only for Limit orders)
- OrderCancelRejected - Order cancel was rejected(only for Limit orders)

Removed events are emmitted during chain reorgs, and are most likely to be temporary. They are only emmitted when subscribing to head block events. If removed=True, the client might need to do a reverse operation for the given event. For example if an OrderMatched event is received with removed=True, the client should add the fillAmount back to unfilled quantity of the order.

## Market feed

All trades of a particular market can be subscribed to using the `subscribe_to_market_updates` method.
Similar to the trader feed, it has 2 confirmation modes - head block or accepted block.

```python
import os
from hubble_exchange import HubbleClient, ConfirmationMode

async def main():
    client = HubbleClient(os.getenv("PRIVATE_KEY"))
    # subscribe to market id 0
    await client.subscribe_to_market_updates(0, ConfirmationMode.accepted, callback)
```

### MarketFeedUpdate description

- Trader: address of the trader who performed the trade
- Market: market id
- Size: size of the trade in decimals
- Price: price at which it was executed
- Removed: whether the event is being removed or not
- EventName: name of the contract event (PositionModified)
- BlockNumber: block number in which the transaction was included
- BlockStatus: (head|accepted) whether the block is accepted or only a preferred block(head block)
- Timestamp: timestamp of the block in unix format
- TransactionHash: transaction hash

Removed events are emmitted during chain reorgs, and are most likely to be temporary. They are only emmitted when subscribing to head block events. If removed=True, the client might need to do a reverse operation for the given event.


## Get limit order status

Get the status of a limit order. It can be used to get the status of any order, even if it is not placed using the sdk.

```python
order_status = await client.get_limit_order_status(order.id, callback)
```

### Status Description

```
Invalid = order does not exist
Placed = order is unfilled
PartiallyFilled = order is partially filled
Filled = order is completely filled
Cancelled = cancelled
```

## Open orders

Returns all open orders of the trader. It can be filtered by market id.

```python
# get all open orders
open_orders = await client.get_open_orders(None, callback)
# get open orders for market_id = 0
open_orders = await client.get_open_orders(0, callback)
```

### OpenOrder description

- Market: market id
- Price: order price
- Size: order size
- FilledSize: filled size
- Timestamp: timestamp of order place transaction in unix format
- Salt: order salt
- OrderId: order id
- ReduceOnly: whether the order is reduce-only or not
- PostOnly: whether the order is post-only or not
- OrderType: order type - "limit" order or "ioc" (market order)


## Get historical user trades

Returns all trades of the trader. It can be filtered by market id.

```python
# get user trades in all markets in a time range
await client.get_trades(None, 1691669740, 1691583340, callback)
# get user trades in market_id = 3 in a time range
await client.get_trades(3, 1691669740, 1691583340, callback)
```

### Trade description
- BlockNumber: block number in which the order match transaction was included
- TransactionHash: transaction hash
- Market: market id
- Timestamp: timestamp of the block in unix format
- TradedAmount: amount of the trade in decimals
- Price: price at which it was executed
- RealizedPnl: realized pnl of the trade in USD
- OpenNotional: open notional
- ExecutionMode: execution mode - "maker" or "taker"

## Candlestick data

Historical [candlestick data](https://www.investopedia.com/terms/o/ohlcchart.asp) can be fetched using the `get_candlestick_data` method using market and time range.

It can be obtained for the following timeframes:
- 1m - 1 minute
- 3m - 3 minutes
- 5m - 5 minutes
- 15m - 15 minutes
- 30m - 30 minutes
- 1h - 1 hour
- 4h - 4 hours
- 8h - 8 hours
- 1d - 1 day
- 1w - 1 week
- 1M - 1 month

```python
from hubble_exchange import HubbleClient

# get candlestick data for market id 0, timeframe = 5 minutes, start time = 1696932000, end time = 1696939200
response = await client.get_candlesticks(0, "5m", 1696932000, 1696939200)
```

### CandlestickData description
Example candlestick data:
```json
{
    start: 1696500000,     // start of the time period in unix format
    end: 1696500300,       // end of the time period in unix format
    volume: 192267.282,    // volume in USD
    open: 0.8745,          // open price for the duration
    close: 0.8851,         // close price for the duration
    high: 0.8947,          // highest price for the duration
    low: 0.8675,           // lowest price for the duration
}
```

## Historical funding rate

Historical funding rate can be queried using the `get_funding_rate` method using market and timestamp.
Funding payment happens every hour at the end of the hour as per UTC time. For example, funding payment for 1pm to 2pm will happen at 2pm.

If the provided timestamp is not the end of an hour, the funding rate for the previous hour will be returned. For example, if the timestamp is 1:25pm, the funding rate for 1pm will be returned.

```python
from hubble_exchange import HubbleClient

# get funding payments for market id 0, time = 1696932000
funding_rate = await client.get_funding_rate(0, 1696932000)
```

## Predicted funding rate

Predicted funding rate can be queried using the `get_predicted_funding_rate` method using market. It returns the predicted funding rate for the current hour.

```python
from hubble_exchange import HubbleClient

# get predicted funding rate for market id 2
funding_rate = await client.get_predicted_funding_rate(2)
```

## Historical open interest

Historical open interest can be queried using the `get_open_interest` method using market and timestamp. It returns the open interest in the units of the base asset. For example, if the market is ETH-Perp, the open interest will be in ETH.

```python
from hubble_exchange import HubbleClient

# get open interest for market id 0, time = 1696932000
open_interest = await client.get_open_interest(0, 1696932000)
```

## Get nonce

An account nonce is a transaction counter in each account, which is used to prevent replay attacks. Nonce should be increased by 1 in every transaction. The function `get_nonce` fetches the number of transactions sent by the address from the blockchain.

Nonce can be fetched once using `get_nonce` and be incremented in every subsequent transaction. However, if the same address is sending multiple transactions in parallel, it is recommended to use `get_nonce` before every transaction.

```python
nonce = await client.get_nonce()
# use this nonce in the next transaction like this
await client.place_limit_orders(orders, True, callback, {"nonce": nonce})
```

## Custom transaction options

The following options can be passed to the client to override the default

```python
{
    "nonce": Nonce,
    "gas": int,
    "maxFeePerGas": Union[str, Wei],
    "maxPriorityFeePerGas": Union[str, Wei],
},
```

It can be used for `place_limit_orders`, `cancel_limit_orders`, `cancel_order_by_id` methods.
Example:
```python

from web3 import Web3

client = HubbleClient(os.getenv("PRIVATE_KEY"))
placed_orders = await client.place_limit_orders(orders, callback, {
    "gas": 500_000,
    "maxFeePerGas": Web3.to_wei(80, 'gwei'),
    "maxPriorityFeePerGas": Web3.to_wei(20, 'gwei'),
})
```

## Transaction modes

There are different modes in which the client can wait for acknowledgement of the transaction. The default behaviour is to send the transaction and not wait for the acknowledgement.
This can be changed by explicitly asking the function to wait while sending the trasaction.

- TransactionMode.no_wait: The default behaviour is to send transactions to the blockchain and NOT wait for the acknowledgement.
- TransactionMode.wait_for_head: Wait for the transaction to be included in the canonical chain. At this time the block is preferred but not yet finalized. However, once the block in included in the canonical chain, the matching engine will start processing the order.
- TransactionMode.wait_for_accept: Wait for the transaction to be finalised.

Example:
```python
from hubble_exchange import TransactionMode
client = HubbleClient(os.getenv("PRIVATE_KEY"))
placed_orders = await client.place_limit_orders(orders, callback, mode=TransactionMode.wait_for_accept)

# or set the default mode for all transactions

client.set_transaction_mode(TransactionMode.wait_for_head)
```

---

## Waiting for response in place_limit_orders and cancel_limit_orders

The `place_limit_orders`, `place_ioc_orders`, and `cancel_limit_orders` methods can be called in 2 modes - wait for response or don't wait for response.
To get the acknowledgement of the transaction, use `wait_for_response=True`. The response will be a list of dicts with order ids and success boolean. Waiting for response will be slower because this can be confirmed only after the transaction is mined(accepted).
When using `wait_for_response=True`, the sdk will automatically set the transaction mode to `TransactionMode.wait_for_accept` because the response can be confirmed only after the transaction is mined.

Alternatively, the client can also use trader feed to listen to all the updates. This is faster when done with ConfirmationMode.head


## Config

- mininum_quantity: minimum quantity that can be placed in an order for a particular market. The quantity also has to be a multiple of this number.
- price_precision: number of decimal places for price for a particular market.

```python
from hubble_exchange.constants import get_minimum_quantity, get_price_precision

# get minimum quantity for market id 3
min_qty = get_minimum_quantity(3)
# get price precision for market id 3
price_precision = get_price_precision(3)
```

            

Raw data

            {
    "_id": null,
    "home_page": "",
    "name": "hubble-exchange",
    "maintainer": "",
    "docs_url": null,
    "requires_python": ">=3.7",
    "maintainer_email": "",
    "keywords": "hubble,exchange,orderbook,perpetual,futures,dex,sdk,python,avalanche,ethereum,web3,crypto",
    "author": "",
    "author_email": "Lumos <lumos@hubble.exchange>",
    "download_url": "https://files.pythonhosted.org/packages/8a/42/aba2187615f87117897cba6129ae63a40021ace710a5628ea549304b90ac/hubble_exchange-0.8.8.tar.gz",
    "platform": null,
    "description": "# Python SDK for Hubble Exchange\n\n[Hubble Exchange](https://hubble.exchange) is a Layer 1 Blockchain for a Decentralised Perps OrderBook\n<br>[Twitter](https://twitter.com/HubbleExchange)\n\n\n## Installation\n\nThe simplest way is to install the package from PyPI:\n```shell\npip install hubble-exchange\n```\n\n## Example usage:\n\nAll read/write functions are async\n<br>Requires HUBBLE_RPC, HUBBLE_WS_RPC, HUBBLE_ENV, PRIVATE_KEY environment variable to be set\n```shell\nexport HUBBLE_RPC=https://candy-hubblenet-rpc.hubble.exchange/ext/bc/iKMFgo49o4X3Pd3UWUkmPwjKom3xZz3Vo6Y1kkwL2Ce6DZaPm/rpc\nexport HUBBLE_WS_RPC=wss://candy-hubblenet-rpc.hubble.exchange/ext/bc/iKMFgo49o4X3Pd3UWUkmPwjKom3xZz3Vo6Y1kkwL2Ce6DZaPm/ws\nexport HUBBLE_ENV=hubblenext\nexport PRIVATE_KEY=<private key>\nexport HUBBLE_INDEXER_API_URL=\"\"\n```\n\n```python\nimport os\nfrom hubble_exchange import HubbleClient, OrderBookDepthResponse, LimitOrder, IOCOrder\n\n\nasync def main():\n\n    async def callback(response):\n        print(f\"Received response: {response}\")\n        return response\n\n    client = HubbleClient(os.getenv(\"PRIVATE_KEY\"))\n\n    # get a dict of all market ids and names - for example {0: \"ETH-Perp\", 1: \"AVAX-Perp\"}\n    markets = await client.get_markets()\n\n    # place limit orders\n    limit_orders = []\n    limit_orders.append(LimitOrder.new(3, 1, 1.2, False, False)) # market = 3, qty = 1, price = 1.2, reduce_only = False, post_only = False\n    limit_orders.append(LimitOrder.new(0, 0.1, 1800, False, True)) # market = 0, qty = 0.1, price = 1800, reduce_only = False, post_only = True\n    # placed_orders list will contain the order ids for the orders placed\n    placed_orders = await client.place_limit_orders(limit_orders, True, callback)\n\n    # place ioc orders\n    ioc_orders = []\n    ioc_orders.append(IOCOrder.new(3, 1, 1.2, False, 3)) # market = 3, qty = 1, price = 1.2, reduce_only = False, expiry = 3 seconds\n    ioc_orders.append(IOCOrder.new(0, 0.1, 1800, False, 3)) # market = 0, qty = 0.1, price = 1800, reduce_only = False, expiry = 3 seconds\n    # placed_orders list will contain the order ids for the orders placed\n    placed_orders = await client.place_ioc_orders(ioc_orders, True, callback)\n\n    # get limit order details - only works for open limit orders\n    order_details = await client.get_limit_order_details(order.id, callback)\n\n    # get status for limit orders - works for all order statuses\n    order_status = await client.get_limit_order_status(order.id, callback)\n    \n    # cancel an order\n    await client.cancel_limit_orders([order], True, callback)\n\n    # order can also be cancelled by order id\n    await client.cancel_order_by_id(order.id, True, callback)\n\n    # get current order book for market = 1\n    order_book = await client.get_order_book(1, callback)\n\n    # get current margin and positions(uses the address for which private key is set)\n    positions = await client.get_margin_and_positions(callback)\n\n    # get order fills\n    order_fills = await client.get_order_fills()\n\n    # get all open orders\n    open_orders = await client.get_open_orders(None, callback)\n    # get open orders for market_id = 0\n    open_orders = await client.get_open_orders(0, callback)\n\n    # get user trades in all markets in a time range\n    await client.get_trades(None, 1691669740, 1691583340, callback)\n    # get user trades in market_id = 3 in a time range\n    await client.get_trades(3, 1691669740, 1691583340, callback)\n\n    # subscribe to order book updates for market = 0; receives a new message every second(only for those prices where the quantity has changed)\n    async def on_message(ws, message):\n        print(f\"Received orderbook update: {message}\")\n\n    asyncio.run(client.subscribe_to_order_book_depth(0, callback=on_message))\n```\n\n---\n\n### GetPositionsResponse description\n- margin: trader's margin\n- reservedMargin: trader's reserved margin\n- positions: List[Position]\n\n### Position description\n- market: market id\n- openNotional: open notional\n- size: size of the position\n- unrealisedFunding: unrealised funding payment for this position\n- liquidationThreshold: max amount of size that can be liquidated at once\n- notionalPosition: notional position size\n- unrealisedProfit: unrealised profit on this position\n- marginFraction: margin/notional position\n- liquidationPrice: the asset price at which this position might get liquidated\n- markPrice: latest price of this market\n\n### OrderStatusResponse description\n- executedQty: executed quantity\n- orderId: order id\n- origQty: total order quantity\n- price: order price\n- reduceOnly: whether the order is reduce-only or not\n- postOnly: whether the order is post-only or not\n- positionSide: (LONG|SHORT)\n- status: order status (NEW|FILLED|CANCELED|REJECTED|PARTIALLY_FILLED)\n- symbol: market id\n- time: time when the order was placed in unix format\n- type: LIMIT_ORDER\n- updateTime: time when the order was last updated in unix format\n- salt: order salt\n\n## Orderbook depth feed\nOrderbook depth feed can be subscribed to using the `subscribe_to_order_book_depth` method. It emits the change in orderbook depth every second. It only emits the prices where the quantity has changed, but with absolute quantities(not the change).\n\n```python\nimport os\nfrom hubble_exchange import HubbleClient\n\nasync def main():\n    client = HubbleClient(os.getenv(\"PRIVATE_KEY\"))\n    # subscribe to market id 0\n    await client.subscribe_to_order_book_depth(0, callback=callback)\n```\n\n### OrderBookDepthUpdateResponse description\n- T: timestamp\n- symbol: market id\n- bids: list of [price, quantity] for bids\n- asks: list of [price, quantity] for asks\n\n\n## Trader feed\n\nAll order updates related to a particular trader can be subscribed to using the `subscribe_to_trader_updates` method.\nIt can be subscribed in 2 confirmation modes - head block or accepted block. Events received in head block mode are not finalised and can be reverted. When an event is removed from the chain, the client will receive a `removed=True` event. Events received in accepted block mode are finalised and will alwats have `removed=False`.\n\n```python\nimport os\nfrom hubble_exchange import HubbleClient, ConfirmationMode\n\nasync def main():\n    client = HubbleClient(os.getenv(\"PRIVATE_KEY\"))\n    await client.subscribe_to_trader_updates(ConfirmationMode.accepted, callback)\n```\n\n\n### TraderFeedUpdate description\n\n- Trader: address of the trader\n- OrderId: order id\n- OrderType: order type - \"limit\" order or \"ioc\" (market order)\n- Removed: whether the event is being removed or not\n- EventName: name of the contract event (OrderAccepted|OrderRejected|OrderMatched|OrderCancelAccepted|OrderCancelRejected)\n- Args: args is a dynamic field and it contains information about the event. For example, OrderPlaced event contains order object, OrderMatched event contains fillAmount and price\n- BlockNumber: block number in which the transaction was included\n- BlockStatus: (head|accepted) whether the block is accepted or only a preferred block(head block)\n- Timestamp: timestamp of the block in unix format\n- TransactionHash: transaction hash\n\n#### EventName description\n- OrderAccepted - Order was successfully placed\n- OrderRejected - Order was rejected\n- OrderMatched - Order was matched\n- OrderCancelAccepted - Order was successfully cancelled(only for Limit orders)\n- OrderCancelRejected - Order cancel was rejected(only for Limit orders)\n\nRemoved events are emmitted during chain reorgs, and are most likely to be temporary. They are only emmitted when subscribing to head block events. If removed=True, the client might need to do a reverse operation for the given event. For example if an OrderMatched event is received with removed=True, the client should add the fillAmount back to unfilled quantity of the order.\n\n## Market feed\n\nAll trades of a particular market can be subscribed to using the `subscribe_to_market_updates` method.\nSimilar to the trader feed, it has 2 confirmation modes - head block or accepted block.\n\n```python\nimport os\nfrom hubble_exchange import HubbleClient, ConfirmationMode\n\nasync def main():\n    client = HubbleClient(os.getenv(\"PRIVATE_KEY\"))\n    # subscribe to market id 0\n    await client.subscribe_to_market_updates(0, ConfirmationMode.accepted, callback)\n```\n\n### MarketFeedUpdate description\n\n- Trader: address of the trader who performed the trade\n- Market: market id\n- Size: size of the trade in decimals\n- Price: price at which it was executed\n- Removed: whether the event is being removed or not\n- EventName: name of the contract event (PositionModified)\n- BlockNumber: block number in which the transaction was included\n- BlockStatus: (head|accepted) whether the block is accepted or only a preferred block(head block)\n- Timestamp: timestamp of the block in unix format\n- TransactionHash: transaction hash\n\nRemoved events are emmitted during chain reorgs, and are most likely to be temporary. They are only emmitted when subscribing to head block events. If removed=True, the client might need to do a reverse operation for the given event.\n\n\n## Get limit order status\n\nGet the status of a limit order. It can be used to get the status of any order, even if it is not placed using the sdk.\n\n```python\norder_status = await client.get_limit_order_status(order.id, callback)\n```\n\n### Status Description\n\n```\nInvalid = order does not exist\nPlaced = order is unfilled\nPartiallyFilled = order is partially filled\nFilled = order is completely filled\nCancelled = cancelled\n```\n\n## Open orders\n\nReturns all open orders of the trader. It can be filtered by market id.\n\n```python\n# get all open orders\nopen_orders = await client.get_open_orders(None, callback)\n# get open orders for market_id = 0\nopen_orders = await client.get_open_orders(0, callback)\n```\n\n### OpenOrder description\n\n- Market: market id\n- Price: order price\n- Size: order size\n- FilledSize: filled size\n- Timestamp: timestamp of order place transaction in unix format\n- Salt: order salt\n- OrderId: order id\n- ReduceOnly: whether the order is reduce-only or not\n- PostOnly: whether the order is post-only or not\n- OrderType: order type - \"limit\" order or \"ioc\" (market order)\n\n\n## Get historical user trades\n\nReturns all trades of the trader. It can be filtered by market id.\n\n```python\n# get user trades in all markets in a time range\nawait client.get_trades(None, 1691669740, 1691583340, callback)\n# get user trades in market_id = 3 in a time range\nawait client.get_trades(3, 1691669740, 1691583340, callback)\n```\n\n### Trade description\n- BlockNumber: block number in which the order match transaction was included\n- TransactionHash: transaction hash\n- Market: market id\n- Timestamp: timestamp of the block in unix format\n- TradedAmount: amount of the trade in decimals\n- Price: price at which it was executed\n- RealizedPnl: realized pnl of the trade in USD\n- OpenNotional: open notional\n- ExecutionMode: execution mode - \"maker\" or \"taker\"\n\n## Candlestick data\n\nHistorical [candlestick data](https://www.investopedia.com/terms/o/ohlcchart.asp) can be fetched using the `get_candlestick_data` method using market and time range.\n\nIt can be obtained for the following timeframes:\n- 1m - 1 minute\n- 3m - 3 minutes\n- 5m - 5 minutes\n- 15m - 15 minutes\n- 30m - 30 minutes\n- 1h - 1 hour\n- 4h - 4 hours\n- 8h - 8 hours\n- 1d - 1 day\n- 1w - 1 week\n- 1M - 1 month\n\n```python\nfrom hubble_exchange import HubbleClient\n\n# get candlestick data for market id 0, timeframe = 5 minutes, start time = 1696932000, end time = 1696939200\nresponse = await client.get_candlesticks(0, \"5m\", 1696932000, 1696939200)\n```\n\n### CandlestickData description\nExample candlestick data:\n```json\n{\n    start: 1696500000,     // start of the time period in unix format\n    end: 1696500300,       // end of the time period in unix format\n    volume: 192267.282,    // volume in USD\n    open: 0.8745,          // open price for the duration\n    close: 0.8851,         // close price for the duration\n    high: 0.8947,          // highest price for the duration\n    low: 0.8675,           // lowest price for the duration\n}\n```\n\n## Historical funding rate\n\nHistorical funding rate can be queried using the `get_funding_rate` method using market and timestamp.\nFunding payment happens every hour at the end of the hour as per UTC time. For example, funding payment for 1pm to 2pm will happen at 2pm.\n\nIf the provided timestamp is not the end of an hour, the funding rate for the previous hour will be returned. For example, if the timestamp is 1:25pm, the funding rate for 1pm will be returned.\n\n```python\nfrom hubble_exchange import HubbleClient\n\n# get funding payments for market id 0, time = 1696932000\nfunding_rate = await client.get_funding_rate(0, 1696932000)\n```\n\n## Predicted funding rate\n\nPredicted funding rate can be queried using the `get_predicted_funding_rate` method using market. It returns the predicted funding rate for the current hour.\n\n```python\nfrom hubble_exchange import HubbleClient\n\n# get predicted funding rate for market id 2\nfunding_rate = await client.get_predicted_funding_rate(2)\n```\n\n## Historical open interest\n\nHistorical open interest can be queried using the `get_open_interest` method using market and timestamp. It returns the open interest in the units of the base asset. For example, if the market is ETH-Perp, the open interest will be in ETH.\n\n```python\nfrom hubble_exchange import HubbleClient\n\n# get open interest for market id 0, time = 1696932000\nopen_interest = await client.get_open_interest(0, 1696932000)\n```\n\n## Get nonce\n\nAn account nonce is a transaction counter in each account, which is used to prevent replay attacks. Nonce should be increased by 1 in every transaction. The function `get_nonce` fetches the number of transactions sent by the address from the blockchain.\n\nNonce can be fetched once using `get_nonce` and be incremented in every subsequent transaction. However, if the same address is sending multiple transactions in parallel, it is recommended to use `get_nonce` before every transaction.\n\n```python\nnonce = await client.get_nonce()\n# use this nonce in the next transaction like this\nawait client.place_limit_orders(orders, True, callback, {\"nonce\": nonce})\n```\n\n## Custom transaction options\n\nThe following options can be passed to the client to override the default\n\n```python\n{\n    \"nonce\": Nonce,\n    \"gas\": int,\n    \"maxFeePerGas\": Union[str, Wei],\n    \"maxPriorityFeePerGas\": Union[str, Wei],\n},\n```\n\nIt can be used for `place_limit_orders`, `cancel_limit_orders`, `cancel_order_by_id` methods.\nExample:\n```python\n\nfrom web3 import Web3\n\nclient = HubbleClient(os.getenv(\"PRIVATE_KEY\"))\nplaced_orders = await client.place_limit_orders(orders, callback, {\n    \"gas\": 500_000,\n    \"maxFeePerGas\": Web3.to_wei(80, 'gwei'),\n    \"maxPriorityFeePerGas\": Web3.to_wei(20, 'gwei'),\n})\n```\n\n## Transaction modes\n\nThere are different modes in which the client can wait for acknowledgement of the transaction. The default behaviour is to send the transaction and not wait for the acknowledgement.\nThis can be changed by explicitly asking the function to wait while sending the trasaction.\n\n- TransactionMode.no_wait: The default behaviour is to send transactions to the blockchain and NOT wait for the acknowledgement.\n- TransactionMode.wait_for_head: Wait for the transaction to be included in the canonical chain. At this time the block is preferred but not yet finalized. However, once the block in included in the canonical chain, the matching engine will start processing the order.\n- TransactionMode.wait_for_accept: Wait for the transaction to be finalised.\n\nExample:\n```python\nfrom hubble_exchange import TransactionMode\nclient = HubbleClient(os.getenv(\"PRIVATE_KEY\"))\nplaced_orders = await client.place_limit_orders(orders, callback, mode=TransactionMode.wait_for_accept)\n\n# or set the default mode for all transactions\n\nclient.set_transaction_mode(TransactionMode.wait_for_head)\n```\n\n---\n\n## Waiting for response in place_limit_orders and cancel_limit_orders\n\nThe `place_limit_orders`, `place_ioc_orders`, and `cancel_limit_orders` methods can be called in 2 modes - wait for response or don't wait for response.\nTo get the acknowledgement of the transaction, use `wait_for_response=True`. The response will be a list of dicts with order ids and success boolean. Waiting for response will be slower because this can be confirmed only after the transaction is mined(accepted).\nWhen using `wait_for_response=True`, the sdk will automatically set the transaction mode to `TransactionMode.wait_for_accept` because the response can be confirmed only after the transaction is mined.\n\nAlternatively, the client can also use trader feed to listen to all the updates. This is faster when done with ConfirmationMode.head\n\n\n## Config\n\n- mininum_quantity: minimum quantity that can be placed in an order for a particular market. The quantity also has to be a multiple of this number.\n- price_precision: number of decimal places for price for a particular market.\n\n```python\nfrom hubble_exchange.constants import get_minimum_quantity, get_price_precision\n\n# get minimum quantity for market id 3\nmin_qty = get_minimum_quantity(3)\n# get price precision for market id 3\nprice_precision = get_price_precision(3)\n```\n",
    "bugtrack_url": null,
    "license": "MIT License  Copyright (c) 2023 Hubble Exchange  Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the \"Software\"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:  The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.  THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. ",
    "summary": "Official python SDK for Hubble Exchange",
    "version": "0.8.8",
    "project_urls": {
        "homepage": "https://github.com/hubble-exchange/python-sdk"
    },
    "split_keywords": [
        "hubble",
        "exchange",
        "orderbook",
        "perpetual",
        "futures",
        "dex",
        "sdk",
        "python",
        "avalanche",
        "ethereum",
        "web3",
        "crypto"
    ],
    "urls": [
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "ca692412e6e88371afc948843fe912c9fbc6489ffafd4527de2dedc60345eb5c",
                "md5": "83072589efc0070b17cab2b686dc0d1e",
                "sha256": "619012a6afc2c918373e67b9692efffca1414dfef80df0a3ce9be6fef7f960e8"
            },
            "downloads": -1,
            "filename": "hubble_exchange-0.8.8-py3-none-any.whl",
            "has_sig": false,
            "md5_digest": "83072589efc0070b17cab2b686dc0d1e",
            "packagetype": "bdist_wheel",
            "python_version": "py3",
            "requires_python": ">=3.7",
            "size": 33959,
            "upload_time": "2023-11-30T13:17:18",
            "upload_time_iso_8601": "2023-11-30T13:17:18.698803Z",
            "url": "https://files.pythonhosted.org/packages/ca/69/2412e6e88371afc948843fe912c9fbc6489ffafd4527de2dedc60345eb5c/hubble_exchange-0.8.8-py3-none-any.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "8a42aba2187615f87117897cba6129ae63a40021ace710a5628ea549304b90ac",
                "md5": "a0771706b396e68dae011399886a1ded",
                "sha256": "5fa4ed91098dc1d9f31ba2d0a3c66470420c41c6022d0c2a8d6641e86482fbef"
            },
            "downloads": -1,
            "filename": "hubble_exchange-0.8.8.tar.gz",
            "has_sig": false,
            "md5_digest": "a0771706b396e68dae011399886a1ded",
            "packagetype": "sdist",
            "python_version": "source",
            "requires_python": ">=3.7",
            "size": 31444,
            "upload_time": "2023-11-30T13:17:20",
            "upload_time_iso_8601": "2023-11-30T13:17:20.418532Z",
            "url": "https://files.pythonhosted.org/packages/8a/42/aba2187615f87117897cba6129ae63a40021ace710a5628ea549304b90ac/hubble_exchange-0.8.8.tar.gz",
            "yanked": false,
            "yanked_reason": null
        }
    ],
    "upload_time": "2023-11-30 13:17:20",
    "github": true,
    "gitlab": false,
    "bitbucket": false,
    "codeberg": false,
    "github_user": "hubble-exchange",
    "github_project": "python-sdk",
    "travis_ci": false,
    "coveralls": false,
    "github_actions": false,
    "lcname": "hubble-exchange"
}
        
Elapsed time: 0.26090s