This repository has been archived by the owner on Jan 8, 2025. It is now read-only.
generated from scaffold-eth/scaffold-eth-2
-
Notifications
You must be signed in to change notification settings - Fork 34
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
Showing
2 changed files
with
291 additions
and
41 deletions.
There are no files selected for viewing
222 changes: 222 additions & 0 deletions
222
packages/nextjs/app/nillion-hello-world/page-complete.tsx
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,222 @@ | ||
"use client"; | ||
|
||
import { useEffect, useState } from "react"; | ||
import type { NextPage } from "next"; | ||
import { useAccount } from "wagmi"; | ||
import { CopyString } from "~~/components/nillion/CopyString"; | ||
import { NillionOnboarding } from "~~/components/nillion/NillionOnboarding"; | ||
import SecretForm from "~~/components/nillion/SecretForm"; | ||
import { Address } from "~~/components/scaffold-eth"; | ||
import { getUserKeyFromSnap } from "~~/utils/nillion/getUserKeyFromSnap"; | ||
import { retrieveSecretBlob } from "~~/utils/nillion/retrieveSecretBlob"; | ||
import { storeSecretsBlob } from "~~/utils/nillion/storeSecretsBlob"; | ||
|
||
const Home: NextPage = () => { | ||
const { address: connectedAddress } = useAccount(); | ||
const [connectedToSnap, setConnectedToSnap] = useState<boolean>(false); | ||
const [userKey, setUserKey] = useState<string | null>(null); | ||
const [userId, setUserId] = useState<string | null>(null); | ||
const [nillion, setNillion] = useState<any>(null); | ||
const [nillionClient, setNillionClient] = useState<any>(null); | ||
const [storedSecretName, setStoredSecretName] = useState<string>("my_blob"); | ||
const [storeId, setStoreId] = useState<string | null>(null); | ||
const [retrievedValue, setRetrievedValue] = useState<string | null>(null); | ||
|
||
// ✅ DONE: complete this function to connect to the MetaMask Snap | ||
async function handleConnectToSnap() { | ||
// call getUserKeyFromSnap | ||
const snapResponse = await getUserKeyFromSnap(); | ||
// update state: set userKey with the response from getUserKeyFromSnap | ||
setUserKey(snapResponse?.user_key || null); | ||
// update state: set connectedToSnap based on the response from getUserKeyFromSnap | ||
setConnectedToSnap(snapResponse?.connectedToSnap || false); | ||
} | ||
|
||
// ✅ DONE: complete this useEffect hook to set up Nillion once a userKey exists | ||
useEffect(() => { | ||
// conditional execution: Check if userKey exists before implementing logic | ||
if (userKey) { | ||
// create an asynchronous getNillionClientLibrary function | ||
const getNillionClientLibrary = async () => { | ||
// dynamically import the nillionClient module using await import("~~/utils/nillion/nillionClient") | ||
const nillionClientUtil = await import("~~/utils/nillion/nillionClient"); | ||
|
||
// asyncronously call the getNillionClient function from the imported nillionClient module with the userKey | ||
const libraries = await nillionClientUtil.getNillionClient(userKey); | ||
// update state: set nillion | ||
setNillion(libraries.nillion); | ||
// update state: set nillionClient | ||
setNillionClient(libraries.nillionClient); | ||
|
||
// return nillionClient | ||
return libraries.nillionClient; | ||
}; | ||
|
||
// call getNillionClientLibrary, then use the returned nillionClient | ||
getNillionClientLibrary().then(nillionClient => { | ||
// get the user_id from the instance of nillionClient | ||
const user_id = nillionClient.user_id(); | ||
// update state: set user_id | ||
setUserId(user_id); | ||
}); | ||
} | ||
}, [userKey]); | ||
|
||
// ✅ DONE: complete this asynchronous function to process the submission of a form used for storing secrets | ||
async function handleSecretFormSubmit(secretName: string, secretValue: string) { | ||
// call storeSecretsBlob, then handle the promise that resolves with a store_id | ||
await storeSecretsBlob(nillion, nillionClient, secretValue, secretName).then((store_id: string) => { | ||
// inside of the "then" method, console log the store_id | ||
console.log("Secret stored at store_id:", store_id); | ||
// update state: set storedSecretName | ||
setStoredSecretName(secretName); | ||
// update state: set storeId | ||
setStoreId(store_id); | ||
}); | ||
} | ||
|
||
// ✅ DONE: complete this asynchronous function to retrieve and read the value of a secret blob | ||
async function handleRetrieveSecretBlob(store_id: string, secret_name: string) { | ||
// call retrieveSecretBlob then handle the promise that resolves with the retrieved value | ||
// update state: set retrievedValue | ||
await retrieveSecretBlob(nillion, nillionClient, store_id, secret_name).then(setRetrievedValue); | ||
} | ||
|
||
// reset nillion values | ||
const resetNillion = () => { | ||
setConnectedToSnap(false); | ||
setUserKey(null); | ||
setUserId(null); | ||
setNillion(null); | ||
setNillionClient(null); | ||
}; | ||
|
||
// reset store blob form to store a new secret | ||
const resetForm = () => { | ||
setStoreId(null); | ||
setRetrievedValue(null); | ||
}; | ||
|
||
useEffect(() => { | ||
// when wallet is disconnected, reset nillion | ||
if (!connectedAddress) { | ||
resetNillion(); | ||
} | ||
}, [connectedAddress]); | ||
|
||
return ( | ||
<> | ||
<div className="flex items-center flex-col pt-10"> | ||
<div className="px-5 flex flex-col"> | ||
<h1 className="text-xl"> | ||
<span className="block text-4xl font-bold">Store and Retrieve "Hello World" with Nillion</span> | ||
{!connectedAddress && <p>Connect your MetaMask Flask wallet</p>} | ||
{connectedAddress && connectedToSnap && !userKey && ( | ||
<a target="_blank" href="https://nillion-snap-site.vercel.app/" rel="noopener noreferrer"> | ||
<button className="btn btn-sm btn-primary mt-4"> | ||
No Nillion User Key - Generate and store user key here | ||
</button> | ||
</a> | ||
)} | ||
</h1> | ||
|
||
{connectedAddress && ( | ||
<div className="flex justify-center items-center space-x-2"> | ||
<p className="my-2 font-medium">Connected Wallet Address:</p> | ||
<Address address={connectedAddress} /> | ||
</div> | ||
)} | ||
|
||
{connectedAddress && !connectedToSnap && ( | ||
<button className="btn btn-sm btn-primary mt-4" onClick={handleConnectToSnap}> | ||
Connect to Snap with your Nillion User Key | ||
</button> | ||
)} | ||
|
||
{connectedToSnap && ( | ||
<div> | ||
{userKey && ( | ||
<div> | ||
<div className="flex justify-center items-center space-x-2"> | ||
<p className="my-2 font-medium"> | ||
🤫 Nillion User Key from{" "} | ||
<a target="_blank" href="https://nillion-snap-site.vercel.app/" rel="noopener noreferrer"> | ||
MetaMask Flask | ||
</a> | ||
: | ||
</p> | ||
|
||
<CopyString str={userKey} /> | ||
</div> | ||
|
||
{userId && ( | ||
<div className="flex justify-center items-center space-x-2"> | ||
<p className="my-2 font-medium">Connected as Nillion User ID:</p> | ||
<CopyString str={userId} /> | ||
</div> | ||
)} | ||
</div> | ||
)} | ||
</div> | ||
)} | ||
</div> | ||
|
||
<div className="flex-grow bg-base-300 w-full mt-16 px-8 py-12"> | ||
<div className="flex justify-center items-center gap-12 flex-col sm:flex-row"> | ||
{!connectedToSnap ? ( | ||
<NillionOnboarding /> | ||
) : ( | ||
<div className="flex flex-row justify-between"> | ||
{/* Store secret blob */} | ||
<div className="flex flex-col bg-base-100 px-10 py-10 text-center items-center w-full rounded-3xl my-2 justify-between mx-5"> | ||
<h1 className="text-xl">Store "Hello World" as a SecretBlob in Nillion</h1> | ||
<div className="flex flex-row w-full justify-between items-center my-10 mx-10"> | ||
<div className="flex-1 px-2"> | ||
{!!storeId ? ( | ||
<p> | ||
✅ Stored SecretBlob {storedSecretName} <br />{" "} | ||
<CopyString str={storeId || ""} textBefore={`store_id: `} full /> | ||
<button className="btn btn-sm btn-primary mt-4" onClick={resetForm}> | ||
Reset | ||
</button> | ||
</p> | ||
) : ( | ||
<SecretForm | ||
secretName={storedSecretName} | ||
onSubmit={handleSecretFormSubmit} | ||
secretType="text" | ||
isLoading={false} | ||
/> | ||
)} | ||
</div> | ||
</div> | ||
</div> | ||
|
||
{/* Retrieve secret blob */} | ||
|
||
<div className="flex flex-col bg-base-100 px-10 py-10 text-center items-center w-full rounded-3xl my-2 justify-between mx-5"> | ||
<h1 className="text-xl">Retrieve SecretBlob from Nillion</h1> | ||
<div className="flex flex-row w-full justify-between items-center my-10 mx-10"> | ||
<div className="flex-1 px-2" key={storedSecretName}> | ||
<button | ||
className="btn btn-sm btn-primary mt-4" | ||
onClick={() => handleRetrieveSecretBlob(storeId || "", storedSecretName)} | ||
disabled={!storeId} | ||
> | ||
Retrieve {storedSecretName} | ||
</button> | ||
|
||
{retrievedValue && <p>✅ Retrieved value: {retrievedValue}</p>} | ||
</div> | ||
</div> | ||
</div> | ||
</div> | ||
)} | ||
</div> | ||
</div> | ||
</div> | ||
</> | ||
); | ||
}; | ||
|
||
export default Home; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters