kin-android

Kin Logo Base Module

KDoc Maven Central

The :base module is the foundation upon which the rest of the SDK stands. However, it can be used on its own to headlessly access the Kin Blockchain. It’s pure JVM but used as the basis for other Android-specific modules in the SDK.

Kin Logo This is a Kotlin first library, but it’s also fully available to Java developers.

For Java developers that prefer more conventional Java callback/listener idioms, see Java Idioms below.

Installation

Add the following to your project’s gradle file.

repositories {
    // ...
    mavenCentral()
}
dependencies {
    // ...
    implementation "org.kin.sdk.android:base:${versions.kin}"
}

Overview

The base module contains two main components that are to be instantiated and used by the developer:

Quick Start

Everything starts with a KinEnvironment instance that describes which blockchain, services, and storage will be used.

Agora Kin Environment

The Agora Kin Environment is now the preferred method of communicating with the Kin Blockchain. Agora is both a gateway to submit payments and a history collector that can be used to resolve your full payment history. When submitting payments, a developer should properly configure an Agora webhook, which acts as a delegate to approve and optionally co-sign a transaction to mediate transaction fees. Agora can also store additional meta-data about your transaction concerning what your payments were for. This bundle of information is called an Invoice, offchain data which is referenced by the payment’s associated Memo. You can read more about both below in the Sending Payments section.

You’ll also need to tell the SDK a bit about your app in an AppInfoProvider implementation to work with some features (like paying and resolving Invoices and the spend module UI). There are two bundles of information an App provides through this interface:

For more information regarding webhooks and webhook integration please read more about how it works.

val environment: KinEnvironment =
    KinEnvironment.Agora.Builder(NetworkEnvironment.KinStellarTestNetKin3)
        .setAppInfoProvider(object : AppInfoProvider {
                    override val appInfo: AppInfo =
                        AppInfo(
                            DemoAppConfig.DEMO_APP_IDX,
                            DemoAppConfig.DEMO_APP_ACCOUNT_ID,
                            "Kin Demo App",
                            R.drawable.app_icon
                        )

                    override fun getPassthroughAppUserCredentials(): AppUserCreds {
                        return AppUserCreds("demo_app_uid", "demo_app_user_passkey")
                    }
                })
        .setStorage(KinFileStorage.Builder("path/to/storage/location"))
        .build()

As you may notice on the KinAccountContext.Builder, there are a few options for configuring a KinAccountContext

Creating An Account

If you want to create a new KinAccount:

.createNewAccount()

Access An Existing Account

If you want to access an existing KinAccount with options to send KinPayments, input the KinAccount.Id with:

.useExistingAccount(KinAccount.Id("GATG_example_and_fake_key"))

Note: this variant requires that the KinEnvironment knows about this KinAccount’s Key.PrivateKey which can be imported the first time by:

environment.importPrivateKey(Key.PrivateKey("SDR2_example_and_fake_key"))
    .then { success: Boolean ->
        // Completed
    }

Read Only Account Data

For KinAccounts that you do not have a Key.PrivateKey for and only desired read-only access to the data associated with that account:

.useExistingAccountReadOnly(KinAccount.Id("GATG_example_and_fake_key"))

Sending Payments

Sending KinPayments is easy. Just add the amount and the destination KinAccount.Id.

Note: successive calls to this function before the previous is completed will be properly queued according to blockchain implementation needs.

context.sendKinPayment(KinAmount(5), KinAccount.Id("GATG_example_and_fake_key"))
    .then { payment: KinPayment ->
        // Payment Completed
    }

Sending a batch of payments to the blockchain to be completed together, in a single transaction, is just as easy.

Note: This operation is atomic. All payments will either succeed or fail together.

context.sendKinPayments(
    listOf(
        KinPaymentItem(
            KinAmount(5),
            KinAccount.Id("GATG_example_and_fake_key")
        ), KinPaymentItem(
            KinAmount(30),
            KinAccount.Id("GBCA_example_and_fake_key")
        )
    )
).then { completedPayments: List<KinPayment> ->
    // Payments Completed
}

Paymments made within this library are automatically retried, if the error permits being retried, up to 5 times with exponential backoff starting at 1 second after the first attempt (i.e. 1s, 2s, 4s, 8s intervals).

Are there Fees?

It depends. By default, payments on the Kin Blockchain are charged a minimal fee of 100 Quark (1 Quark = 0.001 Kin) each. The minimum required fee is dictated by the Blockchain. Fees on the Kin blockchain are an anti-spam feature intended to prevent malicious actors from spamming the network. Registered Kin apps are given a whitelisted account, which they can use to exempt their or their users’ transactions using the Sign Transaction webhook.

When using KinAccountContext configured with the Agora KinEnvironment, by default a fee will not be added to the payment unless you specifically want your users to pay fees instead of you providing whitelisting. This can be achieved by overriding and setting the isWhitelistingAvailable parameter to false in the KinTransactionWhitelistingApi instance when configuring your KinEnvironment instance.

How can I add more data to the payment?

Memos

Memos are only 32 bytes of data that are stored on the blockchain with your payment data. Because of this, it’s recommended to only include data you can use to reference a larger set of data off chain.

The Kin Binary Memo Format is defined by this spec and includes the following fields:

Use the KinBinaryMemo.Builder class to construct a KinBinaryMemo. The KinBinaryMemo.TransferType is important to set appropriately for the type of payment you are making (Earn/Spend/P2P) (See KDoc definitions for more details). The new foreign key field primarily serves as a way for apps to include a reference to some other data stored off-chain, to help address memo space limitations. This field has a max limit of 230 bits. One option available to developers for storing off-chain data is invoices, which can help developers provide their users with richer transaction data and history. However, developers are free to use the foreign key to reference data hosted by their own services.

 KinBinaryMemo.Builder(appIndex)
            .setTranferType(KinBinaryMemo.TransferType.P2P)
            .setForeignKey(someForeignKeyModel.toByteArray())
            .build()

Text Memos (Old style memos)

In this SDK you can provide a text based memo by using the ClassicKinMemo class. This format should only be used by existing apps that have been issued AppIds and have yet to upgrade to the new Kin Binary Memo Format.

Invoices

Invoices are a great way to leverage Agora to store data about your payments off-chain for you to retrieve later (e.g. in your payment history). They can be submitted to Agora via the optional invoice field of sendKinPayment method with a properly formatted KinBinaryMemo which is used to reference the applicable Invoice data at a later time.

An invoice for a payment contains a list of line items, which contain the following information:

The Invoice.Builder class can be used to construct an Invoice to operate on. As an app that sells a set of digital goods or services, you may wish to transform the information you have in your own models into an Invoice object and reference it later via its identifier stored in the SKU.

val invoice = Invoice.Builder()
            .addLineItem(
                LineItem.Builder("Start a Chat", KinAmount(50))
                    .setSKU(SKU(someProductIdModel.toByteArray()))
                    .build()
            )
            .build()

To execute a payment for an Invoice, you can make use of the payInvoice convenience function on KinAccountContext. This essentially calls sendPayment on your behalf with the correctly formatted KinBinaryMemo and TransferType of KinBinaryMemo.TransferType.Spend. Invoices can and should also be used for other TransferTypes such as P2P and Earns.

The destinationKinAccountId must match the expected & registered destinationKinAppIdx provided during registration

context.payInvoice(invoice, destinationKinAccountId, destinationKinAppIdx).then { kinPayment ->
    // Payment Completed
}

If an Invoice was included with the submission, the optional invoice field on the KinPayment will be populated with the Invoice. This is especially helpful when observing your payment history

kinPayment.invoice

Note: If you are paying for an Invoice, you must use the Kin Binary Memo Format for the memo.*

Retrieving Account Data

The KinAccount.Id for a given KinAccountContext instance is always available

context.accountId

If you require more than just the id, the full KinAccount is available by querying with:

context.getAccount()
    .then { kinAccount: KinAccount ->
        // Do something with the account data
    }

Observing balance changes over time is another common account operation:

Note: don’t forget to clean up when the observer is no longer required! This can be accomplished via a DisposeBag, or by calling .remove(listener) on the Observer

val lifecycle = DisposeBag()
context.observeBalance()
    .add { kinBalance: KinBalance ->
        // Do something on balance update
    }.disposedBy(lifecycle)

Retrieving Payment History

Whether you’re looking for the full payment history or just to be notified of new payments, you can observe any changes to payments for a given account with:

context.observePayments()
    .add { payments: List<KinPayment> ->
        // Will emit the full payment history by default
        // @see ObserverMode for more details
    }
    .disposedBy(lifecycle)

Sometimes it’s useful to retrieve payments that were processed together in a single KinTransaction

context.getPaymentsForTransactionHash(TransactionHash("<txnHash>"))
    .then { payments: List<KinPayment> ->
        // Payments related to txn hash
    }

Other

When done with a particular account, you can irrevocably delete the data, including the private key, by performing the following:

context.clearStorage()
    .then { success: Boolean ->
        // The data with this KinAccountContext is now gone forever
    }

Having Trouble with Dependencies?

Alternatively, there is a shaded artifact for those who are having difficulty resolving common dependencies (e.g. grpc, guava, other google transitive deps)

dependencies {
    // ...
    // The shaded repo doesn't pulling transitive dependencies aautomatically, so add these manually
    implementation 'org.slf4j:slf4j-api:1.7.25'
    implementation 'net.i2p.crypto:eddsa:0.3.0'
    implementation 'io.perfmark:perfmark-api:0.23.0'
    implementation "org.kin.sdk.android:base-shaded:${versions.kin}"
}

This variant is recommended when encountering dependency collisions (e.g. duplicate classes, incompatible versions, etc). If you are still encountering difficulties with dependency collisions with this variant, we recommend forking the library and adding additional libraries you want to shade to the list included under the shadowJar task configured in ../base/build.gradle.

Java Idioms

Instead of Promise .then tail calls in Java…

context.getAccount(new Callback<KinAccount>() {
        @Override
        public void onCompleted(@Nullable KinAccount value, @Nullable Throwable error) {

        }
    });

Instead of Observer .add tail calls in Java…

context.observeBalance(ObservationMode.Passive.INSTANCE, new ValueListener<KinBalance>() {
        @Override
        public void onNext(KinBalance value) {
            // do something
        }
        @Override
        public void onError(@NotNull Throwable error) {
            // handle error
        }
    }).disposedBy(lifecycle);

Tail calls in Java with lambdas…

context.getAccount()
    .then(account -> null);

context.observeBalance(ObservationMode.Passive.INSTANCE)
    .add(kinBalance -> Unit.INSTANCE)
    .disposedBy(lifecycle);

Tail calls in Java NO lambdas…

context.getAccount()
    .then(new Function1<KinAccount, Unit>() {
        @Override
        public Unit invoke(KinAccount kinAccount) {
            return null;
        }
    });

context.observeBalance(ObservationMode.Passive.INSTANCE)
    .add(new Function1<KinBalance, Unit>() {
        @Override
        public Unit invoke(KinBalance kinBalance) {
            return null;
        }
    })
    .disposedBy(lifecycle);

Proguard rules

Since base is just a jar we can’t include a consume-proguard rules file automatically (like ../base-compat/consumer-rules.pro for base-compat does) and you may want to use the suggested proguard rules below.

It’s very possible you’re able to refine them much further than we have and will try to refine these further ourselves in the future

## OkHttp
-dontwarn okhttp3.**
-dontwarn okio.**
-dontwarn javax.annotation.**

## Gson
-keepattributes Signature

# For using GSON @Expose annotation
-keepattributes *Annotation*

# Gson specific classes
-dontwarn sun.misc.**

# Application classes that will be serialized/deserialized over Gson
-keep class org.kin.stellarfork.responses.** { <fields>; }

# Prevent R8 from leaving Data object members always null
-keepclassmembers,allowobfuscation class * {
  @com.google.gson.annotations.SerializedName <fields>;
}

# Prevent proguard from stripping interface information from TypeAdapterFactory,
# JsonSerializer, JsonDeserializer instances (so they can be used in @JsonAdapter)
-keep class * implements com.google.gson.TypeAdapterFactory
-keep class * implements com.google.gson.JsonSerializer
-keep class * implements com.google.gson.JsonDeserializer

# Ignore usages of x509 in eddsa library as it's not in use in our case
-dontwarn sun.security.x509.**

-dontwarn io.envoyproxy.**
-dontwarn org.kin.shaded.**
-dontwarn org.kin.sdk.**
-dontwarn java.lang.ClassValue

# definitely need
-keep class org.kin.sdk.base.network.services.KinService.*$* {*;}
-keep class org.kin.agora.gen.** {*;}
-keep class org.kin.agora.gen.**$* {*;}
-keep class org.kin.agora.gen.account.v3.AccountService {*;}

## Trying to avoid these below by refining further in the future...
# For normal builds
-keep class org.kin.*$* {*;}
-keep class org.kin.**$* {*;}
-keep class io.grpc.** {*;}
-keep class org.kin.** {*;}

# For shaded builds
-keep class org.kin.shaded.** {*;}
-keep class org.kin.shaded.*$* {*;}