xumm-sdk-py


Namexumm-sdk-py JSON
Version 1.0.4 PyPI version JSON
download
home_pagehttps://github.com/XRPL-Labs/xumm-sdk-py
SummaryXumm SDK for Python
upload_time2023-05-17 12:30:44
maintainer
docs_urlNone
authorXRPL-Labs
requires_python
licenseMIT
keywords xrp ledger ripple xumm sdk
VCS
bugtrack_url
requirements No requirements were recorded.
Travis-CI No Travis.
coveralls test coverage No coveralls.
            # XUMM SDK (Python) [![python version](https://badge.fury.io/py/xumm-sdk-py.svg)](https://pypi.org/project/xumm-sdk-py/) [![GitHub Actions Python status](https://github.com/XRPL-Labs/xumm-sdk-py/workflows/Python/badge.svg?branch=main)](https://github.com/XRPL-Labs/xumm-sdk-py/actions)

Interact with the XUMM SDK from Python environments.

#### **⚠️ Please note: The XUMM SDK (XUMM API in general) is for BACKEND USE only. Please DO NOT use your API credentials in a FRONTEND environment.**

> To implement the XUMM SDK (or XUMM API directly) in your own web project, make sure your frontend calls your own backend, where the follow up communication with the XUMM SDK (or XUMM API) will take place. Your XUMM credentials should never be publicly available.

- [Getting Started](#getting-started)
- [Usage](#usage)
  - [Payloads](#payloads)
  - [App Storage](#app-storage)
  - [Helper Methods](#helper-methods)
- [Development](#development)

## Getting Started

### Installation

Via pip:

```bash
pip3 install xumm-sdk-py
```

## Usage

```python
import xumm

sdk = xumm.XummSdk()
# Or with manually provided credentials (instead of using dotenv):
sdk = xumm.XummSdk('XUMM_APIKEY', 'XUMM_APISECRET')
```

After constructing the SDK, you can call the methods:

- `sdk.*` for the helper methods (see below)
- `sdk.payload.*` to get/update/create payloads for users to sign
- `sdk.storage.*` for your XUMM app storage (to store meta info for headless applications)

The SDK will look in your environment or dotenv file (`.env`) for the `XUMM_APIKEY` and `XUMM_APISECRET` values. A sample dotenv can be found [here](https://github.com/XRPL-Labs/xumm-sdk-py/blob/main/.env.sample). Alternatively you can provide your XUMM API Key & Secret by passing them to the XummSdk constructor.

> NOTE: If both your environment and the SDK constructor contain credentials, the values provided to the constructor will be used.

### Payloads

#### Intro

Payloads are the primary reason for the XUMM API (thus this SDK) to exist. The [XUMM API Docs explain '**Payloads**'](https://xumm.readme.io/docs/introduction) like this:

> An XRPL transaction "template" can be posted to the XUMM API. Your transaction tample to sign (so: your "sign request") will be persisted at the XUMM API backend. We now call it a a **Payload**. XUMM app user(s) can open the Payload (sign request) by scanning a QR code, opening deeplink or receiving push notification and resolve (reject or sign) on their own device.

A payload can contain an XRPL transaction template. Some properties may be omitted, as they will be added by the XUMM app when a user signs a transaction. A simple payload may look like this:

```python
payload = {
  'txjson': {
    'TransactionType' : 'Payment',
    'Destination' : 'rwiETSee2wMz3SBnAG8hkMsCgvGy9LWbZ1',
    'Amount': '1337'
  }
}
```

As you can see the payload looks like a regular XRPL transaction, wrapped in an `txjson` object, omitting the mandatory `Account`, `Fee` and `Sequence` properties. They will be added containing the correct values when the payload is signed by an app user.

Optionally (besides `txjson`) a payload can contain these properties ([PY definition](https://github.com/XRPL-Labs/xumm-sdk-py/blob/main/xumm/resource/types/xumm_api/__init__.py#L836)):

- `options` to define payload options like a return URL, expiration, etc.
- `custom_meta` to add metadata, user insruction, your own unique ID, ...
- `user_token` to push the payload to a user (after [obtaining a user specific token](https://xumm.readme.io/docs/pushing-sign-requests))

A more complex payload [could look like this](https://gist.github.com/WietseWind/ecdfd58bece14e5d15e41138fa4b0f4a). A [reference for payload options & custom meta](https://xumm.readme.io/reference/post-payload) can be found in the [API Docs](https://xumm.readme.io/reference/post-payload).

Instead of providing a `txjson` transaction, a transaction formatted as HEX blob (string) can be provided in a `txblob` property.

##### sdk.payload.get

```python
sdk.payload.get(
  payload: Union[str, CreatedPayload]
): -> XummPayload
```

To get payload details, status and if resolved & signed: results (transaction, transaction hash, etc.) you can `get()` a payload.

Note! Please don't use _polling_! The XUMM API offers Webhooks (configure your Webhook endpoint in the [Developer Console](https://apps.xumm.dev)) or use [a subscription](#payload-subscriptions-live-updates) to receive live payload updates (for non-SDK users: [Webhooks](https://xumm.readme.io/docs/payload-status)).

You can `get()` a payload by:

- Payload UUID

  ```python
  payload = sdk.payload.get('aaaaaaaa-bbbb-cccc-dddd-1234567890ab')
  ```

- Passing a created Payload object (see: [sdk.payload.create](#sdkpayloadcreate))
  ```python
  new_payload: xumm_types.created_payload = {txjson: {...}}
  created = sdk.payload.create(new_payload)
  payload = sdk.payload.get(created)
  ```

```python
sdk.payload.get('aaaaaaaa-bbbb-cccc-dddd-1234567890ab')
```

##### sdk.payload.create

```python
sdk.payload.create (
payload: create_payload
): -> Union[CreatedPayload, None]
```

To create a payload, a `txjson` XRPL transaction can be provided. Alternatively, a transaction formatted as HEX blob (string) can be provided in a `txblob` property. **See the [intro](#intro) for more information about payloads.** Take a look at the [Developer Docs for more information about payloads](https://xumm.readme.io/docs/your-first-payload).

The response (see: [Developer Docs](https://xumm.readme.io/docs/payload-response-resources)) of a `sdk.payload.create()` operation, a `<CreatedPayload>` json object, looks like this:

```json
{
  "uuid": "1289e9ae-7d5d-4d5f-b89c-18633112ce09",
  "next": {
    "always": "https://xumm.app/sign/1289e9ae-7d5d-4d5f-b89c-18633112ce09",
    "no_push_msg_received": "https://xumm.app/sign/1289e9ae-7d5d-4d5f-b89c-18633112ce09/qr"
  },
  "refs": {
    "qr_png": "https://xumm.app/sign/1289e9ae-7d5d-4d5f-b89c-18633112ce09_q.png",
    "qr_matrix": "https://xumm.app/sign/1289e9ae-7d5d-4d5f-b89c-18633112ce09_q.json",
    "qr_uri_quality_opts": ["m", "q", "h"],
    "websocket_status": "wss://xumm.app/sign/1289e9ae-7d5d-4d5f-b89c-18633112ce09"
  },
  "pushed": true
}
```

The `next.always` URL is the URL to send the end user to, to scan a QR code or automatically open the XUMM app (if on mobile). If a `user_token` has been provided as part of the payload data provided to `sdk.payload.create()`, you can see if the payload has been pushed to the end user. A button "didn't receive a push notification" could then take the user to the `next.no_push_msg_received` URL.

Alternatively user routing / instruction flows can be custom built using the QR information provided in the `refs` object, and a subscription for live status updates (opened, signed, etc.) using a WebSocket client can be setup by conneting to the `refs.websocket_status` URL. **Please note: this SDK already offers subscriptions. There's no need to setup your own WebSocket client, see [Payload subscriptions: live updates](#payload-subscriptions-live-updates).** There's more information about the [payload workflow](https://xumm.readme.io/docs/payload-workflow) and a [payload lifecycle](https://xumm.readme.io/docs/doc-payload-life-cycle) in the Developer Docs.

##### sdk.payload.cancel

```python
sdk.payload.cancel(
  payload: Union[str, XummPayload, CreatedPayload]
): -> Union[DeletedPayload, None]
```

To cancel a payload, provide a payload UUID (string), a `<XummPayload>` (by performing a `sdk.payload.get()` first) or a `<CreatedPayload>` (by using the response of a `sdk.payload.create()` call). By cancelling an existing payload, the payload will be marked as expired and can no longer be opened by users.

**Please note**: _if a user already opened the payload in XUMM APP, the payload cannot be cancelled: the user may still be resolving the payload in the XUMM App, and should have a chance to complete that process_.

A response (generic API types [here](https://github.com/XRPL-Labs/xumm-sdk-py/blob/main/xumm/resource/types/xumm_api/__init__.py)) looks like:

```python
response.result.cancelled  # bool
response.result.reason  # XummCancelReason
response.meta  # XummPayloadMeta
response.custom_meta  # XummCustomMeta
```

#### Payload subscriptions: live updates

To subscribe to live payload status updates, the XUMM SDK can setup a WebSocket connection and monitor live status events. Emitted events include:

- The payload is opened by a XUMM App user (webpage)
- The payload is opened by a XUMM App user (in the app)
- Payload expiration updates (remaining time in seconds)
- The payload was resolved by rejecting
- The payload was resolved by accepting (signing)

More information about the status update events & sample event data [can be found in the Developer Docs](https://xumm.readme.io/docs/payload-status).

Status updates can be processed by providing a _callback function_ to the `sdk.payload.subscribe()` method. Alternatively, the (by the `sdk.payload_subscribe()` method) returned raw websocket can be used to listen for WebSocket `onmessage` events.

The subscription will be closed by either:

- Returning non-void in the _callback function_ passed to the `sdk.payload.subscribe()` method
- Manually calling `<PayloadSubscription>.resolve()` on the object returned by the `sdk.payload.subscribe()` method

##### sdk.payload.subscribe

```python
sdk.payload.subscribe(
  payload: Union[str, XummPayload, CreatedPayload],
  callback: on_payload_event
): -> PayloadSubscription
```

If a callback function is not provided, the subscription will stay active until the `<PayloadSubscription>.resolve()` method is called manually, eg. based on handling `<PayloadSubscription>.websocket.onmessage` events.

When a callback function is provided, for every paylaod specific event the callback function will be called with [`<SubscriptionCallbackParams>`](https://github.com/XRPL-Labs/xumm-sdk-py/blob/main/xumm/resource/types/payload/subscription_callback_params.py). The `<SubscriptionCallbackParams>.data` property contains parsed JSON containing event information. Either by calling `<SubscriptionCallbackParams>.resolve()` or by returning a non-void value in the _callback function_ the subscription will be ended, and the `<PayloadSubscription>.resolved` promise will resolve with the value returned or passed to the `<SubscriptionCallbackParams>.resolve()` method.

Resolving (by returning non-void in the callback or calling `resolve()` manually) closes the WebSocket client the XUMM SDK sets up 'under the hood'.

The [`<PayloadSubscription>`](https://github.com/XRPL-Labs/xumm-sdk-py/blob/main/xumm/resource/types/payload/payload_subscription.py) object looks like this:

```python
response.payload  # XummPayload
response.resolved  # Union[CallbackPromise, None]
response.resolve  # CallbackPromise
response.websocket  # WSClient
```

Examples:

- [Async process after returning data in the callback function](https://github.com/XRPL-Labs/xumm-sdk-py/blob/main/samples/ws/async_callback.py)
- [Await based on returning data in the callback function](https://github.com/XRPL-Labs/xumm-sdk-py/blob/main/samples/ws/await_callback.py)
- [Await based on resolving a callback event](https://github.com/XRPL-Labs/xumm-sdk-py/blob/main/samples/ws/await_event.py)
- [Await based on resolving without using a callback function](https://github.com/XRPL-Labs/xumm-sdk-py/blob/main/samples/ws/await_no_callback.py)

##### sdk.payload.create_subscribe

```python
sdk.payload.create_and_subscribe(
  payload: CreatePayload,
  callback: on_payload_event
): -> PayloadAndSubscription
```

The [`<PayloadAndSubscription>`](https://github.com/XRPL-Labs/xumm-sdk-py/blob/main/xumm/resource/types/payload/payload_and_subscription.py) object is basically a [`<PayloadSubscription>`](https://github.com/XRPL-Labs/xumm-sdk-py/blob/main/xumm/resource/types/payload/payload_subscription.py) object with the created payload results in the `created` property:

All information that applies on [`sdk.payload.create()`](#sdkpayloadcreate) and [`sdk.payload.create_and_subscribe()`](#sdkpayloadsubscribe) applies. Differences are:

1. The input for a `sdk.payload.create_and_subscribe()` call isn't a payload UUID / existing payload, but a paykiad to create.
2. The response object also contains (`<PayloadAndSubscription>.created`) the response obtained when creating the payload
3.

#### App Storage

App Storage allows you to store a JSON object at the XUMM API platform, containing max 60KB of data.
Your XUMM APP storage is stored at the XUMM API backend, meaning it persists until you overwrite or delete it.

This data is private, and accessible only with your own API credentials. This private JSON data can be used to store credentials / config / bootstrap info / ... for your headless application (eg. POS device).

```python
storage_set = await sdk.storage.set({'name': 'Wietse', 'age': 32, 'male': True})
print(storage_set)
# True
```

```python
storage_get = sdk.storage.get()
print(storage_get.data)
# { 'name': 'Wietse', 'age': 32, 'male': True }
```

```python
storage_delete = sdk.storage.delete()
print(storage_delete)
# True
```

```python
storage_get_after_delete = sdk.storage.get()
print(storage_get_after_delete.data)
# None
```

#### Helper methods

##### sdk.ping()

The `ping` method allows you to verify API access (valid credentials) and returns some info on your XUMM APP:

```python
pong = sdk.ping()
```

Returns [`<ApplicationDetails>`](https://github.com/XRPL-Labs/xumm-sdk-py/blob/main/xumm/resource/types/meta/application_details.py#L294):

```python
pong.quota  # {}
pong.application.name  # 'My XUMM APP'
pong.application.uuidv4  # '00000000-1111-2222-3333-aaaaaaaaaaaa'
pong.application.webhookurl  # ''
pong.application.disabled  # 0
pong.call.uuidv4  # 'bbbbbbbb-cccc-dddd-eeee-111111111111'
```

##### sdk.get_curated_assets()

The `get_curated_assets` method allows you to get the list of trusted issuers and IOU's. This is the same list used to
populate the "Add Asset" button at the XUMM home screan.

```python
curated_assets = sdk.get_curated_assets()
```

Returns [`<CuratedAssetsResponse>`](https://github.com/XRPL-Labs/xumm-sdk-py/blob/main/xumm/resource/types/meta/curated_assets_response.py#L426):

```python
curated_assets.issuers  # [ 'Bitstamp', 'GateHub' ]
curated_assets.currencies  # [ 'USD', 'BTC', 'EUR', 'ETH' ]
curated_assets.details.Bitstamp   # {}
curated_assets.details.GateHub   # {}
```

##### sdk.get_kyc_status()

The `get_kyc_status` return the KYC status of a user based on a user_token, issued after the
user signed a Sign Request (from your app) before (see Payloads - Intro).

If a user token specified is invalid, revoked, expired, etc. the method will always
return `NONE`, just like when a user didn't go through KYC. You cannot distinct a non-KYC'd user
from an invalid token.

Alternatively, KYC status can be retrieved for an XPRL account address: the address selected in
XUMM when the session KYC was initiated by.

```python
kyc_status = sdk.get_kyc_status('00000000-0000-0000-0000-000000000000')
```

... or using an account address:

```python
kyc_status = sdk.get_kyc_status('rwu1dgaUq8DCj3ZLFXzRbc1Aco5xLykMMQ')
```

Returns [`<str of PossibleKycStatuses>`](https://github.com/XRPL-Labs/xumm-sdk-py/blob/main/xumm/resource/types/meta/kyc_status_response.py#L66).

###### Notes on KYC information

- Once an account has successfully completed the XUMM KYC flow, the KYC flag will be applied to the account even if the identity document used to KYC expired. The flag shows that the account was **once** KYC'd by a real person with a real identity document.
- Please note that the KYC flag provided by XUMM can't be seen as a "all good, let's go ahead" flag: it should be used as **one of the data points** to determine if an account can be trusted. There are situations where the KYC flag is still `True`, but an account can no longer be trusted. Eg. when account keys are compromised and the account is now controlled by a 3rd party. While unlikely, depending on the level of trust required for your application you may want to mitigate against these kinds of fraud.

##### sdk.get_transaction()

The `get_transaction` method allows you to get the transaction outcome (mainnet)
live from the XRP ledger, as fetched for you by the XUMM backend.

> **Note**: it's best to retrieve these results **yourself** instead of relying on the XUMM platform to get live XRPL transaction information! You can use the **[xrpl-py](https://pypi.org/project/xrpl-py)** package to do this:

```python
tx_info = sdk.get_transaction(tx_hash)
```

Returns: [`<XrplTransaction>`](https://github.com/XRPL-Labs/xumm-sdk-py/blob/main/xumm/resource/types/meta/xrpl_transaction.py#L114).

##### sdk.verify_user_tokens() / sdk.verify_user_tokens()

The `verify_user_tokens` (or single token: `verify_user_token`) method allows you to verify one or more User Tokens obtained from previous sign requests. This allows you to detect if you will be able to push your next Sign Request to specific users.

```python
some_token = '691d5ae8-968b-44c8-8835-f25da1214f35'
token_validity = sdk.verify_user_tokens([
  some_token,
  'b12b59a8-83c8-4bc0-8acb-1d1d743871f1',
  '51313be2-5887-4ae8-9fda-765775a59e51'
])
if sdk.verify_user_token(some_token).active:
  # Push, use `user_token` in payload
else:
  # QR or Redirect (deeplink) flow
```

Returns: [`<UserTokenValidity or [UserTokenValidity]>`](https://github.com/XRPL-Labs/xumm-sdk-py/blob/main/xumm/resource/types/meta/user_tokens.py#L8).

## Development

### Install requirments

```bash
pip install -e ".[develop]"
```

### Build

Please note: at least Python version **3.6+** is required!

To build the code, run `python setup.py install`.

### Debugging

The XUMM SDK will emit debugging info when invoked with a debug environment variable configured like: `DEBUG=xumm-sdk*`

You'll see the XUMM SDK debug messages if you invoke your script instead of this:

```
python3 main.py
```

Example:

```
DEBUG=xumm-sdk* python3 main.py
```

### Lint & test

Lint the code using:

```bash
python3 -m flake8
```

For running tests:

```bash
python3 test.py tests/
```

            

Raw data

            {
    "_id": null,
    "home_page": "https://github.com/XRPL-Labs/xumm-sdk-py",
    "name": "xumm-sdk-py",
    "maintainer": "",
    "docs_url": null,
    "requires_python": "",
    "maintainer_email": "",
    "keywords": "xrp,ledger,ripple,xumm,sdk",
    "author": "XRPL-Labs",
    "author_email": "support@xrpl-labs.com",
    "download_url": "https://files.pythonhosted.org/packages/f2/91/e2f44d07f6e8e163ecd280139b59b27b7faaf4ef9c3a61ecfa1c85f3825f/xumm-sdk-py-1.0.4.tar.gz",
    "platform": null,
    "description": "# XUMM SDK (Python) [![python version](https://badge.fury.io/py/xumm-sdk-py.svg)](https://pypi.org/project/xumm-sdk-py/) [![GitHub Actions Python status](https://github.com/XRPL-Labs/xumm-sdk-py/workflows/Python/badge.svg?branch=main)](https://github.com/XRPL-Labs/xumm-sdk-py/actions)\n\nInteract with the XUMM SDK from Python environments.\n\n#### **\u26a0\ufe0f Please note: The XUMM SDK (XUMM API in general) is for BACKEND USE only. Please DO NOT use your API credentials in a FRONTEND environment.**\n\n> To implement the XUMM SDK (or XUMM API directly) in your own web project, make sure your frontend calls your own backend, where the follow up communication with the XUMM SDK (or XUMM API) will take place. Your XUMM credentials should never be publicly available.\n\n- [Getting Started](#getting-started)\n- [Usage](#usage)\n  - [Payloads](#payloads)\n  - [App Storage](#app-storage)\n  - [Helper Methods](#helper-methods)\n- [Development](#development)\n\n## Getting Started\n\n### Installation\n\nVia pip:\n\n```bash\npip3 install xumm-sdk-py\n```\n\n## Usage\n\n```python\nimport xumm\n\nsdk = xumm.XummSdk()\n# Or with manually provided credentials (instead of using dotenv):\nsdk = xumm.XummSdk('XUMM_APIKEY', 'XUMM_APISECRET')\n```\n\nAfter constructing the SDK, you can call the methods:\n\n- `sdk.*` for the helper methods (see below)\n- `sdk.payload.*` to get/update/create payloads for users to sign\n- `sdk.storage.*` for your XUMM app storage (to store meta info for headless applications)\n\nThe SDK will look in your environment or dotenv file (`.env`) for the `XUMM_APIKEY` and `XUMM_APISECRET` values. A sample dotenv can be found [here](https://github.com/XRPL-Labs/xumm-sdk-py/blob/main/.env.sample). Alternatively you can provide your XUMM API Key & Secret by passing them to the XummSdk constructor.\n\n> NOTE: If both your environment and the SDK constructor contain credentials, the values provided to the constructor will be used.\n\n### Payloads\n\n#### Intro\n\nPayloads are the primary reason for the XUMM API (thus this SDK) to exist. The [XUMM API Docs explain '**Payloads**'](https://xumm.readme.io/docs/introduction) like this:\n\n> An XRPL transaction \"template\" can be posted to the XUMM API. Your transaction tample to sign (so: your \"sign request\") will be persisted at the XUMM API backend. We now call it a a **Payload**. XUMM app user(s) can open the Payload (sign request) by scanning a QR code, opening deeplink or receiving push notification and resolve (reject or sign) on their own device.\n\nA payload can contain an XRPL transaction template. Some properties may be omitted, as they will be added by the XUMM app when a user signs a transaction. A simple payload may look like this:\n\n```python\npayload = {\n  'txjson': {\n    'TransactionType' : 'Payment',\n    'Destination' : 'rwiETSee2wMz3SBnAG8hkMsCgvGy9LWbZ1',\n    'Amount': '1337'\n  }\n}\n```\n\nAs you can see the payload looks like a regular XRPL transaction, wrapped in an `txjson` object, omitting the mandatory `Account`, `Fee` and `Sequence` properties. They will be added containing the correct values when the payload is signed by an app user.\n\nOptionally (besides `txjson`) a payload can contain these properties ([PY definition](https://github.com/XRPL-Labs/xumm-sdk-py/blob/main/xumm/resource/types/xumm_api/__init__.py#L836)):\n\n- `options` to define payload options like a return URL, expiration, etc.\n- `custom_meta` to add metadata, user insruction, your own unique ID, ...\n- `user_token` to push the payload to a user (after [obtaining a user specific token](https://xumm.readme.io/docs/pushing-sign-requests))\n\nA more complex payload [could look like this](https://gist.github.com/WietseWind/ecdfd58bece14e5d15e41138fa4b0f4a). A [reference for payload options & custom meta](https://xumm.readme.io/reference/post-payload) can be found in the [API Docs](https://xumm.readme.io/reference/post-payload).\n\nInstead of providing a `txjson` transaction, a transaction formatted as HEX blob (string) can be provided in a `txblob` property.\n\n##### sdk.payload.get\n\n```python\nsdk.payload.get(\n  payload: Union[str, CreatedPayload]\n): -> XummPayload\n```\n\nTo get payload details, status and if resolved & signed: results (transaction, transaction hash, etc.) you can `get()` a payload.\n\nNote! Please don't use _polling_! The XUMM API offers Webhooks (configure your Webhook endpoint in the [Developer Console](https://apps.xumm.dev)) or use [a subscription](#payload-subscriptions-live-updates) to receive live payload updates (for non-SDK users: [Webhooks](https://xumm.readme.io/docs/payload-status)).\n\nYou can `get()` a payload by:\n\n- Payload UUID\n\n  ```python\n  payload = sdk.payload.get('aaaaaaaa-bbbb-cccc-dddd-1234567890ab')\n  ```\n\n- Passing a created Payload object (see: [sdk.payload.create](#sdkpayloadcreate))\n  ```python\n  new_payload: xumm_types.created_payload = {txjson: {...}}\n  created = sdk.payload.create(new_payload)\n  payload = sdk.payload.get(created)\n  ```\n\n```python\nsdk.payload.get('aaaaaaaa-bbbb-cccc-dddd-1234567890ab')\n```\n\n##### sdk.payload.create\n\n```python\nsdk.payload.create (\npayload: create_payload\n): -> Union[CreatedPayload, None]\n```\n\nTo create a payload, a `txjson` XRPL transaction can be provided. Alternatively, a transaction formatted as HEX blob (string) can be provided in a `txblob` property. **See the [intro](#intro) for more information about payloads.** Take a look at the [Developer Docs for more information about payloads](https://xumm.readme.io/docs/your-first-payload).\n\nThe response (see: [Developer Docs](https://xumm.readme.io/docs/payload-response-resources)) of a `sdk.payload.create()` operation, a `<CreatedPayload>` json object, looks like this:\n\n```json\n{\n  \"uuid\": \"1289e9ae-7d5d-4d5f-b89c-18633112ce09\",\n  \"next\": {\n    \"always\": \"https://xumm.app/sign/1289e9ae-7d5d-4d5f-b89c-18633112ce09\",\n    \"no_push_msg_received\": \"https://xumm.app/sign/1289e9ae-7d5d-4d5f-b89c-18633112ce09/qr\"\n  },\n  \"refs\": {\n    \"qr_png\": \"https://xumm.app/sign/1289e9ae-7d5d-4d5f-b89c-18633112ce09_q.png\",\n    \"qr_matrix\": \"https://xumm.app/sign/1289e9ae-7d5d-4d5f-b89c-18633112ce09_q.json\",\n    \"qr_uri_quality_opts\": [\"m\", \"q\", \"h\"],\n    \"websocket_status\": \"wss://xumm.app/sign/1289e9ae-7d5d-4d5f-b89c-18633112ce09\"\n  },\n  \"pushed\": true\n}\n```\n\nThe `next.always` URL is the URL to send the end user to, to scan a QR code or automatically open the XUMM app (if on mobile). If a `user_token` has been provided as part of the payload data provided to `sdk.payload.create()`, you can see if the payload has been pushed to the end user. A button \"didn't receive a push notification\" could then take the user to the `next.no_push_msg_received` URL.\n\nAlternatively user routing / instruction flows can be custom built using the QR information provided in the `refs` object, and a subscription for live status updates (opened, signed, etc.) using a WebSocket client can be setup by conneting to the `refs.websocket_status` URL. **Please note: this SDK already offers subscriptions. There's no need to setup your own WebSocket client, see [Payload subscriptions: live updates](#payload-subscriptions-live-updates).** There's more information about the [payload workflow](https://xumm.readme.io/docs/payload-workflow) and a [payload lifecycle](https://xumm.readme.io/docs/doc-payload-life-cycle) in the Developer Docs.\n\n##### sdk.payload.cancel\n\n```python\nsdk.payload.cancel(\n  payload: Union[str, XummPayload, CreatedPayload]\n): -> Union[DeletedPayload, None]\n```\n\nTo cancel a payload, provide a payload UUID (string), a `<XummPayload>` (by performing a `sdk.payload.get()` first) or a `<CreatedPayload>` (by using the response of a `sdk.payload.create()` call). By cancelling an existing payload, the payload will be marked as expired and can no longer be opened by users.\n\n**Please note**: _if a user already opened the payload in XUMM APP, the payload cannot be cancelled: the user may still be resolving the payload in the XUMM App, and should have a chance to complete that process_.\n\nA response (generic API types [here](https://github.com/XRPL-Labs/xumm-sdk-py/blob/main/xumm/resource/types/xumm_api/__init__.py)) looks like:\n\n```python\nresponse.result.cancelled  # bool\nresponse.result.reason  # XummCancelReason\nresponse.meta  # XummPayloadMeta\nresponse.custom_meta  # XummCustomMeta\n```\n\n#### Payload subscriptions: live updates\n\nTo subscribe to live payload status updates, the XUMM SDK can setup a WebSocket connection and monitor live status events. Emitted events include:\n\n- The payload is opened by a XUMM App user (webpage)\n- The payload is opened by a XUMM App user (in the app)\n- Payload expiration updates (remaining time in seconds)\n- The payload was resolved by rejecting\n- The payload was resolved by accepting (signing)\n\nMore information about the status update events & sample event data [can be found in the Developer Docs](https://xumm.readme.io/docs/payload-status).\n\nStatus updates can be processed by providing a _callback function_ to the `sdk.payload.subscribe()` method. Alternatively, the (by the `sdk.payload_subscribe()` method) returned raw websocket can be used to listen for WebSocket `onmessage` events.\n\nThe subscription will be closed by either:\n\n- Returning non-void in the _callback function_ passed to the `sdk.payload.subscribe()` method\n- Manually calling `<PayloadSubscription>.resolve()` on the object returned by the `sdk.payload.subscribe()` method\n\n##### sdk.payload.subscribe\n\n```python\nsdk.payload.subscribe(\n  payload: Union[str, XummPayload, CreatedPayload],\n  callback: on_payload_event\n): -> PayloadSubscription\n```\n\nIf a callback function is not provided, the subscription will stay active until the `<PayloadSubscription>.resolve()` method is called manually, eg. based on handling `<PayloadSubscription>.websocket.onmessage` events.\n\nWhen a callback function is provided, for every paylaod specific event the callback function will be called with [`<SubscriptionCallbackParams>`](https://github.com/XRPL-Labs/xumm-sdk-py/blob/main/xumm/resource/types/payload/subscription_callback_params.py). The `<SubscriptionCallbackParams>.data` property contains parsed JSON containing event information. Either by calling `<SubscriptionCallbackParams>.resolve()` or by returning a non-void value in the _callback function_ the subscription will be ended, and the `<PayloadSubscription>.resolved` promise will resolve with the value returned or passed to the `<SubscriptionCallbackParams>.resolve()` method.\n\nResolving (by returning non-void in the callback or calling `resolve()` manually) closes the WebSocket client the XUMM SDK sets up 'under the hood'.\n\nThe [`<PayloadSubscription>`](https://github.com/XRPL-Labs/xumm-sdk-py/blob/main/xumm/resource/types/payload/payload_subscription.py) object looks like this:\n\n```python\nresponse.payload  # XummPayload\nresponse.resolved  # Union[CallbackPromise, None]\nresponse.resolve  # CallbackPromise\nresponse.websocket  # WSClient\n```\n\nExamples:\n\n- [Async process after returning data in the callback function](https://github.com/XRPL-Labs/xumm-sdk-py/blob/main/samples/ws/async_callback.py)\n- [Await based on returning data in the callback function](https://github.com/XRPL-Labs/xumm-sdk-py/blob/main/samples/ws/await_callback.py)\n- [Await based on resolving a callback event](https://github.com/XRPL-Labs/xumm-sdk-py/blob/main/samples/ws/await_event.py)\n- [Await based on resolving without using a callback function](https://github.com/XRPL-Labs/xumm-sdk-py/blob/main/samples/ws/await_no_callback.py)\n\n##### sdk.payload.create_subscribe\n\n```python\nsdk.payload.create_and_subscribe(\n  payload: CreatePayload,\n  callback: on_payload_event\n): -> PayloadAndSubscription\n```\n\nThe [`<PayloadAndSubscription>`](https://github.com/XRPL-Labs/xumm-sdk-py/blob/main/xumm/resource/types/payload/payload_and_subscription.py) object is basically a [`<PayloadSubscription>`](https://github.com/XRPL-Labs/xumm-sdk-py/blob/main/xumm/resource/types/payload/payload_subscription.py) object with the created payload results in the `created` property:\n\nAll information that applies on [`sdk.payload.create()`](#sdkpayloadcreate) and [`sdk.payload.create_and_subscribe()`](#sdkpayloadsubscribe) applies. Differences are:\n\n1. The input for a `sdk.payload.create_and_subscribe()` call isn't a payload UUID / existing payload, but a paykiad to create.\n2. The response object also contains (`<PayloadAndSubscription>.created`) the response obtained when creating the payload\n3.\n\n#### App Storage\n\nApp Storage allows you to store a JSON object at the XUMM API platform, containing max 60KB of data.\nYour XUMM APP storage is stored at the XUMM API backend, meaning it persists until you overwrite or delete it.\n\nThis data is private, and accessible only with your own API credentials. This private JSON data can be used to store credentials / config / bootstrap info / ... for your headless application (eg. POS device).\n\n```python\nstorage_set = await sdk.storage.set({'name': 'Wietse', 'age': 32, 'male': True})\nprint(storage_set)\n# True\n```\n\n```python\nstorage_get = sdk.storage.get()\nprint(storage_get.data)\n# { 'name': 'Wietse', 'age': 32, 'male': True }\n```\n\n```python\nstorage_delete = sdk.storage.delete()\nprint(storage_delete)\n# True\n```\n\n```python\nstorage_get_after_delete = sdk.storage.get()\nprint(storage_get_after_delete.data)\n# None\n```\n\n#### Helper methods\n\n##### sdk.ping()\n\nThe `ping` method allows you to verify API access (valid credentials) and returns some info on your XUMM APP:\n\n```python\npong = sdk.ping()\n```\n\nReturns [`<ApplicationDetails>`](https://github.com/XRPL-Labs/xumm-sdk-py/blob/main/xumm/resource/types/meta/application_details.py#L294):\n\n```python\npong.quota  # {}\npong.application.name  # 'My XUMM APP'\npong.application.uuidv4  # '00000000-1111-2222-3333-aaaaaaaaaaaa'\npong.application.webhookurl  # ''\npong.application.disabled  # 0\npong.call.uuidv4  # 'bbbbbbbb-cccc-dddd-eeee-111111111111'\n```\n\n##### sdk.get_curated_assets()\n\nThe `get_curated_assets` method allows you to get the list of trusted issuers and IOU's. This is the same list used to\npopulate the \"Add Asset\" button at the XUMM home screan.\n\n```python\ncurated_assets = sdk.get_curated_assets()\n```\n\nReturns [`<CuratedAssetsResponse>`](https://github.com/XRPL-Labs/xumm-sdk-py/blob/main/xumm/resource/types/meta/curated_assets_response.py#L426):\n\n```python\ncurated_assets.issuers  # [ 'Bitstamp', 'GateHub' ]\ncurated_assets.currencies  # [ 'USD', 'BTC', 'EUR', 'ETH' ]\ncurated_assets.details.Bitstamp   # {}\ncurated_assets.details.GateHub   # {}\n```\n\n##### sdk.get_kyc_status()\n\nThe `get_kyc_status` return the KYC status of a user based on a user_token, issued after the\nuser signed a Sign Request (from your app) before (see Payloads - Intro).\n\nIf a user token specified is invalid, revoked, expired, etc. the method will always\nreturn `NONE`, just like when a user didn't go through KYC. You cannot distinct a non-KYC'd user\nfrom an invalid token.\n\nAlternatively, KYC status can be retrieved for an XPRL account address: the address selected in\nXUMM when the session KYC was initiated by.\n\n```python\nkyc_status = sdk.get_kyc_status('00000000-0000-0000-0000-000000000000')\n```\n\n... or using an account address:\n\n```python\nkyc_status = sdk.get_kyc_status('rwu1dgaUq8DCj3ZLFXzRbc1Aco5xLykMMQ')\n```\n\nReturns [`<str of PossibleKycStatuses>`](https://github.com/XRPL-Labs/xumm-sdk-py/blob/main/xumm/resource/types/meta/kyc_status_response.py#L66).\n\n###### Notes on KYC information\n\n- Once an account has successfully completed the XUMM KYC flow, the KYC flag will be applied to the account even if the identity document used to KYC expired. The flag shows that the account was **once** KYC'd by a real person with a real identity document.\n- Please note that the KYC flag provided by XUMM can't be seen as a \"all good, let's go ahead\" flag: it should be used as **one of the data points** to determine if an account can be trusted. There are situations where the KYC flag is still `True`, but an account can no longer be trusted. Eg. when account keys are compromised and the account is now controlled by a 3rd party. While unlikely, depending on the level of trust required for your application you may want to mitigate against these kinds of fraud.\n\n##### sdk.get_transaction()\n\nThe `get_transaction` method allows you to get the transaction outcome (mainnet)\nlive from the XRP ledger, as fetched for you by the XUMM backend.\n\n> **Note**: it's best to retrieve these results **yourself** instead of relying on the XUMM platform to get live XRPL transaction information! You can use the **[xrpl-py](https://pypi.org/project/xrpl-py)** package to do this:\n\n```python\ntx_info = sdk.get_transaction(tx_hash)\n```\n\nReturns: [`<XrplTransaction>`](https://github.com/XRPL-Labs/xumm-sdk-py/blob/main/xumm/resource/types/meta/xrpl_transaction.py#L114).\n\n##### sdk.verify_user_tokens() / sdk.verify_user_tokens()\n\nThe `verify_user_tokens` (or single token: `verify_user_token`) method allows you to verify one or more User Tokens obtained from previous sign requests. This allows you to detect if you will be able to push your next Sign Request to specific users.\n\n```python\nsome_token = '691d5ae8-968b-44c8-8835-f25da1214f35'\ntoken_validity = sdk.verify_user_tokens([\n  some_token,\n  'b12b59a8-83c8-4bc0-8acb-1d1d743871f1',\n  '51313be2-5887-4ae8-9fda-765775a59e51'\n])\nif sdk.verify_user_token(some_token).active:\n  # Push, use `user_token` in payload\nelse:\n  # QR or Redirect (deeplink) flow\n```\n\nReturns: [`<UserTokenValidity or [UserTokenValidity]>`](https://github.com/XRPL-Labs/xumm-sdk-py/blob/main/xumm/resource/types/meta/user_tokens.py#L8).\n\n## Development\n\n### Install requirments\n\n```bash\npip install -e \".[develop]\"\n```\n\n### Build\n\nPlease note: at least Python version **3.6+** is required!\n\nTo build the code, run `python setup.py install`.\n\n### Debugging\n\nThe XUMM SDK will emit debugging info when invoked with a debug environment variable configured like: `DEBUG=xumm-sdk*`\n\nYou'll see the XUMM SDK debug messages if you invoke your script instead of this:\n\n```\npython3 main.py\n```\n\nExample:\n\n```\nDEBUG=xumm-sdk* python3 main.py\n```\n\n### Lint & test\n\nLint the code using:\n\n```bash\npython3 -m flake8\n```\n\nFor running tests:\n\n```bash\npython3 test.py tests/\n```\n",
    "bugtrack_url": null,
    "license": "MIT",
    "summary": "Xumm SDK for Python",
    "version": "1.0.4",
    "project_urls": {
        "Homepage": "https://github.com/XRPL-Labs/xumm-sdk-py"
    },
    "split_keywords": [
        "xrp",
        "ledger",
        "ripple",
        "xumm",
        "sdk"
    ],
    "urls": [
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "ffbcda1fee9e6540427d374f5f8c8b8d8e6e2601c6e64f6cdf6810d6d3dc5403",
                "md5": "e39709f003092620aa8b069b312471e6",
                "sha256": "e6d82a5d99d7c5f6ca6825a5b2885159e4b9dff3ae7d335c90bae6862b2a9d56"
            },
            "downloads": -1,
            "filename": "xumm_sdk_py-1.0.4-py3-none-any.whl",
            "has_sig": false,
            "md5_digest": "e39709f003092620aa8b069b312471e6",
            "packagetype": "bdist_wheel",
            "python_version": "py3",
            "requires_python": null,
            "size": 61397,
            "upload_time": "2023-05-17T12:30:42",
            "upload_time_iso_8601": "2023-05-17T12:30:42.767463Z",
            "url": "https://files.pythonhosted.org/packages/ff/bc/da1fee9e6540427d374f5f8c8b8d8e6e2601c6e64f6cdf6810d6d3dc5403/xumm_sdk_py-1.0.4-py3-none-any.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "f291e2f44d07f6e8e163ecd280139b59b27b7faaf4ef9c3a61ecfa1c85f3825f",
                "md5": "2778ebc89734cc431944665e64e5d62c",
                "sha256": "a0ec7ef2cb42d90c281ea3201a5387975fd0716d1cddc3eb7e0f9629ec98164b"
            },
            "downloads": -1,
            "filename": "xumm-sdk-py-1.0.4.tar.gz",
            "has_sig": false,
            "md5_digest": "2778ebc89734cc431944665e64e5d62c",
            "packagetype": "sdist",
            "python_version": "source",
            "requires_python": null,
            "size": 53894,
            "upload_time": "2023-05-17T12:30:44",
            "upload_time_iso_8601": "2023-05-17T12:30:44.876201Z",
            "url": "https://files.pythonhosted.org/packages/f2/91/e2f44d07f6e8e163ecd280139b59b27b7faaf4ef9c3a61ecfa1c85f3825f/xumm-sdk-py-1.0.4.tar.gz",
            "yanked": false,
            "yanked_reason": null
        }
    ],
    "upload_time": "2023-05-17 12:30:44",
    "github": true,
    "gitlab": false,
    "bitbucket": false,
    "codeberg": false,
    "github_user": "XRPL-Labs",
    "github_project": "xumm-sdk-py",
    "travis_ci": false,
    "coveralls": false,
    "github_actions": true,
    "tox": true,
    "lcname": "xumm-sdk-py"
}
        
Elapsed time: 0.08259s