New Case Study:See how Anthropic automated 95% of dependency reviews with Socket.Learn More β†’
Socket
Sign inDemoInstall
Socket

@radixdlt/application

Package Overview
Dependencies
Maintainers
4
Versions
128
Alerts
File Explorer

Advanced tools

Socket logo

Install Socket

Detect and block malicious and high-risk dependencies

Install

@radixdlt/application

A JavaScript client library for interacting with the Radix Distributed Ledger.

  • 1.0.7-alpha.3
  • Source
  • npm
  • Socket score

Version published
Weekly downloads
45
increased by246.15%
Maintainers
4
Weekly downloads
Β 
Created
Source

@radixdlt/application

High-level user-facing API for interacting with the Radix decentralized ledger.

Intro

import { Radix } from '@radixdlt/application'
import { Subscription } from 'rxjs'

const radix = Radix.create()
	.login('my strong password', loadKeystore)
	.connect(new URL('https://api.radixdlt.com'))

const subs = new Subscription()

radix.tokenBalances.subscribe(
	(tokenBalances) => console.log(`πŸ’Ž My token balances ${tokenBalances.toString()}`)
).add(subs)

/* In the near future... */

// "πŸ’Ž My token balances:
// [ 
//      1337.0 'XRD' ("Rads"), 
//      0.42 'rwBTC' ("Radix-Wrapped Bitcoin")
// ]"

Above code assumes you have a wallet. Looking for wallet creation?

πŸ’‘ Please see README of @radixdlt/account package for a detailed documentation about getting started with a wallet.

Table of Contents

RadixT

All interactions with the Radix ledger is exposed via the reactive interface Radix (of type RadixT) - built with RxJS 7 Beta (v13). Let's see an example of how everything "just works" thanks to using RxJS.

In the code block above we create a RadixT instance and provide it with a Hierarchical Deterministic (HD) wallet by loading a keystore using a function of type () => Promise<KeystoreT> (above named loadKeystore). After the keystore has been loaded, it will be decrypted using the provided password and create a wallet value (of type WalletT). Which the RadixT type will use internally to manage accounts and expose method for creating new ones. By default an initial account will be derived automatically1.

Lastly we subscribe to the reactive stream tokenBalances which will automatically fetch the token balances for the address of the active "account". See (Fetch Trigger)[#fetchTrigger] continuous update of balance ("polling").

Since the Radix Ledger supports multiple tokens for each address this will be a list of tokens, the amount, their symbol ("ticker") and name.

πŸ’‘ Friendly reminder: the observables will not start emitting values until you have subscribed to them

πŸ’‘ Friendly reminder: make sure to handle the returned value of the subscribe() function, by adding then to a Subscription object, otherwise behaviour is undefined, and you might experience all sorts of weird errors (e.g. memory leaks).

However, we can also interact with the Radix Core API without using any wallet, using the property api, like so:

const subs = new Subscription()

const radix = Radix.create()
	.connect(new URL('https://api.radixdlt.com'))
	.setLogLevel(LogLevel.INFO)
	.ledger // accessing all RPC methods
	.nativeToken() // get token info about "XRD"
	.subscribe()
	.add(subs)

/* In the near future... */
// "πŸ’™ got nativeToken response: {
//      "name": "Rads",
//      "resourceIdentifier": "/9SAU2m7yis9iE5u2L44poZ6rYf5JiTAN6GtiRnsBk6JnXoMoAdks/XRD",
//      "symbol": "XRD",
//      "description": "The native currency of the Radix network",
//      "granularity": "1",
//      "hasMutableSupply": false,
//      "currentSupply": "12000000000",
//      "url": "https://https://www.radixdlt.com/",
//      "tokenUrl": "https://avatars.githubusercontent.com/u/34097377?s=280&v=4",
//      "tokenPermission": { "burn": "NONE", "mint": "NONE" }
// } "

In the code block above we did not provide any wallet and notice we access the property ledger, on which we called the method nativeToken() which observable stream we subsequently subscribe to. Lastly we handle the subscription. If we were to set the log level to INFO (from default of WARNING), we would not have seen the output "πŸ’™ got nativeToken response...", neither would we if we wouldn't have called subscribe(), since all observables returned by function calls are lazy (using defer).

πŸ’‘ Everytime you'll see a heart emoji πŸ’œπŸ’šπŸ’™πŸ’›β€οΈ it's a message logged from within this library (inspired by SwiftBeaver), representing VERBOSE, DEBUG, INFO, WARNING and ERROR log levels respectively.

The ledger property is separately documented in the end of this document

Reactive properties

In a GUI wallet you will most likely not use radix.ledger so much, but rather all the reactive properties (Observable variables) on the radix value directly.

Immortal state listeners

If any error where to be emitted on these reactive properties, they would complete (terminate), and you would miss out on any subsequently emitted value. We don't want that, why we've made sure that these never emit any value. All errors are redirected to the specific errors property, but more about that later. For now just remember that you will always get the latest and greatest data given the current active account from the radix interface.

Active address

We can subscribe to the active address, which will emit the formatted radix public address of the active account.

If we create and switch to a new account, we will see how both our active address, and our token balances would update automatically.

radix.activeAddress.subscribe(
	(address) => console.log(`πŸ™‹πŸ½β€β™€οΈ my address is: '${address.toString()}'`)
).add(subs)

/* Instant */
// "πŸ™‹πŸ½β€β™€οΈ my address is: '9S8khLHZa6FsyGo634xQo9QwLgSHGpXHHW764D5mPYBcrnfZV6RT'"

radix.deriveNextAccount({ alsoSwitchTo: true })

/* In the near future... */
// "πŸ™‹πŸ½β€β™€οΈ my address is: '9S8PWQF9smUics1sZEo7CrYgKgCkcopvt9HfWJMTrtPyV2rg7RAG'"

// "πŸ’Ž My token balances:
// [ 
//      237.0 'xwETH' ("Radix-Wrapped Ether")
// ]"

We subscribe to activeAddress which will automatically update with the address of our active "account". We will instantly see our initial account having address ending with "6RT" logged, since we already have an active account. Moments later we create a new account and also switched to it. Since we have subscribed to both tokenBalances and activeAddres, this will emit the token balances of the new account as well as the address of the new account.

⚠️ activeAddress will not emit any address until you have called connected to a node, because a network identifier fetched from the node is required to format an address.

Account listing

We can subscribe to all our accounts and list them using the observable property accounts, like so:

radix.accounts.subscribe(
	(accounts) => console.log(`[πŸ™‹πŸΎβ€β™€οΈ, πŸ™‹πŸΌβ€β™€οΈ] my accounts are: ${accounts.toString()}`)
).add(subs)

/* Instant */
// "[πŸ™‹πŸΎβ€β™€οΈ, πŸ™‹πŸΌβ€β™€οΈ] my accounts are: [
//      {
//          hdPath: "m/44'/536'/0'/0/0'"
//      },
//      {
//          hdPath: "m/44'/536'/0'/0/1'"
//      },
// ]

Well, that is not helpful! What are those? An account (of type AccountT) itself is not so user-friendly or beautiful to look at, it holds a reference to the derivation path used to derive it, and is in itself mostly a collection of functions. So printing them our in a console like this is not so helpful. However, when building a GUI wallet, we can display a clickable dropdown list of accounts and when a user selects an account we should switch to it. You can read about account switching further down.

Active account

You can also subscribe to just the single active account

radix.activeAccount.subscribe(
	(account) => console.log(`πŸ™‹πŸΎβ€β™€οΈ my active account: ${account.toString()}`)
).add(subs)

/* Instant */
// "πŸ™‹πŸΌβ€ my active account: {
//  hdPath: "m/44'/536'/0'/0/0'"
// },

The activeAccount is probably not so useful, better to use the activeAddress and accounts properties

Token Balances

In the intro we subscribed to the tokenBalances. This will get updated automatically when you switch account.

radix
	.tokenBalances
	.subscribe((tokenBalances) => {
		console.log(`πŸ’Ž: ${tokenBalances.toString()}`)
	}
).add(subs)

/* In the near future... */
// "πŸ’Ž My token balances:
// [ 
//      5.37 'rwBTC' ("Radix-Wrapped Bitcoin")
// ]"

// Later
radix
	.deriveNextAccount({ alsoSwitchTo: true })

/* In the near future... */
// "πŸ’Ž My token balances:
// [ 
//      8541.37 'rwETH' ("Radix-Wrapped Ether")
// ]"

See (Fetch Trigger)[#fetchTrigger] for a way either scheduling fetching of token balances at a regular interval ("polling"), or triggering a single fetch when user presses a "Fetch Now" button in GUI.

Errors sink

Since RxJS observable finishes on error, and would stop emitting values after an error, we have made sure all errors are caught and redirected to the errors property (of type `Observable). Meaning that all reactive properties you can listen for values on are immortal.

Apart from logging (controlled with the setLogLevel method as seen in intro and documented below) it is probably a good idea to listen to errors and handle them appropriately. To be clear, you probably should act upon these errors, either you (as a GUI wallet developer) or prompt the user to take appropriate action(s).

radix.errors.subscribe(
	(errorNotification) => { 
		console.log(`☣️ error ${error.toString()}`)
		// Maybe tot only log, but also act upon...	
	},
)

// "☣️ error { 'tag': 'node', msg: 'Invalid SSL certificate' }"
// "☣️ error { 'tag': 'wallet', msg: 'Failed to decrypt wallet' }"
// "☣️ error { 'tag': 'api', msg: 'Request timed out' }"

The radix.errors reactive property is in itself immortal and will never error out, so do not add a subscriber to the error event, but rather the next events.

Error "categories"

The errors property emits three different category of errors, each error is tagged with a 'category', each can be regarded as a separate error channel/stream, and you can choose to split it into separate channels if you'd like.

import { Observable } from 'rxjs'
import { ErrorNotification, WalletError, ErrorNotification } from '@radixdlt/application'

const splitErrorNotificationsOnCategory = (category: ErrorCategory): Observable<ErrorNotificationT> => radix.errors.pipe(
	filter((errorNotification) => errorNotification.category === category),
)

const walletErrors = splitErrorNotificationsOnCategory(ErrorCategory.WALLET)

walletErrors.subscribe(
	(errorNotification) => {
		if (errorNotification.cause === WalletErrorCause.LOAD_KEYSTORE_FAILED) {
			console.log(`⚠️ failed to load keystore: '${errorNotification.message}'`)
			// Aslo display error message in GUI.
		}
	}
)

You can access the underlying error cause and even message with more details.

Methods

Local methods

None of these methods will result in any RPC call to the Radix Core API. All methods perform local computation only.

setLogLevel

⚠️ Not yet implemented, subject to change.

Sets the log level of the internal logger of this SDK. We use roarr. By default, only error and warning logs will visible to you. Lower the log level to see more information.

Account derivation

You can create new accounts with deriveNextAccount(), which takes an optional alsoSwitchTo argument, which changes the current active account.

If you build a GUI wallet you probably want to locally save either a list of the derived accounts, i.e. their hdpaths or you might want to save the account with the highest value (the index of the last one), so that you can restore them upon app start.

For your convenience we provide you with a specific method for this

restoreAccountsUpToIndex

⚠️ Not yet implemented, subject to change.

You can "restore" all accounts up to some last known index. This does not switch the active account. If you passed in the value 0 nothing will happen.

const localStore = localPersistentStoreAt('some/local/path') // or similar
const lastAccountIndex: number = localStore.loadLastAccountIndex() //  or similar

radix.accounts.subscribe(
	(accounts) => console.log(`πŸ™‹πŸΎβ€[] I have #${accounts.length} accounts`)
).add(subs)

radix
	.restoreAccountsUpToIndex(lastAccountIndex)
	.subscribe({
		complete: () => console.log(`βœ… Finished restoring accounts`)
	})
	.add(subs)


/* Later */
// "πŸ™‹πŸΎβ€β™€οΈ[] I have #10 accounts"
// "βœ… Finished restoring accounts"

Account switching

radix
	.switchAccount('first')
	.switchAccount({ toIndex: 1 })
	.switchAccount({ toIndex: 0 })
	.switchAccount('last')

/* Instant */
// "πŸ™‹πŸ½β€β™€οΈ my address is: '9S8k...V6RT'"
// "πŸ™‹πŸ½β€β™€οΈ my address is: '9S8P...7RAG'"
// "πŸ™‹πŸ½β€β™€οΈ my address is: '9S8k...V6RT'"
// "πŸ™‹πŸ½β€β™€οΈ my address is: '9S8P...7RAG'"

A GUI wallet would probably want to send in the selected account (of type AccountT) rather than use 'first' | 'last' | { toIndex: number } though. Which looks like this:

const selectedAccount: AccountT = accountListInGUI.selectedItem() // or similar
radix.switchAccount({ toAccount: selectedAccount })

/* Instant */
// "πŸ™‹πŸ½β€β™€οΈ my address is: <THE_ADDRESS_OF_THE_SELECTED_ACCOUNT>"

TODO: πŸ‘€ we might want to make it possible to give each account a human-readable name, or that might be something a GUI wallet should be responsible for.

Fetch trigger

⚠️ Not yet implemented, subject to change.

You can specify a fetch trigger (polling), by use of withFetchTrigger method.

import { timer } from 'rxjs'

radix
	.withFetchTrigger({
		trigger: timer(3 * 60 * 1_000), // every third minute
		fetch: {
			tokenBalances: true,
			transactionHistory: false,
		}
	})

The above code will make sure you automatically perform a fetch of token balances every third minute. If you change from transactionHistory: false to transactionHistory: true, also transaction history will be fetched with the same interval.

import { Subject } from 'rxjs'

const fetchNowSubject = new Subject<void>()
const trigger = merge(
	timer(3 * 60 * 1_000), // every third minute,
	fetchNowSubject
)

radix
	.withFetchTrigger({
		trigger,
		fetch: {
			tokenBalances: true,
			transactionHistory: true,
		}
	})

// If you "bind" a "Fetch Now"-button in GUI to call `next` on the subject
// this will trigger a fetch
fetchNowSubject.next(undefined) 

Decrypt

⚠️ Not yet implemented, subject to change.

You can decrypt encrypted messages using the private key of the active account in transactions like so:

radix
	.decryptMessageInTransaction(transaction)
	.subscribe({
		next: (decrypted) => { console.log(`βœ…πŸ”“ successfully decrypted message: ${decrypted.message.toString()}`) },
		error: (failure) => { console.log(`βŒπŸ” failed to decrypt message, wrong account? ${failure.toString()}`) },
	})
	.add(subs)

Sign

⚠️ Not yet implemented, subject to change.

You can sign arbitrary data using the private key of the active account, typically you will not use this since you will use higher level method transferTokens which also handles the signing part. This should be considered a more low level API for signing generic data.

radix
	.wallet
	.sign({ unhashed: 'I approve of this message.'.toString('hex')}) // will sha256 hash
	.subscribe(
		(signed) => console.log(`πŸ“ 
			signed message '${signed.message.toString()}', 
			with related private key of public key: ${signed.publicKey.toString()}, 
			resulting in signature: '${signed.signature.toString()}'
		`)
	)
	.add(subs)

Methods resulting in RPC calls

Transaction history

⚠️ Not yet implemented, subject to change.

A transaction is not a token transfer, however, a token transfer might be one action amongst many in a transaction.

Transaction history might be long, and is for that sake paginated. So RadixT needs some kind of "cursor" together with a size, telling it where and how many transactions to fetch from the Radix Distributed Ledger.

radix.transactionHistory({
	size: 3,
}).subscribe(
	(txs) => console.log(`πŸ“’β± transaction history: ${txs.toString()} β±πŸ“’`)
).add(subs)

/* In the near future... */
// "πŸ“’β± transaction history: 
//	{
//		"cursor": "FadedBeeFadedBeeFadedBeeFadedBeeFadedBeeFadedBeeFadedBeeFadedBee",
//		"transactions": [
//			{
//				"id": "DeadBeefDeadBeefDeadBeefDeadBeefDeadBeefDeadBeefDeadBeefDeadBeef",
//				"type": "incoming",
//				"sentAt": "2021-03-14",
//				"fee": "0.0123",
//				"message": {
//					"msg": "51358bd242d0436b738dad123ebf1d8b2103ca9978dbb11cb9764e0bcae41504b4521f0290ac0f33fa659528549//	d9ce84d230000003096dc6785ea0dec1ac1ae15374e327635115407f9ae268aad8b4b6ebae1afefbc83c5792de6fc3550d3//	e0383918d182e87876c9c0e3b5ca0c960fd95b4bd18421ead2aaf472012e7cfbfd7b314cbae588",
//					"encryptionScheme": "ECIES_DH_ADD_AES_GCM_V0"
//				},
//				"actions": [
//					{
//						"type": "TokenTransfer",
//						"from": "9SBRrNSxu6zacM8qyuUpDh4gNqou8QX6QEu53LKVsT4FXjvD77ou",
//						"to": "9S8khLHZa6FsyGo634xQo9QwLgSHGpXHHW764D5mPYBcrnfZV6RT",
//						"amount": "1337",
//						"resourceIdentifier": "/9SAU2m7yis9iE5u2L44poZ6rYf5JiTAN6GtiRnsBk6JnXoMoAdks/XRD"
//					}
//				]
//			},
//			{
//				"id": "ADeadBeeADeadBeeADeadBeeADeadBeeADeadBeeADeadBeeADeadBeeADeadBee",
//				"type": "outgoing",
//				"sentAt": "2021-03-09",
//				"fee": "0.0095",
//				"actions": [
//					{
//						"type": "TokenTransfer",
//						"from": "9S8khLHZa6FsyGo634xQo9QwLgSHGpXHHW764D5mPYBcrnfZV6RT",
//						"to": "9S9tQA7v1jSEUTvLk3hTp9fTmWNsA1ppJ3D6dHLxoqnPcYayAmQf",
//						"amount": "1.25",
//						"resourceIdentifier": "/9SAU2m7yis9iE5u2L44poZ6rYf5JiTAN6GtiRnsBk6JnXoMoAdks/GOLD",
//					}
//				]
//			},
//			{
//				"id": "FadedBeeFadedBeeFadedBeeFadedBeeFadedBeeFadedBeeFadedBeeFadedBee",
//				"type": "outgoing",
//				"sentAt": "2021-01-27",
//				"fee": "0.0087",
//				"actions": [
//					{
//						"type": "Stake",
//						"from": "9S8khLHZa6FsyGo634xQo9QwLgSHGpXHHW764D5mPYBcrnfZV6RT",
//						"toDelegate": "9S81XtkW3H9XZrmnzWqYSuTFPhWXdRnnpL3XXk7h5XxAM6zMdH7k",
//						"amount": "250",
//						"resourceIdentifier": "/9SAU2m7yis9iE5u2L44poZ6rYf5JiTAN6GtiRnsBk6JnXoMoAdks/XRD",
//					}
//				]
//			},
//		]
//	}
// β±πŸ“’
// "

Wow πŸ˜…, that's a mouthful... Let's break it down. We call subscribe to the observable returned by the method call to transactionHistory(...) and after a short delay log the received object. It contains some "cursor", being a pointer to the last transaction, we can use this for subsequent pagination pages. We also see an array of "transactions". Each transaction has:

  1. An identifier.
  2. A type (incoming, outgoing or unrelated).
  3. A date.
  4. A token fee (paid in the native token ("Rad"/XRD)).
  5. An optional, encrypted, message.
  6. A list of actions, more about these below.

Above we saw an example fetching the 3 earliest transactions for an address.

We are unable to read the message attached to the first transaction, since it is encrypted. Messages are not decrypted automatically when received, you have to manual ask for a message to be decrypted, reade more about decryption here.

You ought to keep track of the returned cursor value in the transactionHistory response, since you can use that you query the next "page", like so:


import { Option, none } from 'prelude-ts'
import { AtomIdentifierT } from '@radixdlt/application'
import { Subject } from 'rxjs'

const cursor: Option<AtomIdentifierT> = none()
const fetchTXTrigger = new Subject<number>()

fetchTXTrigger.pipe(
	mergeMap((pageSize) => {
		radix.transactionHistory({
			size: pageSize,
			cursor: cursor.getOrNull()
		})
	})
).subscribe(
	(txs) => { 
		cursor = Option.of(txs.cursor)
		console.log(`πŸ“’πŸ“ƒ got #${txs.size} transactions`)
	}
).add(subs)

fetchTXTrigger.next(20) // fetch tx 0-19
fetchTXTrigger.next(20) // fetch tx 20-33

/* In the near future... */
// πŸ“’πŸ“ƒ got #20 transactions
// πŸ“’πŸ“ƒ got #14 transactions

In the code block above we use cursor to fetch two different "pages" of the transaction history, but this account only had 34 transactions, so the second page only contained 14 entries.

We use a Subject (RxJS) to trigger the multiple calls to transactionHistory, in combination with mergeMap ("flatMap) to transform the observable from number => TransactionHistory . An important thing to note is that we update the cursor upon receiving each new "page".

See (Fetch Trigger)[#fetchTrigger] for a way either scheduling fetching of transaction history at a regular interval ("polling"), or triggering a single fetch when user presses a "Fetch Now" button in GUI.

Actions

TokenTransfer

A transfer of some tokens, of a specific amount. This is probably the most relevant action.

StakeTokens

Staking tokens.

UnstakeTokens

Unstake tokens

ClaimEmissionReward

Claim emission reward.

BurnTokens

Burn tokens.

MintTokens

Burn tokens.

Unknown

The Radix Core API failed to recognize the instructions as a well-formed/well-known canonical action. Will reveal low-level constructs named "particles". For more info, see the Atom Model.

Transfer Tokens

⚠️ Not yet implemented, subject to change.

Transfer input

Let us transfer some tokens! All methods accept specific types such as AddressT for recipient address, AmountT for token amounts and ResourceIdentierT for token asset identifier. All these will have been exposed to you already via tokenBalances, ledger.nativeToken() and/or transactionHistory.

πŸ’‘ Amount of tokens to send must be a multiple of the token's granularity

You can read out the granularity (of type AmountT) from the token info, by using radix.ledger.tokenInfo(tokenResourceIdentifier).

You will also need to make sure to correctly translate unsafe user input into these safe types.

import { Amount } from '@radixdlt/primitives'
import { Address } from '@radixdlt/account'
import { Transaction } from '@radixdlt/application'

const recipientAddressString = recipientTextField.value() // or similar
const recipientAddressResult = Address.fromBase58String(recipientAddressString)
if (recipientAddressResult.isErr()) {
	console.log(`Invalid addres string, error: ${recipientAddressResult.error.message}`)
}
const recipientAddress: AddressT = recipientAddressResult.value

const fooToken: ResourceIdentifierT = selectedToken.id // or similar, read from `tokenBalances`.
const tokenGranularity = radix.ledger
	.tokenInfo(fooToken)
	.subscribe((token) => {
		console.log(`πŸ”ΆπŸŸ πŸ”Έ granularity of token ${token.name} : ${token.granularity.toString()}, any transfer of this token MUST be a multiple of this amount.`)
	}).add(subs)
	
// Later when we know granularity of token.

const amountString = amountTextField.value() // or similar
const amountResult = Amount.fromUnsafe(amountString)
if (amountResult.isErr()) {
	console.log(`Invalid amount string, did you input a number?`)
}
const unsafeAmount: AmmountT = amountResult.value

if (!unsafeAmount.isMultipleOf(granularity)) {
	console.log(`⚠️ requested amount to send is not a mulltiple of token granularity, will be unable to send`)
	// πŸ’‘ also inform user in GUI
	// Abort token sending
}

// β˜‘οΈ Amount is checked against token granularity, safe to send.
const amount = unsafeAmount

Transaction Flow Summary

  1. Gather and transform unsafe inputs into validated and type safe values.
  2. Create a transaction intent (may contain multiple actions), no fee is specified.
  3. From Radix Core API fetch transaction (including fee) translated from intent. Upon response JS lib performs some soundness check that the content of the transaction matches the intent (TBD).
  4. A ready-to-be-signed transaction (including human-readable fee) is returned to the GUI wallet.
  5. The GUI wallet tells JS lib to sign and submit the transaction to the Radix Core API.
  6. Tbe JS lib immediately returns the actual transaction id (txId), back to the GUI wallet (which it was able to compute locally since it has the signature now.)
  7. GUI wallet tells JS lib to poll status of transaction using the txId from last step.

Transaction flow pseudocode (Promise)

Here is a concept of the flow, using await syntax. This is not the actual API, it's mere pseudocode to help visualize the flow. Since transfer of tokens is a multi-stage rocket, there are many things to keep track of.

// ⛔️⛔️⛔️ NOT THE ACTUAL API ⛔️⛔️⛔️
// THIS IS JUST AN OUTLINE OF FLOW
// Step 1️⃣ Gather and transform unsafe inputs into validated and type safe values. Already done in code block above

// Step 2️⃣ Create a transaction intent (may contain multiple actions), no fee is specified.
const transactionIntent = TransactionIntent.create()
	.transferTokens({
		to: recipientAddress,
		amount: amount,
		token: fooToken
	})
	.message(`Thx for lunch Bob, here's for my salad.`)

// 3️⃣ From API fetch transaction (incl fee)
const unsignedTransactionWithReadableFee = await radix.transactionFrom({ 
	intent: intent
})

// 4️⃣ GUI wallet now has access to ready-to-be-signed transaction with human readable fee

// 5️⃣ GUI wallet tells JS lib to sign and submit blob/transaction to the Radix Core API.

// 6️⃣ JS lib immeediatly returns the actual transaction id 
const transactionId = await radix.signAndSubmitTransaction(
	unsignedTransaction
)

// 7️⃣ GUI wallet tells JS lib to poll status of transaction using the txId from last step.
radix.ledger.statusOfTransactionById(transactionId)
	
// 🧩 And when returned transaction status is `CONFIRMED` or `REJECTED` we know that the transaction is complete. Update UI accordingly.

// THIS IS JUST AN OUTLINE OF FLOW
// ⛔️⛔️⛔️ NOT THE ACTUAL API ⛔️⛔️⛔️

Transaction Flow Code

Here follows the actual, RxJS based, transaction flow.

TODO πŸ‘€ write this.

Stake Tokens

⚠️ Not yet implemented, subject to change.

Unstake Tokens

⚠️ Not yet implemented, subject to change.

Ledger

This outlines all the requests you can make to the Radix Core API. All these requests are completely independent of any wallet, thus they have no notion of any "active address".

tokenBalancesForAddress

tokenBalancesForAddress: (address: AddressT) => Observable<TokenBalances>

executedTransactions

executedTransactions: (
	input: Readonly<{
		address: AddressT

		// pagination
		size: number // must be larger than 0
		cursor?: AtomIdentifierT
	}>,
) => Observable<ExecutedTransactions>

nativeToken

nativeToken: () => Observable<Token>

tokenInfo

tokenInfo: (resourceIdentifier: ResourceIdentifierT) => Observable<Token>

tokenFeeForTransaction

tokenFeeForTransaction: (transaction: Transaction) => Observable<TokenFeeForTransaction>

stakesForAddress

stakesForAddress: (address: AddressT) => Observable<Stakes>

transactionStatus

transactionStatus: (id: AtomIdentifierT) => Observable<TransactionStatus>

networkTransactionThroughput

networkTransactionThroughput: () => Observable<NetworkTransactionThroughput>

networkTransactionDemand

networkTransactionDemand: () => Observable<NetworkTransactionDemand>

getAtomForTransaction

getAtomForTransaction: (
	transaction: Transaction,
) => Observable<AtomFromTransactionResponse>

submitSignedAtom

submitSignedAtom: (
	signedAtom: SignedAtom,
) => Observable<SubmittedAtomResponse>

Unsubscribe

πŸ’‘ Friendly reminder: when deemed appropriate, dispose of tour subscriptions by unsubscribe

It's impossible to say when appropriate, that is up to you.

// Earlier
const subs = new Subscription()

// Sometime you did
someObservable.subscribe().add(subs)

// Later
subs.unsubscribe()

Footnotes

1: At derivation path (BIP44) "m/44'/536'/0'/0/0'".

Keywords

FAQs

Package last updated on 25 Mar 2021

Did you know?

Socket

Socket for GitHub automatically highlights issues in each pull request and monitors the health of all your open source dependencies. Discover the contents of your packages and block harmful activity before you install or update your dependencies.

Install

Related posts

SocketSocket SOC 2 Logo

Product

  • Package Alerts
  • Integrations
  • Docs
  • Pricing
  • FAQ
  • Roadmap
  • Changelog

Packages

npm

Stay in touch

Get open source security insights delivered straight into your inbox.


  • Terms
  • Privacy
  • Security

Made with ⚑️ by Socket Inc