@ledgerhq/live-common
Version:
Common ground for the Ledger Live apps
677 lines (575 loc) • 22 kB
text/typescript
import {
Account,
AccountLike,
AnyMessage,
getCurrencyForAccount,
SignedOperation,
TokenAccount,
} from "@ledgerhq/types-live";
import {
accountToWalletAPIAccount,
getWalletAPITransactionSignFlowInfos,
getAccountIdFromWalletAccountId,
} from "./converters";
import type { TrackingAPI } from "./tracking";
import { AppManifest, TranslatableString, WalletAPITransaction } from "./types";
import {
isTokenAccount,
isAccount,
getMainAccount,
makeEmptyTokenAccount,
getParentAccount,
} from "../account/index";
import { Transaction } from "../generated/types";
import { prepareMessageToSign } from "../hw/signMessage/index";
import { getAccountBridge } from "../bridge";
import { Exchange } from "../exchange/types";
import { getCryptoAssetsStore } from "@ledgerhq/cryptoassets/state";
import { WalletState } from "@ledgerhq/live-wallet/store";
import { getWalletAccount } from "@ledgerhq/coin-bitcoin/wallet-btc/index";
import { CryptoOrTokenCurrency } from "@ledgerhq/types-cryptoassets";
export function translateContent(content: string | TranslatableString, locale = "en"): string {
if (!content || typeof content === "string") return content;
return content[locale] || content.en;
}
export type WalletAPIContext = {
manifest: AppManifest;
accounts: AccountLike[];
tracking: TrackingAPI;
};
export async function receiveOnAccountLogic(
walletState: WalletState,
{ manifest, accounts, tracking }: WalletAPIContext,
walletAccountId: string,
uiNavigation: (
account: AccountLike,
parentAccount: Account | undefined,
accountAddress: string,
) => Promise<string>,
tokenCurrency?: string,
): Promise<string> {
tracking.receiveRequested(manifest);
const accountId = getAccountIdFromWalletAccountId(walletAccountId);
if (!accountId) {
tracking.receiveFail(manifest);
throw new Error(`accountId ${walletAccountId} unknown`);
}
const account = accounts.find(account => account.id === accountId);
if (!account) {
tracking.receiveFail(manifest);
throw new Error("Account required");
}
const parentAccount = getParentAccount(account, accounts);
const mainAccount = getMainAccount(account, parentAccount);
const currency = tokenCurrency ? await getCryptoAssetsStore().findTokenById(tokenCurrency) : null;
const receivingAccount = currency ? makeEmptyTokenAccount(mainAccount, currency) : account;
const accountAddress = accountToWalletAPIAccount(walletState, account, parentAccount).address;
return uiNavigation(receivingAccount, parentAccount, accountAddress);
}
export async function signTransactionLogic(
{ manifest, accounts, tracking }: WalletAPIContext,
walletAccountId: string,
transaction: WalletAPITransaction,
uiNavigation: (
account: AccountLike,
parentAccount: Account | undefined,
signFlowInfos: {
canEditFees: boolean;
hasFeesProvided: boolean;
liveTx: Partial<Transaction>;
},
) => Promise<SignedOperation>,
tokenCurrency?: string,
isEmbeddedSwap?: boolean,
partner?: string,
): Promise<SignedOperation> {
tracking.signTransactionRequested(manifest, isEmbeddedSwap, partner);
if (!transaction) {
tracking.signTransactionFail(manifest, isEmbeddedSwap, partner);
throw new Error("Transaction required");
}
const accountId = getAccountIdFromWalletAccountId(walletAccountId);
if (!accountId) {
tracking.signTransactionFail(manifest, isEmbeddedSwap, partner);
throw new Error(`accountId ${walletAccountId} unknown`);
}
const account = accounts.find(account => account.id === accountId);
if (!account) {
tracking.signTransactionFail(manifest, isEmbeddedSwap, partner);
throw new Error("Account required");
}
const parentAccount = getParentAccount(account, accounts);
const accountFamily = isTokenAccount(account)
? parentAccount?.currency.family
: account.currency.family;
const mainAccount = getMainAccount(account, parentAccount);
const currency = tokenCurrency ? await getCryptoAssetsStore().findTokenById(tokenCurrency) : null;
const signerAccount = currency ? makeEmptyTokenAccount(mainAccount, currency) : account;
const { canEditFees, liveTx, hasFeesProvided } = getWalletAPITransactionSignFlowInfos({
walletApiTransaction: transaction,
account: mainAccount,
});
if (accountFamily !== liveTx.family) {
throw new Error(
`Account and transaction must be from the same family. Account family: ${accountFamily}, Transaction family: ${liveTx.family}`,
);
}
return uiNavigation(signerAccount, parentAccount, {
canEditFees,
liveTx,
hasFeesProvided,
});
}
export function signRawTransactionLogic(
{ manifest, accounts, tracking }: WalletAPIContext,
walletAccountId: string,
transaction: string,
uiNavigation: (
account: AccountLike,
parentAccount: Account | undefined,
transaction: string,
) => Promise<SignedOperation>,
): Promise<SignedOperation> {
tracking.signRawTransactionRequested(manifest);
if (!transaction) {
tracking.signRawTransactionFail(manifest);
throw new Error("Transaction required");
}
const accountId = getAccountIdFromWalletAccountId(walletAccountId);
if (!accountId) {
tracking.signRawTransactionFail(manifest);
throw new Error(`accountId ${walletAccountId} unknown`);
}
const account = accounts.find(account => account.id === accountId);
if (!account) {
tracking.signRawTransactionFail(manifest);
throw new Error("Account required");
}
const parentAccount = getParentAccount(account, accounts);
return uiNavigation(account, parentAccount, transaction);
}
export async function broadcastTransactionLogic(
{ manifest, accounts, tracking }: WalletAPIContext,
walletAccountId: string,
signedOperation: SignedOperation,
uiNavigation: (
account: AccountLike,
parentAccount: Account | undefined,
signedOperation: SignedOperation,
) => Promise<string>,
tokenCurrency?: string,
): Promise<string> {
if (!signedOperation) {
tracking.broadcastFail(manifest);
throw new Error("Transaction required");
}
const accountId = getAccountIdFromWalletAccountId(walletAccountId);
if (!accountId) {
tracking.broadcastFail(manifest);
throw new Error(`accountId ${walletAccountId} unknown`);
}
const account = accounts.find(account => account.id === accountId);
if (!account) {
tracking.broadcastFail(manifest);
throw new Error("Account required");
}
const currency = tokenCurrency ? await getCryptoAssetsStore().findTokenById(tokenCurrency) : null;
const parentAccount = getParentAccount(account, accounts);
const mainAccount = getMainAccount(account, parentAccount);
const signerAccount = currency ? makeEmptyTokenAccount(mainAccount, currency) : account;
return uiNavigation(signerAccount, parentAccount, signedOperation);
}
export function signMessageLogic(
{ manifest, accounts, tracking }: WalletAPIContext,
walletAccountId: string,
message: string,
uiNavigation: (account: AccountLike, message: AnyMessage) => Promise<Buffer>,
): Promise<Buffer> {
tracking.signMessageRequested(manifest);
const accountId = getAccountIdFromWalletAccountId(walletAccountId);
if (!accountId) {
tracking.signMessageFail(manifest);
return Promise.reject(new Error(`accountId ${walletAccountId} unknown`));
}
const account = accounts.find(account => account.id === accountId);
if (account === undefined) {
tracking.signMessageFail(manifest);
return Promise.reject(new Error("account not found"));
}
let formattedMessage: AnyMessage;
try {
if (isAccount(account)) {
formattedMessage = prepareMessageToSign(account, message);
} else {
throw new Error("account provided should be the main one");
}
} catch (error) {
tracking.signMessageFail(manifest);
return Promise.reject(error);
}
return uiNavigation(account, formattedMessage);
}
export const bitcoinFamilyAccountGetAddressLogic = (
{ manifest, accounts, tracking }: WalletAPIContext,
walletAccountId: string,
derivationPath?: string,
): Promise<string> => {
tracking.bitcoinFamilyAccountAddressRequested(manifest);
const accountId = getAccountIdFromWalletAccountId(walletAccountId);
if (!accountId) {
tracking.bitcoinFamilyAccountAddressFail(manifest);
return Promise.reject(new Error(`accountId ${walletAccountId} unknown`));
}
const account = accounts.find(account => account.id === accountId);
if (account === undefined) {
tracking.bitcoinFamilyAccountAddressFail(manifest);
return Promise.reject(new Error("account not found"));
}
if (!isAccount(account) || account.currency.family !== "bitcoin") {
tracking.bitcoinFamilyAccountAddressFail(manifest);
return Promise.reject(new Error("account requested is not a bitcoin family account"));
}
if (derivationPath) {
const path = derivationPath.split("/");
const accountNumber = Number(path[0]);
const index = Number(path[1]);
if (Number.isNaN(accountNumber) || Number.isNaN(index)) {
tracking.bitcoinFamilyAccountAddressFail(manifest);
return Promise.reject(new Error("Invalid derivationPath"));
}
const walletAccount = getWalletAccount(account);
const address = walletAccount.xpub.crypto.getAddress(
walletAccount.xpub.derivationMode,
walletAccount.xpub.xpub,
accountNumber,
index,
);
tracking.bitcoinFamilyAccountAddressSuccess(manifest);
return Promise.resolve(address);
}
tracking.bitcoinFamilyAccountAddressSuccess(manifest);
return Promise.resolve(account.freshAddress);
};
function getRelativePath(path: string) {
const splitPath = path.split("'/");
return splitPath[splitPath.length - 1];
}
export const bitcoinFamilyAccountGetPublicKeyLogic = async (
{ manifest, accounts, tracking }: WalletAPIContext,
walletAccountId: string,
derivationPath?: string,
): Promise<string> => {
tracking.bitcoinFamilyAccountPublicKeyRequested(manifest);
const accountId = getAccountIdFromWalletAccountId(walletAccountId);
if (!accountId) {
tracking.bitcoinFamilyAccountPublicKeyFail(manifest);
return Promise.reject(new Error(`accountId ${walletAccountId} unknown`));
}
const account = accounts.find(account => account.id === accountId);
if (account === undefined) {
tracking.bitcoinFamilyAccountPublicKeyFail(manifest);
return Promise.reject(new Error("account not found"));
}
if (!isAccount(account) || account.currency.family !== "bitcoin") {
tracking.bitcoinFamilyAccountPublicKeyFail(manifest);
return Promise.reject(new Error("account requested is not a bitcoin family account"));
}
const path = derivationPath?.split("/") ?? getRelativePath(account.freshAddressPath).split("/");
const accountNumber = Number(path[0]);
const index = Number(path[1]);
if (Number.isNaN(accountNumber) || Number.isNaN(index)) {
tracking.bitcoinFamilyAccountPublicKeyFail(manifest);
return Promise.reject(new Error("Invalid derivationPath"));
}
const walletAccount = getWalletAccount(account);
const publicKey = await walletAccount.xpub.crypto.getPubkeyAt(
walletAccount.xpub.xpub,
accountNumber,
index,
);
tracking.bitcoinFamilyAccountPublicKeySuccess(manifest);
return publicKey.toString("hex");
};
export const bitcoinFamilyAccountGetXPubLogic = (
{ manifest, accounts, tracking }: WalletAPIContext,
walletAccountId: string,
): Promise<string> => {
tracking.bitcoinFamilyAccountXpubRequested(manifest);
const accountId = getAccountIdFromWalletAccountId(walletAccountId);
if (!accountId) {
tracking.bitcoinFamilyAccountXpubFail(manifest);
return Promise.reject(new Error(`accountId ${walletAccountId} unknown`));
}
const account = accounts.find(account => account.id === accountId);
if (account === undefined) {
tracking.bitcoinFamilyAccountXpubFail(manifest);
return Promise.reject(new Error("account not found"));
}
if (!isAccount(account) || account.currency.family !== "bitcoin") {
tracking.bitcoinFamilyAccountXpubFail(manifest);
return Promise.reject(new Error("account requested is not a bitcoin family account"));
}
if (!account.xpub) {
tracking.bitcoinFamilyAccountXpubFail(manifest);
return Promise.reject(new Error("account xpub not available"));
}
tracking.bitcoinFamilyAccountXpubSuccess(manifest);
return Promise.resolve(account.xpub);
};
export interface BitcoinGetAddressesResultItem {
address: string;
publicKey?: string;
path?: string;
intention?: string;
}
const PAYMENT_INTENTION = "payment";
export const bitcoinFamilyAccountGetAddressesLogic = async (
{ manifest, accounts, tracking }: WalletAPIContext,
walletAccountId: string,
intentions?: string[],
): Promise<BitcoinGetAddressesResultItem[]> => {
tracking.bitcoinFamilyAccountAddressesRequested(manifest);
const accountId = getAccountIdFromWalletAccountId(walletAccountId);
if (!accountId) {
tracking.bitcoinFamilyAccountAddressesFail(manifest);
throw new Error(`accountId ${walletAccountId} unknown`);
}
const account = accounts.find(account => account.id === accountId);
if (account === undefined) {
tracking.bitcoinFamilyAccountAddressesFail(manifest);
throw new Error("account not found");
}
if (!isAccount(account) || account.currency.family !== "bitcoin") {
tracking.bitcoinFamilyAccountAddressesFail(manifest);
throw new Error("account requested is not a bitcoin family account");
}
if (
intentions !== undefined &&
intentions.length > 0 &&
!intentions.includes(PAYMENT_INTENTION)
) {
tracking.bitcoinFamilyAccountAddressesSuccess(manifest);
return [];
}
try {
const walletAccount = getWalletAccount(account);
const { xpub } = walletAccount;
const { path: rootPath, index: accountIndex } = walletAccount.params;
const allAddresses = await xpub.getXpubAddresses();
let maxReceiveIndex = 0;
let maxChangeIndex = 0;
const receiveIndicesToInclude = new Set<number>([0]);
const changeIndicesToInclude = new Set<number>();
for (const a of allAddresses) {
const hasUtxo = xpub.storage.getAddressUnspentUtxos(a).length > 0;
if (a.account === 0) {
if (a.index > maxReceiveIndex) maxReceiveIndex = a.index;
if (hasUtxo) receiveIndicesToInclude.add(a.index);
} else if (a.account === 1) {
if (a.index > maxChangeIndex) maxChangeIndex = a.index;
if (hasUtxo) changeIndicesToInclude.add(a.index);
}
}
receiveIndicesToInclude.add(maxReceiveIndex + 1);
receiveIndicesToInclude.add(maxReceiveIndex + 2);
changeIndicesToInclude.add(maxChangeIndex + 1);
changeIndicesToInclude.add(maxChangeIndex + 2);
const buildPath = (addressAccount: number, addressIndex: number): string =>
`m/${rootPath}/${accountIndex}'/${addressAccount}/${addressIndex}`;
const toInclude: Array<{ account: number; index: number }> = [];
receiveIndicesToInclude.forEach(index => toInclude.push({ account: 0, index }));
changeIndicesToInclude.forEach(index => toInclude.push({ account: 1, index }));
const storedAddressByAccountIndex = new Map<string, string>();
for (const a of allAddresses) {
storedAddressByAccountIndex.set(`${a.account}:${a.index}`, a.address);
}
const addressPromises = toInclude.map(({ account: addrAccount, index: addrIndex }) => {
const key = `${addrAccount}:${addrIndex}`;
const cached = storedAddressByAccountIndex.get(key);
return cached !== undefined
? Promise.resolve(cached)
: xpub.crypto.getAddress(xpub.derivationMode, xpub.xpub, addrAccount, addrIndex);
});
const publicKeyPromises = toInclude.map(({ account: addrAccount, index: addrIndex }) =>
xpub.crypto.getPubkeyAt(xpub.xpub, addrAccount, addrIndex),
);
const [addressStrs, publicKeys] = await Promise.all([
Promise.all(addressPromises),
Promise.all(publicKeyPromises),
]);
const result: BitcoinGetAddressesResultItem[] = toInclude.map(
({ account: addrAccount, index: addrIndex }, i) => ({
address: addressStrs[i],
publicKey: publicKeys[i].toString("hex"),
path: buildPath(addrAccount, addrIndex),
intention: PAYMENT_INTENTION,
}),
);
tracking.bitcoinFamilyAccountAddressesSuccess(manifest);
return result;
} catch (error) {
tracking.bitcoinFamilyAccountAddressesFail(manifest);
throw error;
}
};
export function startExchangeLogic(
{ manifest, tracking }: WalletAPIContext,
exchangeType: "SWAP" | "FUND" | "SELL" | "SWAP_NG" | "SELL_NG" | "FUND_NG",
uiNavigation: (
exchangeType: "SWAP" | "FUND" | "SELL" | "SWAP_NG" | "SELL_NG" | "FUND_NG",
) => Promise<string>,
): Promise<string> {
tracking.startExchangeRequested(manifest);
return uiNavigation(exchangeType);
}
export type CompleteExchangeRequest = {
provider: string;
fromAccountId: string;
toAccountId?: string;
transaction: WalletAPITransaction;
binaryPayload: string;
signature: string;
feesStrategy: string;
exchangeType: number;
swapId?: string;
rate?: number;
amountExpectedTo?: number;
tokenCurrency?: string;
};
export type CompleteExchangeUiRequest = {
provider: string;
exchange: Exchange;
transaction: Transaction;
binaryPayload: string;
signature: string;
feesStrategy: string;
exchangeType: number;
swapId?: string;
rate?: number;
amountExpectedTo?: number;
tokenCurrency?: string;
};
export async function completeExchangeLogic(
{ manifest, accounts, tracking }: WalletAPIContext,
{
provider,
fromAccountId,
toAccountId,
transaction,
binaryPayload,
signature,
feesStrategy,
exchangeType,
swapId,
rate,
tokenCurrency,
}: CompleteExchangeRequest,
uiNavigation: (request: CompleteExchangeUiRequest) => Promise<string>,
): Promise<string> {
tracking.completeExchangeRequested(manifest);
const realFromAccountId = getAccountIdFromWalletAccountId(fromAccountId);
if (!realFromAccountId) {
throw new Error(`accountId ${fromAccountId} unknown`);
}
// Nb get a hold of the actual accounts, and parent accounts
const fromAccount = accounts.find(a => a.id === realFromAccountId);
let toAccount;
if (toAccountId) {
const realToAccountId = getAccountIdFromWalletAccountId(toAccountId);
if (!realToAccountId) {
throw new Error(`accountId ${toAccountId} unknown`);
}
toAccount = accounts.find(a => a.id === realToAccountId);
}
if (!fromAccount) {
throw new Error("From account not found");
}
if (exchangeType === 0x00 && !toAccount) {
// if we do a swap, a destination account must be provided
throw new Error("To account required for swap");
}
const fromParentAccount = getParentAccount(fromAccount, accounts);
const currency = tokenCurrency ? await getCryptoAssetsStore().findTokenById(tokenCurrency) : null;
const newTokenAccount = currency ? makeEmptyTokenAccount(toAccount, currency) : undefined;
const toParentAccount = toAccount ? getParentAccount(toAccount, accounts) : undefined;
const exchange = {
fromAccount,
fromParentAccount: fromAccount !== fromParentAccount ? fromParentAccount : undefined,
fromCurrency: getCurrencyForAccount(fromAccount),
toAccount: newTokenAccount ? newTokenAccount : toAccount,
toParentAccount: newTokenAccount ? toAccount : toParentAccount,
toCurrency: toAccount ? getToCurrency(toAccount, newTokenAccount) : undefined,
};
const accountBridge = getAccountBridge(fromAccount, fromParentAccount);
const mainFromAccount = getMainAccount(fromAccount, fromParentAccount);
const mainFromAccountFamily = mainFromAccount.currency.family;
const { liveTx } = getWalletAPITransactionSignFlowInfos({
walletApiTransaction: transaction,
account: fromAccount,
});
if (liveTx.family !== mainFromAccountFamily) {
throw new Error(
`Account and transaction must be from the same family. Account family: ${mainFromAccountFamily}, Transaction family: ${liveTx.family}`,
);
}
/**
* 'subAccountId' is used for ETH and it's ERC-20 tokens.
* This field is ignored for BTC
*/
const subAccountId = exchange.fromParentAccount ? fromAccount.id : undefined;
const bridgeTx = accountBridge.createTransaction(fromAccount);
/**
* We append the `recipient` to the tx created from `createTransaction`
* to avoid having userGasLimit reset to null for ETH txs
* cf. libs/ledger-live-common/src/families/ethereum/updateTransaction.ts
*/
const tx = accountBridge.updateTransaction(
{
...bridgeTx,
recipient: liveTx.recipient,
},
{
...liveTx,
feesStrategy: feesStrategy.toLowerCase(),
subAccountId,
},
);
return uiNavigation({
provider,
exchange,
transaction: tx,
binaryPayload,
signature,
feesStrategy,
exchangeType,
swapId,
rate,
});
}
function getToCurrency(account: AccountLike, tokenAccount?: TokenAccount): CryptoOrTokenCurrency {
return tokenAccount ? getCurrencyForAccount(tokenAccount) : getCurrencyForAccount(account);
}
type StorageGetArgs = {
key: string;
storeId: string;
};
type StorageSetArgs = {
key: string;
value: unknown;
storeId: string;
};
type StorageHandlerArgs = StorageGetArgs | StorageSetArgs;
export function protectStorageLogic<T extends StorageHandlerArgs, R>(
manifest: AppManifest,
handler: (args: T) => R,
) {
return (args: T) => {
const { storeId } = args;
// Either the live app can access storage created by itself OR storage explitly listed in the manifest's permissions
if (storeId !== manifest.id && (!manifest.storage || !manifest.storage.includes(storeId))) {
throw new Error(`Live App "${manifest.id}" is not permitted to access storage "${storeId}".`);
}
// Forward call to original handler
return handler(args);
};
}