UNPKG

@ledgerhq/coin-stellar

Version:
187 lines (172 loc) 5.91 kB
import { AlpacaApi, Block, BlockInfo, FeeEstimation, Operation, Pagination, TransactionIntent, } from "@ledgerhq/coin-framework/api/index"; import coinConfig, { type StellarConfig } from "../config"; import { broadcast, combine, craftTransaction, estimateFees, getBalance, lastBlock, listOperations, } from "../logic"; import { ListOperationsOptions } from "../logic/listOperations"; import { StellarAsset, StellarMemo } from "../types"; import { LedgerAPI4xx } from "@ledgerhq/errors"; import { log } from "@ledgerhq/logs"; import { xdr } from "@stellar/stellar-sdk"; export function createApi(config: StellarConfig): AlpacaApi<StellarAsset, StellarMemo> { coinConfig.setCoinConfig(() => ({ ...config, status: { type: "active" } })); return { broadcast, combine: compose, craftTransaction: craft, estimateFees: estimate, getBalance, lastBlock, listOperations: operations, getBlock(_height): Promise<Block<StellarAsset>> { throw new Error("getBlock is not supported"); }, getBlockInfo(_height: number): Promise<BlockInfo> { throw new Error("getBlockInfo is not supported"); }, }; } async function craft( transactionIntent: TransactionIntent<StellarAsset, StellarMemo>, customFees?: bigint, ): Promise<string> { const fees = customFees !== undefined ? customFees : await estimateFees(); // NOTE: check how many memos, throw if more than one? // if (transactionIntent.memos && transactionIntent.memos.length > 1) { // throw new Error("Stellar only supports one memo per transaction."); // } const memo = "memo" in transactionIntent ? transactionIntent.memo : undefined; const hasMemoValue = memo && memo.type !== "NO_MEMO"; const tx = await craftTransaction( { address: transactionIntent.sender }, { type: transactionIntent.type, recipient: transactionIntent.recipient, amount: transactionIntent.amount, fee: fees, ...(transactionIntent.asset.type === "token" ? { assetCode: transactionIntent.asset.assetCode, assetIssuer: transactionIntent.asset.assetIssuer, } : {}), memoType: memo?.type, ...(hasMemoValue ? { memoValue: (memo as { value: string }).value } : {}), }, ); // Note: the API returns the signature base, not the full XDR, see BACK-8727 for more context return tx.signatureBase; } function compose(tx: string, signature: string, pubkey?: string): string { if (!pubkey) { throw new Error("Missing pubkey"); } // note: accept here `TransactionEnvelope` or `TransactionSignaturePayload`, see BACK-8727 for more context return combine(envelopeFromAnyXDR(tx, "base64"), signature, pubkey); } async function estimate(): Promise<FeeEstimation> { const value = await estimateFees(); return { value }; } async function operations( address: string, { minHeight }: Pagination, ): Promise<[Operation<StellarAsset>[], string]> { return operationsFromHeight(address, minHeight); } type PaginationState = { readonly pageSize: number; readonly heightLimit: number; continueIterations: boolean; apiNextCursor?: string; accumulator: Operation<StellarAsset>[]; }; async function operationsFromHeight( address: string, minHeight: number, ): Promise<[Operation<StellarAsset>[], string]> { const state: PaginationState = { pageSize: 200, heightLimit: minHeight, continueIterations: true, accumulator: [], }; // unfortunately, the stellar API does not support an option to filter by min height // so the only strategy to get ALL operations is to iterate over all of them in descending order // until we reach the desired minHeight while (state.continueIterations) { const options: ListOperationsOptions = { limit: state.pageSize, order: "desc", minHeight }; if (state.apiNextCursor) { options.cursor = state.apiNextCursor; } try { const [operations, nextCursor] = await listOperations(address, options); state.accumulator.push(...operations); state.apiNextCursor = nextCursor; state.continueIterations = nextCursor !== ""; } catch (e: unknown) { if (e instanceof LedgerAPI4xx && (e as unknown as { status: number }).status === 429) { log("coin:stellar", "(api/operations): TooManyRequests, retrying in 4s"); await new Promise(resolve => setTimeout(resolve, 4000)); } else { throw e; } } } return [state.accumulator, state.apiNextCursor ? state.apiNextCursor : ""]; } /** * Deserialize a transaction envelope, also accepting transaction signature payload form. * * @param input serialized `TransactionEnvelope` or `TransactionSignaturePayload` * @param format serialization encoding */ export function envelopeFromAnyXDR( input: string, format: "hex" | "base64", ): xdr.TransactionEnvelope { try { return xdr.TransactionEnvelope.fromXDR(input, format); } catch (envelopeError) { try { return signatureBaseToEnvelope(xdr.TransactionSignaturePayload.fromXDR(input, format)); } catch (signatureBaseError) { throw new Error( `Failed decoding transaction as an envelope (${envelopeError}) or as a signature base: (${signatureBaseError})`, ); } } } /** * Convert a `TransactionSignaturePayload` into a `TransactionEnvelope`. * * @param signatureBase deserialized `TransactionSignaturePayload` */ function signatureBaseToEnvelope( signatureBase: xdr.TransactionSignaturePayload, ): xdr.TransactionEnvelope { const tx = signatureBase.taggedTransaction().value(); if (tx instanceof xdr.Transaction) { return xdr.TransactionEnvelope.envelopeTypeTx( new xdr.TransactionV1Envelope({ tx, signatures: [] }), ); } else { return xdr.TransactionEnvelope.envelopeTypeTxFeeBump( new xdr.FeeBumpTransactionEnvelope({ tx, signatures: [] }), ); } }