stellar_flutter_sdk

Stellar SDK for flutter - dart, Stellar, Horizon, Soneso

View on GitHub

SEP-0008 - Regulated Assets

Regulated Assets are assets that require an issuer’s approval (or a delegated third party’s approval, such as a licensed securities exchange) on a per-transaction basis. SEP-08 standardizes the identification of such assets as well as defines the protocol for performing compliance checks and requesting issuer approval.

Create a RegulatedAssetsService instance

Let’s start with creating a RegulatedAssetsService object, which we’ll use for all SEP-08 interactions.

Via it’s constructor:

final service = RegulatedAssetsService(tomlData);

The parameter tomlData of type StellarToml represents stellar.toml data provided by the Server (e.g. Anchor) as described in the SEP-01 Example

Or by providing the domain hosting the stellar.toml file

final service = await RegulatedAssetsService.fromDomain("place.domain.com");

This will automatically load and parse the stellar.toml file. It will then create the RegulatedAssetsService instance by using the needed data provided in the stellar.toml file by the Server.

Get regulated assets

During initialization, the service extracts the relevant assets from the provided toml data. It considers only those currencies that are regulated and have an approval server set.

You can access them as follows:

List<RegulatedAssets> regulatedAssets = service.regulatedAssets;

Authorization required

By using the service, you can check if a given asset needs authorization.

bool needsAuthorization = await service.authorizationRequired(regulatedAsset);

This loads the issuer account data from the Stellar Network and checks if the both flags authRequired and authRevocable are set.

Send transaction to approval server

First let’s create the transaction:

var xAsset = regulatedAssets.first;

// Operation 1: AllowTrust op where issuer fully authorizes account A, asset X
var op1 = SetTrustLineFlagsOperationBuilder(
        accountAId, xAsset, 0, XdrTrustLineFlags.AUTHORIZED_FLAG.value)
    .setSourceAccount(xAsset.issuerId)
    .build();

// Operation 2: Account A manages offer to buy asset X
var op2 =
    ManageBuyOfferOperationBuilder(Asset.NATIVE, xAsset, '10', '0.1')
        .build();

// Operation 3: AllowTrust op where issuer sets account A, asset X to AUTHORIZED_TO_MAINTAIN_LIABILITIES_FLAG state
var op3 = SetTrustLineFlagsOperationBuilder(accountAId, xAsset, 0,
        XdrTrustLineFlags.AUTHORIZED_TO_MAINTAIN_LIABILITIES_FLAG.value)
    .setSourceAccount(xAsset.issuerId)
    .build();

var tx = TransactionBuilder(accountA)
    .addOperation(op1)
    .addOperation(op2)
    .addOperation(op3)
    .build();

String txBase64Xdr = tx.toXdrBase64();

Next let’s send it to the approval server using our service:

var postResponse =
        await service.postTransaction(txBase64Xdr, xAsset.approvalServer);

Depending on the postResponse type you can now access the corresponding data.

if (postResponse is PostTransactionSuccess) {
  // Transaction has been approved and signed by the issuer
  print(postResponse.tx);
  print(postResponse.message);
} else if (postResponse is PostTransactionRevised) {
  // Transaction has been revised to be made compliant, and signed by the issuer. 
  print(postResponse.tx);
  print(postResponse.message);
} else if (postResponse is PostTransactionPending) {
  // The issuer could not determine whether to approve this transaction at the moment. 
  print(postResponse.timeout);
  print(postResponse.message);
} else if (postResponse is PostTransactionActionRequired) {
  // Transaction requires a user action to be completed.
  print(postResponse.actionUrl);
  print(postResponse.actionMethod);
  print(postResponse.actionFields);
  print(postResponse.message);
} else if (postResponse is PostTransactionRejected) {
  // Wallet should display the associated error message to the user.
  print(postResponse.error);
}

Following the Action URL

If the approval server response is PostTransactionActionRequired and the postResponse.actionMethod is POST you can use the service to send the values for the requested fields.

var actionResponse = await service.postAction(actionUrl,
    {'email_address': 'test@mail.com', 
    'mobile_number': '+3472829839222'});

if (actionResponse is PostActionNextUrl) {
    print(actionResponse.message);
    print(actionResponse.nextUrl);
    // ...
} else if (actionResponse is PostActionDone) {
  // resend tx
}

Further readings

SDK’s SEP-08 test cases.