Product
Socket Now Supports uv.lock Files
Socket now supports uv.lock files to ensure consistent, secure dependency resolution for Python projects and enhance supply chain security.
@radixdlt/application
Advanced tools
A JavaScript client library for interacting with the Radix Distributed Ledger.
@radixdlt/application
High-level user-facing API for interacting with the Radix decentralized ledger.
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.
RadixT
tokenBalancesForAddress
- executedTransactions
- nativeToken
- tokenInfo
- tokenFeeForTransaction
- stakesForAddress
- transactionStatus
- networkTransactionThroughput
- networkTransactionDemand
- getAtomForTransaction
- submitSignedAtom
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 aSubscription
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
andERROR
log levels respectively.
The ledger
property is separately documented in the end of this document
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.
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.
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.
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.
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
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.
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.
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.
None of these methods will result in any RPC call to the Radix Core API. All methods perform local computation only.
β οΈ 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.
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
β οΈ 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"
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.
β οΈ 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)
β οΈ 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)
β οΈ 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)
β οΈ 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:
incoming
, outgoing
or unrelated
).XRD
)).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.
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.
β οΈ Not yet implemented, subject to change.
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
txId
), back to the GUI wallet (which it was able to compute locally since it has the signature now.)txId
from last step.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 βοΈβοΈβοΈ
Here follows the actual, RxJS based, transaction flow.
TODO π write this.
β οΈ Not yet implemented, subject to change.
β οΈ Not yet implemented, subject to change.
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>
π‘ 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()
1: At derivation path (BIP44) "m/44'/536'/0'/0/0'"
.
FAQs
A JavaScript client library for interacting with the Radix Distributed Ledger.
We found that @radixdlt/application demonstrated a not healthy version release cadence and project activity because the last version was released a year ago.Β It has 6 open source maintainers collaborating on the project.
Did you know?
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.
Product
Socket now supports uv.lock files to ensure consistent, secure dependency resolution for Python projects and enhance supply chain security.
Research
Security News
Socket researchers have discovered multiple malicious npm packages targeting Solana private keys, abusing Gmail to exfiltrate the data and drain Solana wallets.
Security News
PEP 770 proposes adding SBOM support to Python packages to improve transparency and catch hidden non-Python dependencies that security tools often miss.