@ledgerhq/coin-filecoin
Version:
Ledger Filecoin Coin integration
126 lines (102 loc) • 4.09 kB
text/typescript
import { log } from "@ledgerhq/logs";
import { AxiosRequestConfig, AxiosResponse } from "axios";
import network from "@ledgerhq/live-network/network";
import { makeLRUCache } from "@ledgerhq/live-network/cache";
import { getEnv } from "@ledgerhq/live-env";
import {
BalanceResponse,
BroadcastTransactionRequest,
BroadcastTransactionResponse,
EstimatedFeesRequest,
EstimatedFeesResponse,
NetworkStatusResponse,
TransactionResponse,
TransactionsResponse,
FetchERC20TransactionsResponse,
ERC20Transfer,
ERC20BalanceResponse,
} from "../types";
import { FilecoinFeeEstimationFailed } from "../errors";
const getFilecoinURL = (path?: string): string => {
const baseUrl = getEnv("API_FILECOIN_ENDPOINT");
if (!baseUrl) throw new Error("API base URL not available");
return `${baseUrl}${path ? path : ""}`;
};
const fetch = async <T>(path: string) => {
const url = getFilecoinURL(path);
// We force data to this way as network func is not using the correct param type. Changing that func will generate errors in other implementations
const opts: AxiosRequestConfig = {
method: "GET",
url,
};
const rawResponse = await network(opts);
// We force data to this way as network func is not using the correct param type. Changing that func will generate errors in other implementations
const { data } = rawResponse as AxiosResponse<T>;
log("http", url);
return data;
};
const send = async <T>(path: string, data: Record<string, any>) => {
const url = getFilecoinURL(path);
const opts: AxiosRequestConfig = {
method: "POST",
url,
data: JSON.stringify(data),
headers: { "Content-Type": "application/json" },
};
const rawResponse = await network(opts);
// We force data to this way as network func is not using generics. Changing that func will generate errors in other implementations
const { data: responseData } = rawResponse as AxiosResponse<T>;
log("http", url);
return responseData;
};
export const fetchBalances = async (addr: string): Promise<BalanceResponse> => {
const data = await fetch<BalanceResponse>(`/addresses/${addr}/balance`);
return data; // TODO Validate if the response fits this interface
};
export const fetchEstimatedFees = makeLRUCache(
async (request: EstimatedFeesRequest): Promise<EstimatedFeesResponse> => {
try {
const data = await send<EstimatedFeesResponse>(`/fees/estimate`, request);
return data; // TODO Validate if the response fits this interface
} catch (e: any) {
log("error", "filecoin fetchEstimatedFees", e);
throw new FilecoinFeeEstimationFailed();
}
},
request => `${request.from}-${request.to}`,
{
ttl: 5 * 1000, // 5 seconds
},
);
export const fetchBlockHeight = async (): Promise<NetworkStatusResponse> => {
const data = await fetch<NetworkStatusResponse>("/network/status");
return data as NetworkStatusResponse; // TODO Validate if the response fits this interface
};
export const fetchTxs = async (addr: string): Promise<TransactionResponse[]> => {
const response = await fetch<TransactionsResponse>(`/addresses/${addr}/transactions`);
return response.txs; // TODO Validate if the response fits this interface
};
export const broadcastTx = async (
message: BroadcastTransactionRequest,
): Promise<BroadcastTransactionResponse> => {
const response = await send<BroadcastTransactionResponse>(`/transaction/broadcast`, message);
return response; // TODO Validate if the response fits this interface
};
export const fetchERC20TokenBalance = async (
ethAddr: string,
contractAddr: string,
): Promise<string> => {
const res = await fetch<ERC20BalanceResponse>(
`/contract/${contractAddr}/address/${ethAddr}/balance/erc20`,
);
if (res.data.length) {
return res.data[0].balance;
}
return "0";
};
export const fetchERC20Transactions = async (ethAddr: string): Promise<ERC20Transfer[]> => {
const res = await fetch<FetchERC20TransactionsResponse>(
`/addresses/${ethAddr}/transactions/erc20`,
);
return res.txs.sort((a, b) => b.timestamp - a.timestamp);
};