From 1144b7d22041064e3e8934f0ca9fc6543de892c9 Mon Sep 17 00:00:00 2001
From: Kheops <26880866+0xKheops@users.noreply.github.com>
Date: Fri, 17 Jan 2025 14:56:57 +0900
Subject: [PATCH] feat: display nonce and lifetime in substrate tx details
(#1778)
---
.../Sign/ViewDetails/ViewDetailsSub.tsx | 80 ++++++++++++++++---
apps/extension/src/ui/util/scaleApi/sapi.ts | 7 +-
2 files changed, 74 insertions(+), 13 deletions(-)
diff --git a/apps/extension/src/ui/domains/Sign/ViewDetails/ViewDetailsSub.tsx b/apps/extension/src/ui/domains/Sign/ViewDetails/ViewDetailsSub.tsx
index 0e29b38747..1c3681366c 100644
--- a/apps/extension/src/ui/domains/Sign/ViewDetails/ViewDetailsSub.tsx
+++ b/apps/extension/src/ui/domains/Sign/ViewDetails/ViewDetailsSub.tsx
@@ -1,4 +1,6 @@
import { TypeRegistry } from "@polkadot/types"
+import { classNames } from "@talismn/util"
+import { useQuery } from "@tanstack/react-query"
import { FC, useEffect, useMemo } from "react"
import { useTranslation } from "react-i18next"
import { Button, Drawer } from "talisman-ui"
@@ -21,6 +23,19 @@ import { ViewDetailsButton } from "./ViewDetailsButton"
import { ViewDetailsField } from "./ViewDetailsField"
import { ViewDetailsTxObject } from "./ViewDetailsTxObject"
+export const ViewDetailsSub: FC = () => {
+ const { isOpen, open, close } = useOpenClose()
+
+ return (
+ <>
+
+
+
+
+ >
+ )
+}
+
const ViewDetailsContent: FC<{
onClose: () => void
}> = ({ onClose }) => {
@@ -75,6 +90,8 @@ const ViewDetailsContent: FC<{
return { methodName, args }
}, [extrinsic, t])
+ const { data: lifetimeRows } = useLifetimeRows()
+
useEffect(() => {
genericEvent("open sign transaction view details", { type: "substrate" })
}, [genericEvent])
@@ -100,6 +117,16 @@ const ViewDetailsContent: FC<{
token={nativeToken}
/>
+
+ {decodedPayload?.nonce.toNumber()}
+
+
+ {lifetimeRows?.map((str, i) => (
+
+ {str}
+
+ ))}
+
{
- const { isOpen, open, close } = useOpenClose()
+const useLifetimeRows = () => {
+ const { t } = useTranslation("request")
+ const { sapi, payload } = usePolkadotSigningRequest()
- return (
- <>
-
-
-
-
- >
- )
+ const period = useMemo(() => {
+ try {
+ if (!isJsonPayload(payload)) return null
+
+ // if blockhash is equal to genesis hash then transaction is immortal
+ if (payload.genesisHash === payload.blockHash) return null
+
+ const registry = new TypeRegistry()
+ const ep = registry.createType("ExtrinsicPayload", payload)
+
+ return ep.era.isMortalEra ? ep.era.asMortalEra.period.toNumber() : null
+ } catch (err) {
+ return null
+ }
+ }, [payload])
+
+ const mortality = useMemo(() => {
+ if (!isJsonPayload(payload)) return ""
+
+ // note: for mortal transaction the period is never 0
+ if (!period) return t("Immortal")
+
+ const startBlock = parseInt(payload.blockNumber, 16)
+ return t("From block {{startBlock}} to block {{endBlock}}", {
+ startBlock,
+ endBlock: startBlock + period - 1,
+ })
+ }, [payload, period, t])
+
+ return useQuery({
+ queryKey: ["System.Number", sapi?.id],
+ queryFn: () => {
+ if (!sapi) return null
+ return sapi.getStorage("System", "Number", [])
+ },
+ initialData: null,
+ refetchInterval: 2_000,
+ select: (blockNumber) =>
+ period ? [mortality, t("Current block: {{blockNumber}}", { blockNumber })] : [mortality],
+ })
}
diff --git a/apps/extension/src/ui/util/scaleApi/sapi.ts b/apps/extension/src/ui/util/scaleApi/sapi.ts
index d1246fd1bc..8b2f8ad7fb 100644
--- a/apps/extension/src/ui/util/scaleApi/sapi.ts
+++ b/apps/extension/src/ui/util/scaleApi/sapi.ts
@@ -73,8 +73,8 @@ export const getScaleApi = (
getConstant: (pallet: string, constant: string) =>
getConstantValue(chainId, metadata, builder, pallet, constant),
- getStorage: (pallet: string, entry: string, keys: unknown[]) =>
- getStorageValue(chainId, builder, pallet, entry, keys),
+ getStorage: (pallet: string, entry: string, keys: unknown[], at?: string) =>
+ getStorageValue(chainId, builder, pallet, entry, keys, at),
getDecodedCall: (pallet: string, method: string, args: unknown) =>
getDecodedCall(pallet, method, args),
@@ -483,11 +483,12 @@ const getStorageValue = async (
pallet: string,
entry: string,
keys: unknown[],
+ at?: string,
) => {
const storageCodec = scaleBuilder.buildStorage(pallet, entry)
const stateKey = storageCodec.enc(...keys)
- const hexValue = await api.subSend(chainId, "state_getStorage", [stateKey])
+ const hexValue = await api.subSend(chainId, "state_getStorage", [stateKey, at])
if (!hexValue) return null as T // caller will need to expect null when applicable
return storageCodec.dec(hexValue) as T