Integration
One of the main points of interaction with the Anchor Platform is notifying the Anchor Platform about events related to the transaction.
In general, you'll want to provide updates for the following events:
- Your business is processing the KYC information provided by the user
- Your business is ready to receive funds from the user
- Your business has received funds from the user
- Your business has sent funds to the user
- Your business has processed a refund for the user's transaction
- Your business experienced an unexpected error
This is done by making JSON-RPC requests to the Platform API's endpoint. JSON-RPC requests allow you to update the status of the transaction. To move the transaction to a specific status, it's necessary to make a corresponding JSON-RPC request and pass data that is required by this RPC method.
The Anchor Platform JSON-RPC API is designed to notify the platform about changes in the status of the transaction. Given that, the API will be called every time a user or the anchor takes any action that progresses the transaction status in the flow.
You can find out more about transaction flow and statuses in the SEP-24 protocol document
Securing Platform API
By default, the Platform API's endpoints such as GET /transactions
and GET /transactions/:id
are not protected, and are accessible by anyone who has access to the server, including wallet applications.
It's recommended to keep Platform server accessible only from the private network. However, you may want to add additional layer of protection via securing the API.
Using API Key
To enable API key authentication, modify your dev.env
file:
- bash
# dev.env
PLATFORM_SERVER_AUTH_TYPE=api_key
# Will be used as API key
SECRET_PLATFORM_API_AUTH_SECRET="your API key that business server will use"
After it's enabled, all requests must contain a valid X-Api-Key
header, set to the configured API key.
Using JWT
To enable API key authentication, modify your dev.env
file:
- bash
# dev.env
PLATFORM_SERVER_AUTH_TYPE=api_key
# Will be used as API key
SECRET_PLATFORM_API_AUTH_SECRET="your API key that business server will use"
After it's enabled, all requests must contain a valid X-Api-Key
header, set to the configured API key.
Making JSON-RPC Requests
Before making JSON-RPC requests, let's first create a template for making a request to the Anchor Platform.
- bash
# call-json-rpc.sh
#!/usr/bin/env bash
curl localhost:8085 \
-X POST \
-H 'Content-Type: application/json' \
--data "@$1"
This small script will make a JSON-RPC request to the Anchor Platform hosted on the default port (8085). JSON transaction data stored in the provided file will be used as body (requests must be an array).
JSON-RPC Request
The Request object must contain the following attributes:
ATTRIBUTEDATA TYPE
DESCRIPTION
jsonrpcstring
A String specifying the version of the JSON-RPC protocol. MUST be exactly "2.0"
methodstring
A String containing the name of the method to be invoked. List of available methods you can see in [JSON-RPC Methods][json-rpc-methods]
paramsobject
A Structured value that holds the parameter values, corresponding to method call, to be used during the invocation of the method
idstring
An identifier established by the client. The Server will reply with the same value in the Response object
It's possible to provide multiple updates in a single JSON-RPC request (by placing multiple JSON-RPC request objects). When an update is done in this way, all updates will be done sequentially.
Most importantly, each JSON-RPC request is not atomic. If one update fails, all previous updates WILL be applied and all subsequent updates WILL be processed and applied as well.
JSON-RPC Response
The Response is expressed as a single JSON Object, with the following attributes:
ATTRIBUTEDATA TYPE
DESCRIPTION
jsonrpcstring
A String specifying the version of the JSON-RPC protocol. It's set to "2.0"
resultobject
A Structured value that holds the updated transaction details
idstring
An identifier sent by the client
errorobject
A Structured value that holds the error details
Show child attributes
idstring
Unique id of the transaction for which an error occurred
codenumber
A number that indicates the error type that occurred. Please see a list of error codes below
messagestring
A String providing a short description of the error
datastring
A primitive or structured value that contains additional information about the error
Error Codes
Error code | Meaning |
---|---|
-32600 | The JSON sent is not a valid Request object |
-32601 | The method does not exist / is not available |
-32602 | Invalid method parameter(s) |
-32603 | Internal JSON-RPC error |
We will also reference a $transaction_id
variable. This is an identification of transaction that is being returned from the Anchor Platform on an withdrawal or deposit start request. You can obtain the transaction ID by connecting the test wallet to your local Anchor Platform instance.
Updating Deposit Transaction Via JSON-RPC
SEP-24 deposit flow diagram defines sequence/rules of the transaction's status transition and a set of JSON-RPC methods that should be called to change that status. You can't define the status you want to set for a specific transaction in your requests. Each JSON-RPC method defines data structures that it expects in request. If request doesn't contain a required attributes, the Anchor Platform will return and error and won't change status of the transaction.
Statuses in green are mandatory and define the shortest way.
Statuses in yellow are optional and can be skipped.
Statuses in red mean the transaction is in an error status or it has expired.
Ready to Receive Funds
The first step of the deposit flow after starting the deposit itself is collecting KYC. It's usually done in the web-app, but can also be optionally provided by the wallet application, using SEP-9. Once the necessary KYC is collected, a request_offchain_funds
JSON-RPC request should be made.
- JSON
// request-offchain-funds.json
[
{
"id": 1,
"jsonrpc": "2.0",
"method": "request_offchain_funds",
"params": {
"transaction_id": "<transaction_id>",
"message": "Request offchain funds",
"amount_in": {
"amount": 10,
"asset": "iso4217:USD"
},
"amount_out": {
"amount": 9,
"asset": "stellar:USDC:GBBD47IF6LWK7P7MDEVSCWR7DPUWV3NY3DTQEVFL4NAT4AQH3ZLLFLA5"
},
"fee_details": {
"total": 1,
"asset": "iso4217:USD"
},
"amount_expected": {
"amount": 10
}
}
}
]
amount_in
is the amount the user has to sent to the business.amount_out
is the amount the user will receive.fee_details
is the total amount of fees collected by the business.asset
is part of theamount_x
field and is in a SEP-38 format. In this example, it's set to USD, assuming the user made a bank transfer to the system using USD.
Information abouts amounts (in/out/fee) is required if you want to move the transaction from the incomplete
to the pending_user_transfer_start
status. If transaction status is changed from pending_anchor
to pending_user_transfer_start
, you can skip defining the amounts.
To execute this, you need to run:
- bash
./call-json-rpc.sh request-offchain-funds.json
When the KYC process is long (for example, ID verification), it's advised to first set the transaction status to pending_anchor
by using notify_interactive_flow_completed
JSON-RPC request. This will indicate to the user that KYC is being processed.
Processing KYC Information
This step is optional. Most businesses don't use it. You can skip it and go to the next step.
Using this status is recommended when KYC verification may need to be performed asynchronously.
You must specify the amount_x
fields.
- JSON
// kyc-in-process.json
[
{
"id": 1,
"jsonrpc": "2.0",
"method": "notify_interactive_flow_completed",
"params": {
"transaction_id": "<transaction_id>",
"message": "Interactive flow completed.",
"amount_in": {
"amount": 10,
"asset": "iso4217:USD"
},
"amount_out": {
"amount": 9,
"asset": "stellar:USDC:GBBD47IF6LWK7P7MDEVSCWR7DPUWV3NY3DTQEVFL4NAT4AQH3ZLLFLA5"
},
"fee_details": {
"total": 1,
"asset": "iso4217:USD"
},
"amount_expected": {
"amount": 10
}
}
}
]
To execute this, you need to run:
- bash
./call-json-rpc.sh kyc-in-process.json
Funds Received
If offchain funds were received, you'll want to provide an updated transaction information.
- JSON
// offchain-funds-received.json
[
{
"id": 1,
"jsonrpc": "2.0",
"method": "notify_offchain_funds_received",
"params": {
"transaction_id": "<transaction_id>",
"message": "Offchain funds received",
"funds_received_at": "2023-07-04T12:34:56Z",
"external_transaction_id": "7...9",
"amount_in": {
"amount": 10
},
"amount_out": {
"amount": 9
},
"fee_details": {
"total": 1
},
"amount_expected": {
"amount": 10
}
}
}
]
funds_received_at
is the date and time of receiving fundsexternal_transaction_id
is the ID of transaction on external network
The amount fields are optional. If skipped, the values from previous JSON-RPC requests will be taken.
To execute this, you need to run:
- bash
./call-json-rpc.sh offchain-funds-received.json
Waiting For User Funds
In a real world, the transfer confirmation process may take time. In such cases, transactions should be set to a new status indicating that the confirmation of the transfer has been received but the funds themselves have not been received yet.
- JSON
// offchain-funds-sent.json
[
{
"id": 1,
"jsonrpc": "2.0",
"method": "notify_offchain_funds_sent",
"params": {
"transaction_id": "<transaction_id>",
"message": "Offchain funds sent",
"funds_received_at": "2023-07-04T12:34:56Z",
"external_transaction_id": "7...9"
}
}
]
To execute this, you need to run:
- bash
./call-json-rpc.sh offchain-funds-sent.json
Sending Onchain Funds
Next, send a transaction on the Stellar network to fulfill a user request. After the transaction completion, it's necessary to send the notify_onchain_funds_sent
JSON-RPC request to notify a user that the funds were successfully sent.
- JSON
// onchain-funds-sent.json
[
{
"id": 1,
"jsonrpc": "2.0",
"method": "notify_onchain_funds_sent",
"params": {
"transaction_id": "<transaction_id>",
"message": "Onchain funds sent",
"stellar_transaction_id": "7...9"
}
}
]
stellar_transaction_id
is the transaction id on Stellar network of the transfer
To execute this, you need to run:
- bash
./call-json-rpc.sh onchain-funds-sent.json
After this JSON-RPC request, the transaction will be transferred to the completed
status.
Sending Payment Via Custody Service
The Anchor Platform provides a possibility to send a payment via custody services, such as Fireblocks. To make a payment via custody service, it's necessary to make the following JSON-RPC request:
- JSON
// do-stellar-payment.json
[
{
"id": 1,
"jsonrpc": "2.0",
"method": "do_stellar_payment",
"params": {
"transaction_id": "<transaction_id>",
"message": "Custody payment started"
}
}
]
To execute this, you need to run:
- bash
./call-json-rpc.sh do-stellar-payment.json
After successful processing of the payment on a custody service, the Anchor Platform will automatically make the notify_onchain_funds_sent
JSON-RPC request and the status of the transaction will be changed to completed
.
A user account may not be ready to receive funds. You can check that the account has established a trustline. Otherwise, you can set the status of the transaction to the pending_trust
to indicate that the anchor is waiting for the user to establish the trustline.
If custody integration is enabled, the Anchor Platform will do this validation for you automatically.
Pending Trust
This status has to be set if a payment requires an asset trustline that wasn't configured by the user. There are two ways of how the transaction may be moved to the pending_trust
status. The first one is processing of a payment via custody service in case it detected that the trustline isn't configured. The second one is when the business itself detects that the trustline is missing and wants to notify the user that it has to be configured. To move the transaction to the pending_trust
status, it's necessary to make the following JSON-RPC request:
- JSON
// request-trust.json
[
{
"id": 1,
"jsonrpc": "2.0",
"method": "request_trust",
"params": {
"transaction_id": "<transaction_id>",
"message": "Asset trustine not configured"
}
}
]
To execute this, you need to run:
- bash
./call-json-rpc.sh request-trust.json
Payment via custody service periodically checks if the trustline was configured. If it was, it will automatically send a payment to a custody service and change the status of the transaction to pending_stellar
.
Trust Set
This status has to be set if the business has detected that the trustline was or wasn't configured by user.
- JSON
// trust-set.json
[
{
"id": 1,
"jsonrpc": "2.0",
"method": "notify_trust_set",
"params": {
"transaction_id": "<transaction_id>",
"message": "Asset trustine set",
"success": "true"
}
}
]
success
flag which defines if trustline was or wasn't configured by user
To execute this, you need to run:
- bash
./call-json-rpc.sh trust-set.json
Depending on the success
flag, the status of the transaction will be changed to pending_stellar
if the trustline was set, or to pending_anchor
if it wasn't.
Sending Refund Via Custody Service
There is a possibility to send funds back to the user (refund). You can refund the whole sum(full refund) or do a set of partial refunds. Also, if user sent more money than expected, you can refund a part of the sum back to the user and send the rest as onchain funds.
- JSON
// refund-sent.json
[
{
"id": 1,
"jsonrpc": "2.0",
"method": "notify_refund_sent",
"params": {
"transaction_id": "<transaction_id>",
"message": "Refund sent",
"refund": {
"id": "1c186184-09ee-486c-82a6-aa7a0ab1119c",
"amount": {
"amount": 10,
"asset": "iso4217:USD"
},
"amount_fee": {
"amount": 1,
"asset": "iso4217:USD"
}
}
}
}
]
To execute this, you need to run:
- bash
./call-json-rpc.sh refund-sent.json
If a sum of refunds is less than amount_in
, the status of the transaction will be set to pending_anchor
. Only if the sum of refunds is equal to amount_in
, the status of the transaction will be set to refunded
.
Refund Pending
It's similar to Refund sent, but it handles a case when a refund has been submitted to external network but is not yet confirmed. The status of the transaction is set to pending_external
. This is the status that will be set when waiting for Bitcoin or other external crypto network to complete a transaction, or when waiting for a bank transfer.
Transaction Error
If you encounter an unrecoverable error when processing the transaction, it's required to set the transaction status to error
. You can use the message field to describe the error details.
- JSON
// transaction-error.json
[
{
"id": 1,
"jsonrpc": "2.0",
"method": "notify_transaction_error",
"params": {
"transaction_id": "<transaction_id>",
"message": "Error occurred"
}
}
]
To execute this, you need to run:
- bash
./call-json-rpc.sh transaction-error.json
If a user has made a transfer, you should do a transaction recovery, and then you can retry processing the transaction or initiate a refund.
Expired Transaction
Your business may want to handle abandoned transactions by expiring those have remained inactive for a certain period. To achieve this, check the transaction status using the GET /transactions
endpoint and sort the results by the user_action_required_by
timestamp. If the timestamp has passed, manually execute the appropriate logic, such as expiring the transaction or initiating an auto-refund, based on the transaction's current status. For example, to expire transaction business should change the transaction status to expired
:
- JSON
// transaction-expired.json
[
{
"id": 1,
"jsonrpc": "2.0",
"method": "notify_transaction_expired",
"params": {
"transaction_id": "<transaction_id>",
"message": "Transaction expired"
}
}
]
To execute this, you need to run:
- bash
./call-json-rpc.sh transaction-expired.json
This JSON-RPC method can't be used after the user has made a transfer.
On-Hold Transaction
In rare cases, you may want to pause current transaction and request more information from the user (after the transfer has been received). This could be used for compliance use cases.
- JSON
// transaction-hold.json
[
{
"id": 1,
"jsonrpc": "2.0",
"method": "notify_transaction_on_hold",
"params": {
"transaction_id": "<transaction_id>",
"message": "Transaction is on hold. Please contact customer support to resolve the hold."
}
}
]
To execute this, you need to run:
- bash
./call-json-rpc.sh transaction-hold.json
Transaction Recovery
Transaction status can be changed from error/expired
to pending_anchor
. After recovery, you can refund the received assets or proceed with processing of the transaction. To recover a transaction, it's necessary to make the following JSON-RPC request:
- JSON
// transaction-recovery.json
[
{
"id": 1,
"jsonrpc": "2.0",
"method": "notify_transaction_recovery",
"params": {
"transaction_id": "<transaction_id>",
"message": "Transaction recovered"
}
}
]
To execute this, you need to run:
- bash
./call-json-rpc.sh transaction-recovery.json
Updating Withdrawal Transaction Via JSON-RPC
This diagram defines a sequence/rules of transaction's status transition for SEP-24 withdrawal flow.
Statuses in green are mandatory and define the shortest way.
Statuses in yellow are optional and can be skipped.
Statuses in red mean the transaction is in an error status or it has expired.
Once the deposit flow is finished, implementing the withdrawal is straightforward. Some parts of the flow are similar and can be reused.
The starting point both for withdrawal and for deposit is the same.
Ready to Receive Funds
Similar to deposit, the next step is to notify the user that the anchor is ready to receive funds. However, as your service will be receiving transactions over the Stellar network, the update will look differently.
- JSON
// request-onchain-funds.json
[
{
"id": 1,
"jsonrpc": "2.0",
"method": "request_onchain_funds",
"params": {
"transaction_id": "<transaction_id>",
"message": "Request onchain funds",
"amount_in": {
"amount": 10,
"asset": "stellar:USDC:GBBD47IF6LWK7P7MDEVSCWR7DPUWV3NY3DTQEVFL4NAT4AQH3ZLLFLA5"
},
"amount_out": {
"amount": 9,
"asset": "iso4217:USD"
},
"fee_details": {
"total": 1,
"asset": "stellar:USDC:GBBD47IF6LWK7P7MDEVSCWR7DPUWV3NY3DTQEVFL4NAT4AQH3ZLLFLA5"
},
"amount_expected": {
"amount": 10
},
"destination_account": "GD...G",
"memo": "12345",
"memo_type": "id"
}
}
]
memo
Value of memo to attach to the transactionmemo_type
Type of memo that the anchor should attach to the transactiondestination_account
Destination account
To execute this, you need to run:
- bash
./call-json-rpc.sh request-onchain-funds.json
Setting memo
, memo_type
, and destination_account
is optional.
If integration with a third-party custodian is enabled, the Anchor Platform can generate memo
, memo_type
, and destination_address
if a corresponding deposit_info_generator_type
is chosen. Also, you can provide memo
and memo_type
to the request as shown above. Note that the memo must be unique, this is what helps to associate Stellar transactions with SEP transactions.
If your business manages the assets, the Anchor Platform can generate memos for you. When the status is changed to pending_user_transfer_start
, the Anchor Platform sets the memo
and memo_type
automatically (only if it's not included in the request).
The Stellar account that will be used to receive funds should be configured.
Processing KYC Information
This step is optional, and it's similar to Processing KYC Information of the deposit flow.
Funds Received
If onchain funds were received, you need to provide amounts and change the status of the transaction to pending_anchor
.
- JSON
// onchain-funds-received.json
[
{
"id": 1,
"jsonrpc": "2.0",
"method": "notify_onchain_funds_received",
"params": {
"transaction_id": "<transaction_id>",
"message": "Onchain funds received",
"stellar_transaction_id": "7...9",
"amount_in": {
"amount": 10
},
"amount_out": {
"amount": 9
},
"fee_details": {
"total": 1
}
}
}
]
To execute this, you need to run:
- bash
./call-json-rpc.sh onchain-funds-received.json
This method will be called automatically by the custody server if the custody integration is enabled.
Amount Updated
If onchain funds were received, but for some reason the amount_in
differs from specified in the interactive flow (amount_expected
), you can update amount_out
and fee_details
to make them correspond to the actual amount_in
. The status of the transaction in this case won't be changed and will be equal to pending_anchor
.
- JSON
// amounts-updated.json
[
{
"id": 1,
"jsonrpc": "2.0",
"method": "notify_amounts_updated",
"params": {
"transaction_id": "<transaction_id>",
"message": "Amounts updated",
"amount_out": {
"amount": 9
},
"fee_details": {
"total": 1
}
}
}
]
To execute this, you need to run:
- bash
./call-json-rpc.sh amounts-updated.json
Only amount_out
and fee_details
can be updated using this JSON-RPC request, and you don't need to specify the assets of the amounts.
Offchain Funds Sent
To complete the transaction and change its status to completed
, you need to make the notify_offchain_funds_sent
JSON-RPC request.
- JSON
// offchain-funds-sent.json
[
{
"id": 1,
"jsonrpc": "2.0",
"method": "notify_offchain_funds_sent",
"params": {
"transaction_id": "<transaction_id>",
"message": "Offchain funds sent",
"funds_sent_at": "2023-07-04T12:34:56Z",
"external_transaction_id": "a...c"
}
}
]
To execute this, you need to run:
- bash
./call-json-rpc.sh offchain-funds-sent.json
Offchain Funds Available
You can move transaction status to pending_user_transfer_complete
if offchain funds were sent, and if it's ready for the user / recipient to pick it up.
- JSON
// offchain-funds-available.json
[
{
"id": 1,
"jsonrpc": "2.0",
"method": "notify_offchain_funds_available",
"params": {
"transaction_id": "<transaction_id>",
"message": "Offchain funds available",
"external_transaction_id": "a...c"
}
}
]
To execute this, you need to run:
- bash
./call-json-rpc.sh offchain-funds-available.json
Offchain Funds Pending
Another option is to move the transaction's status to pending_external
. This status means that the payment has been submitted to an external network, but is not yet confirmed.
- JSON
// offchain-funds-pending.json
[
{
"id": 1,
"jsonrpc": "2.0",
"method": "notify_offchain_funds_pending",
"params": {
"transaction_id": "<transaction_id>",
"message": "Offchain funds pending",
"external_transaction_id": "a...c"
}
}
]
To execute this, you need to run:
- bash
./call-json-rpc.sh offchain-funds-pending.json
Refund Sent
The refund logic works in the same way as for the deposit flow. For more details, see Refund Sent of the deposit flow.
Sending Refund Via Custody Service
Integration with a custody service allows you to do a refund via a custody service, such as Fireblocks.
- JSON
// do-stellar-refund.json
[
{
"id": 1,
"jsonrpc": "2.0",
"method": "do_stellar_refund",
"params": {
"transaction_id": "<transaction_id>",
"message": "Do stellar refund",
"refund": {
"amount": {
"amount": 9,
"asset": "stellar:USDC:GBBD47IF6LWK7P7MDEVSCWR7DPUWV3NY3DTQEVFL4NAT4AQH3ZLLFLA5"
},
"amount_fee": {
"amount": 1,
"asset": "stellar:USDC:GBBD47IF6LWK7P7MDEVSCWR7DPUWV3NY3DTQEVFL4NAT4AQH3ZLLFLA5"
}
}
}
}
]
To execute this, you need to run:
- bash
./call-json-rpc.sh do-stellar-refund.json
Similarly to the deposit flow, you can make a full refund or a set of partial refunds. The transaction will stay in pending_anchor
status until the sum of refunds is less than amount_in
. If the sum of refunds is equal to amount_in
, the Anchor Platform will automatically change the status of the transaction to refunded
.
Transaction Error
Works in the same manner as for the deposit flow. For more details, see Transaction Error of the deposit flow.
Expired Transaction
Works in the same manner as for the deposit flow. For more details, see Expired Transaction of the deposit flow.
On-Hold Transaction
Works in the same manner as for the deposit flow. For more details, see On-Hold Transaction of the deposit flow.
Transaction Recovery
Works in the same manner as for the deposit flow. For more details, see Transaction Recovery of the deposit flow.
Tracking Stellar Transactions
Using the Payment Observer allows you to delegate this step to the Anchor Platform. To enable the Payment Observer, use the --stellar-observer
flag in the command section of the compose file.
The Payment Observer will track all transactions sent to the distribution account. When the transaction with the expected memo is detected in the network, the status will automatically change to pending_anchor
and event will be the emitted (if Kafka is used).
In order to update the transaction's statuses, the observer makes corresponding JSON-RPC requests to the platform. It should use the following URL.
- bash
# dev.env
PLATFORM_API_BASE_URL=http://platform-server:8085
The Payment Observer won't validate the amounts. It's your responsibility to verify that the amount sent by the user is correct.
If you already have a system that monitors payments, make sure that the logic of the system matches the description below:
First, wait for the transaction to be included in the ledger (using an SDK). This transaction must have the expected memo and destination address (distribution account). Once this transaction has been detected and verified, notify the user that the funds have been received using the notify_onchain_funds_received JSON-RPC request.
The Fireblocks custody service will automatically track transactions and notify the user that the funds have been received. See the Fireblocks custody service documentation for more details.