diff --git a/examples/daily-reward-dapp/src/Agent.tsx b/examples/daily-reward-dapp/src/Agent.tsx index 0b9c7a4..a00297f 100644 --- a/examples/daily-reward-dapp/src/Agent.tsx +++ b/examples/daily-reward-dapp/src/Agent.tsx @@ -1,13 +1,13 @@ import { Buffer } from "buffer"; import { BencodexDictionary } from "@planetarium/bencodex"; -import { useMemo, useState } from "react"; +import { useEffect, useMemo, useState } from "react"; import { v4 as uuidv4 } from "uuid"; import { - useGetAvatarsWithTipQuery, useStageTransactionMutation, } from "./generated/graphql"; import { getChronoSdk } from "@planetarium/chrono-sdk"; import { Address } from "@planetarium/account"; +import { GetAvatarsResponse, getAvatars, getTip } from "./mimir-client"; interface RefillButtonProps { signer: Address; @@ -106,62 +106,48 @@ function RefillButton({ signer, avatarAddress }: RefillButtonProps) { } interface AgentProps { + network: "odin" | "heimdall"; agentAddress: Address; } -function Agent({ agentAddress }: AgentProps) { - const { data, loading, error } = useGetAvatarsWithTipQuery({ - variables: { - agentAddress: agentAddress.toString(), - }, - pollInterval: 500, - }); +function Agent({ network, agentAddress }: AgentProps) { + const [agent, setAgent] = useState(); + const [tip, setTip] = useState(); - if (loading) { - return

Loading

; - } - - if (error) { - return ( -

Failed to fetch agent-related states.

- ); - } + useEffect(() => { + const interval = setInterval(() => { + getAvatars(network, agentAddress.toString()).then(setAgent); + getTip(network).then(setTip); + }, 500); - if (data === undefined) { - return ( -

Unexpected failure while fetching data.

- ); - } + return () => clearInterval(interval); + }, [network, agentAddress]); - if (data.stateQuery.agent === null || data.stateQuery.agent === undefined) { + if (agent === undefined || tip === undefined) { return ( -

- There is no such agent. (address: {agentAddress.toString()}) -

+

Loading or unexpected failure while fetching data.

); } - const agent = data.stateQuery.agent; - if (agent.avatarStates === null || agent.avatarStates === undefined) { + if (agent.avatars.length < 1) { return ( -

The agent may not have avatar states.

+

The agent may not have any avatars.

); } - const avatarStates = agent.avatarStates; - const tipIndex = data.nodeStatus.tip.index; + const avatars = agent.avatars; return (
- {avatarStates.map((avatar) => ( -
+ {avatars.map(({ avatarAddress, avatarName, actionPoint, dailyRewardReceivedIndex }) => ( +
- {avatar.name} ({avatar.actionPoint} / 120) + {avatarName} ({actionPoint} / 120) - {tipIndex - avatar.dailyRewardReceivedIndex > 2550 ? ( + {tip - dailyRewardReceivedIndex > 2550 ? ( ) : ( <> diff --git a/examples/daily-reward-dapp/src/App.tsx b/examples/daily-reward-dapp/src/App.tsx index 8680c0d..3ab07ef 100644 --- a/examples/daily-reward-dapp/src/App.tsx +++ b/examples/daily-reward-dapp/src/App.tsx @@ -1,8 +1,9 @@ import { ApolloClient, ApolloProvider, InMemoryCache } from "@apollo/client"; -import { useEffect, useState } from "react"; +import { useEffect, useMemo, useState } from "react"; import Agent from "./Agent"; import { getChronoSdk } from "@planetarium/chrono-sdk"; import { Address } from "@planetarium/account"; +import { HEIMDALL_GENESIS_HASH, ODIN_GENESIS_HASH } from "./constants"; function App() { const [accounts, setAccounts] = useState< @@ -14,8 +15,18 @@ function App() { const [isConnected, setConnected] = useState(false); const [currentNetwork, setCurrentNetwork] = useState<{ gqlEndpoint: string, + genesisHash: string, id: string, } | null>(null); + const guessedNetworkName = useMemo<"odin" | "heimdall" | "unknown">(() => { + if (currentNetwork?.genesisHash?.toLowerCase() === ODIN_GENESIS_HASH) { + return "odin"; + } else if (currentNetwork?.genesisHash?.toLowerCase() === HEIMDALL_GENESIS_HASH) { + return "heimdall"; + } else { + return "unknown"; + } + }, [currentNetwork]); const chronoWallet = getChronoSdk(); @@ -88,6 +99,10 @@ function App() { return <>Loading... (network) } + if (guessedNetworkName === "unknown") { + return <>Unknown network (genesis hash: {currentNetwork.genesisHash}) + } + const client = new ApolloClient({ uri: currentNetwork.gqlEndpoint, cache: new InMemoryCache(), @@ -107,7 +122,7 @@ function App() { ))} - +
); diff --git a/examples/daily-reward-dapp/src/constants.ts b/examples/daily-reward-dapp/src/constants.ts new file mode 100644 index 0000000..233425d --- /dev/null +++ b/examples/daily-reward-dapp/src/constants.ts @@ -0,0 +1,2 @@ +export const ODIN_GENESIS_HASH = "4582250d0da33b06779a8475d283d5dd210c683b9b999d74d03fac4f58fa6bce" as const; +export const HEIMDALL_GENESIS_HASH = "729fa26958648a35b53e8e3905d11ec53b1b4929bf5f499884aed7df616f5913" as const; \ No newline at end of file diff --git a/examples/daily-reward-dapp/src/mimir-client.ts b/examples/daily-reward-dapp/src/mimir-client.ts new file mode 100644 index 0000000..bec24a4 --- /dev/null +++ b/examples/daily-reward-dapp/src/mimir-client.ts @@ -0,0 +1,83 @@ +export const BASE_URL = "https://mimir.nine-chronicles.dev/"; + +const defaultHeaders: HeadersInit = { + "Content-Type": "application/json", +}; + +interface FetchOptions { + method?: "GET" | "POST"; + body?: TBody; + headers?: HeadersInit; +} + +async function fetchAPI( + endpoint: string, + options: FetchOptions = {} +): Promise { + try { + const { method = "GET", body, headers } = options; + + const config: RequestInit = { + method, + headers: { ...defaultHeaders, ...headers }, + body: body ? JSON.stringify(body) : null, + }; + + const url = `${BASE_URL}${endpoint}`; + + const response = await fetch(url, config); + + if (!response.ok) { + throw new Error(`Error: ${response.status}`); + } + + const contentType = response.headers.get("Content-Type") || ""; + if (contentType.includes("application/json")) { + return (await response.json()) as TResponse; + } else if (contentType.includes("text/csv")) { + return (await response.text()) as unknown as TResponse; + } else { + throw new Error(`Unsupported response format: ${contentType}`); + } + } catch (error) { + console.error("Error fetching data:", error); + throw error; + } +} + +export async function getSheetNames(network: string): Promise { + return await fetchAPI(`${network}/sheets/names`); +} + +export async function getSheet(network: string, name: string): Promise { + return await fetchAPI(`${network}/sheets/${name}`, { + headers: { accept: "text/csv" }, + }); +} + +interface Avatar { + avatarAddress: string; + avatarName: string; + level: number; + actionPoint: number; + dailyRewardReceivedIndex: number; +} + +export interface GetAvatarsResponse { + avatars: Avatar[]; +} + +export async function getAvatars(network: string, agentAddress: string): Promise { + return await fetchAPI(`${network}/agent/${agentAddress}/avatars`, { + headers: { accept: "application/json" }, + }); +} + +export async function getTip(network: string): Promise { + const resp = await fetchAPI<{ + index: number; + }>(`${network}/tip`, { + headers: { accept: "application/json" }, + }); + return resp.index; +}