The wallet itself is a BIP32 hierarchical deterministic (HD) wallet. The Gem wallet takes the approach of calling the root node a wallet. Going to depth1 gets you to the Account nodes and depth2 the addresses underneath the accounts.
The Gem wallet has convenience methods to make managing the wallet easy to do. There are key methods to use off of the wallet object:
wallet.is_locked()
: returns True if lockedwallet.accounts
: returns a collection of round accounts.
A gem account is the main object to interact with. The account is where payments are made from and where you access transaction collections. The gem wallet can have many accounts, but each account is limited to one coin type (also called network). As mentioned in the wallet section, a Gem account within a wallet is a collection of bitcoin addresses and the complexity of dealing with addresses is now abstracted away.
The key methods on an account to use are:
account.balance
: returns the sum of all transactions with 1 or more confirmationsaccount.pending_balance
: returns the sum of all incoming outgoing transactions with 0 confirmationsaccount.available_balance
: returns the usable balance, nothing that is pending or claimed outputs for an outgoing transactionaccount.pay(payees,confirmations,redirect_url)
: send bitcoin out of an account must call wallet.unlock() firstaccount.transactions
: return the collections of transactions
A pending_balance in Gem is any address involved in a transaction with 0 confirmations. This means that in multiple transactions both incoming and outgoing will produce a net pending_balance. As they confirm with a single confirmation, the account balance in the API reflects the change. Objects get cached for speed in the client, so to fetch a new state of an account on the API, call account = account.refresh()
.
Transaction collections have a relationship to an account. When getting the transaction collection, you can specify as an argument incoming or outgoing.
txs = account.transactions(type=‘incoming’)
Now lets look at a single transaction: tx = txs[0]
There is a lot of information on the tx. You can call the attributes to get at the full list tx.attributes
. Additionally there are some convenience methods to get at key information quickly. For example, tx.hash
returns the transaction hash.
Fees are estimated by requesting for an unsigned transaction from the API. The Gem API will then lock the unspent outputs to prevent a potential double spend. The returned unsigned transaction will have a fee in the attributes that you can then inspect. If you decide you don't want to perform the transaction you'll have to cancel the transaction [back]
Example snippet to generate an unsigned transaction:
account = w.accounts['default']
toAddress = u'2N4MtK1rZ88UWXDGWWVf1gYz1Runj4FMDr7'
payees = [{'address':toAddress, 'amount':483034}]
content = dict(payees=payees, utxo_confirmations=1)
unsigned = self.resource.transactions().create(content)
unsigned['fee']
You can accomplish this by calling tx.cancel()
on a transaction. If you have a lot of transactions you can loop over the collection and cancel.
for tx in account.transactions(type='outgoing'):
tx.cancel() if tx.attributes['status'] == 'unsigned'
All objects in the round client have attributes in a key/value store. If you want to see information within the attributes all you have to do is access it like any k/v object.
to see all the attributes of an object:
from pprint import pprint as pp
pp(account.attributes)
To access a particular attribute:
fee = tx.attributes[u’fee’]
user_email = user.attributes[u'email']
If there are no convenience methods for attributes you use often, please file an issue with what you need or make a pr if you build it in yourself.
The data on objects are cached client-side for performance versuses having to make API calls for every single method. What this also means is that if you have for example an instance method for an account, then the information on the account could get into a stale state. You will have to trigger a refresh of the object with any changes from the API.
When calling refresh, the object will be returned with the updated information. Refresh can be called on individual objects as well as the corresponding collections. For example:
account = account.refresh()
: returns the account with any updated informationaccount_collection = account_collection.refresh()
: returns an updated collection
Setting up a subscription on your application will allow you to be notified via a webhook about any incoming/outgoing transaction for any address associated with an account in a wallet of your users. There is no need to manage webhooks at an address level anymore. Gem's API will automatically register any new address or change address added to accounts automatically. When a subscription is triggered, Gem will attempt delivery to the provided callback_url. If your app server does not respond with a 200, Gem will continue to try.
-
Go to the console and add a
subscription token
to the application. This token is shared with the API and Gem will embed the token in any subscription notification that is sent to your app. -
Expand the application by clicking on the name. You will see a section called “subscriptions”
-
Click the “add new subscription” and provide the callback_url . Any new address added to any users wallet authorized on your app will automatically registered for you.
You will start to receive a webhook subscription at the provided url for incoming/outgoing transactions. The payload of the subscription will contain information about the transaction, amount, and UIDs for the user/wallet/account information. You’ll be able to use this information to query your app.
For example - the following snippet will retrieve the user in a given subscription
generate the client
client = round.client()
Authenticate with application credentials
app = client.authenticate_application(app_url, api_token, instance_token)
get the user given the user key from a subscription.
sub_user_key = ‘2309rjefvgnu1340jvfvj24r0j’
user = None
for u in app.users.itervalues():
user = u if u.attributes[u’key’] == sub_user_key
Gem has built 2FA into the API but additionally built a system to add additional 2FA challenges to your app, so you don’t have to integrate yet another api. You can ask Gem to send an sms challenge to the user to then pass back to your app. The user will not get an SMS if the user has a TOTP app installed like Google Authenticator, Authy, Duo etc.
Example of how to incorporate 2FA into your app.
def login_user(user):
user.send_mfa()
verify_password()
unlock_account(user) if user.verify_mfa(USER_ENTERED_MFA)
There are certain scenarios where you want to implement a wallet that you are in posession of that is used for business or custodial purposes. In the operational/custodial model you will have two keys, the primary used for daily signing and the backup used for recovery. This means that you hold funds be it the business or your end users.
- Create a new instance token in the management console.
- Instance tokens are used in the application authentication scheme. When authenticating as an application, you will have full control of the applications wallets and allows a read only view of end user data if your app supports both.
- Keep the token safe
To authenticate as an application to get to an application wallet and/or pull information about the application call:
app = client.authenticate_application(app_url=app_url,
api_token=api_token,
admin_token=admin_token)
backup_key, totp_secret, wallet = app.wallets.create(<PASSPHRASE>)
- The Time-based One Time Password(TOTP) secret is to be stored in a config file on the server operating the round client for this wallet if you want to automate transactions. Otherwise you could set up a mobile device with Authy, Google Authenticator etc to generate Multi-Factor-Authentication (MFA) tokens. This will be a part of the payment process.
- The backup key is the root node that can derive all accounts, addresses. This key will only be returned once via this call. YOU MUST STORE IT IN A SAFE PLACE OFFLINE. If you loose the backup_key and then later forget the passphrase to unlock the primary key, you will not be able to recover the wallet.
- The wallet is the full wallet. You can generate the accounts, addresses etc same as an end user in the previous steps.
In this section you’ll learn how to make a payment for an operational/custodial wallet.
- Authenticate as the application
app = client.authenticate_application(app_url, api_token, admin_token)
- Set the TOTP secret on the application
app.set_totp(TOTP)
- Unlock the wallet.
wallet.unlock(passphrase)
- make a payment
account.pay(payee,confirmations=4, app.get_mfa())
The Gem client will use the totp_secret to generate an MFA token that will be sent as part of the payment calls and verify on the Gem API side.