Compare commits

..

4 Commits

Author SHA1 Message Date
eba6cd5d68 Merge pull request 'Skin onboarding app' (#33) from style/vaidator into main
Reviewed-on: #33
2024-08-11 21:05:04 +00:00
zramsay
5baccf3a84 bump version 2024-08-11 17:02:20 -04:00
b39afe386f style: validator success page 2024-08-11 12:34:35 -04:00
565d1887e0 style: add codeblock and style validator page 2024-08-11 12:32:55 -04:00
7 changed files with 133 additions and 152 deletions

View File

@ -1,6 +1,6 @@
{ {
"name": "testnet-onboarding-app", "name": "testnet-onboarding-app",
"version": "0.1.0", "version": "0.1.2",
"private": true, "private": true,
"dependencies": { "dependencies": {
"@cerc-io/registry-sdk": "^0.2.5", "@cerc-io/registry-sdk": "^0.2.5",

View File

@ -0,0 +1,25 @@
import { Box } from "@mui/material";
import React, { PropsWithChildren } from "react";
export const CodeBlock: React.FC<PropsWithChildren> = ({ children }) => (
<Box
sx={{
backgroundColor: "#48474F",
padding: 3,
wordWrap: "break-word",
mt: 1,
borderRadius: 1,
}}
>
<pre
style={{
whiteSpace: "pre-wrap",
margin: 0,
backgroundColor: "#48474F",
color: "#FBFBFB",
}}
>
{children}
</pre>
</Box>
);

View File

@ -15,6 +15,7 @@ import {
} from "../utils/sumsub"; } from "../utils/sumsub";
import { ENABLE_KYC, HASHED_SUBSCRIBER_ID_KEY } from "../constants"; import { ENABLE_KYC, HASHED_SUBSCRIBER_ID_KEY } from "../constants";
import { Participant } from "../types"; import { Participant } from "../types";
import { CodeBlock } from "../components/CodeBlock";
const registry = new Registry(process.env.REACT_APP_REGISTRY_GQL_ENDPOINT!); const registry = new Registry(process.env.REACT_APP_REGISTRY_GQL_ENDPOINT!);
@ -88,33 +89,17 @@ const OnboardingSuccess = () => {
<Typography variant="body1"> <Typography variant="body1">
Participant onboarded: <br /> Participant onboarded: <br />
</Typography> </Typography>
<Box <CodeBlock>
sx={{ {participant && (
backgroundColor: "lightgray", <div>
padding: 3, Laconic Address: {participant.cosmosAddress} <br />
wordWrap: "break-word", Nitro Address: {participant.nitroAddress} <br />
marginBottom: 6, Role: {participant.role} <br />
}} KYC ID: {participant.kycId} <br />
> <br />
<pre </div>
style={{ )}
whiteSpace: "pre-wrap", </CodeBlock>
margin: 0,
backgroundColor: "#48474F",
color: "#FBFBFB",
}}
>
{participant && (
<div>
Laconic Address: {participant.cosmosAddress} <br />
Nitro Address: {participant.nitroAddress} <br />
Role: {participant.role} <br />
KYC ID: {participant.kycId} <br />
<br />
</div>
)}
</pre>
</Box>
{ENABLE_KYC ? ( {ENABLE_KYC ? (
<Box> <Box>
<Typography variant="h5">KYC Status</Typography> <Typography variant="h5">KYC Status</Typography>
@ -148,8 +133,8 @@ const OnboardingSuccess = () => {
padding={5} padding={5}
> >
<Typography variant="body1" gutterBottom sx={{ p: 2 }}> <Typography variant="body1" gutterBottom sx={{ p: 2 }}>
For app publishers, await the start of the stage 1 chain, which will be For app publishers, await the start of the stage 1 chain, which will
announced in various social media channels. In the meantime, be announced in various social media channels. In the meantime,
familiarize yourself with the{" "} familiarize yourself with the{" "}
<a <a
href="https://github.com/hyphacoop/loro-testnet/blob/main/docs/publishing-webapps.md" href="https://github.com/hyphacoop/loro-testnet/blob/main/docs/publishing-webapps.md"

View File

@ -14,6 +14,7 @@ import { useWalletConnectContext } from "../context/WalletConnectContext";
import SelectRoleCard, { Role } from "../components/SelectRoleCard"; import SelectRoleCard, { Role } from "../components/SelectRoleCard";
import { HASHED_SUBSCRIBER_ID_KEY } from "../constants"; import { HASHED_SUBSCRIBER_ID_KEY } from "../constants";
import { Layout } from "../layout/Layout"; import { Layout } from "../layout/Layout";
import { CodeBlock } from "../components/CodeBlock";
const SignWithCosmos = () => { const SignWithCosmos = () => {
const { session, signClient } = useWalletConnectContext(); const { session, signClient } = useWalletConnectContext();
@ -194,26 +195,7 @@ const SignWithCosmos = () => {
<Divider flexItem sx={{ my: 2 }} /> <Divider flexItem sx={{ my: 2 }} />
<Typography variant="body1">Onboarding message:</Typography> <Typography variant="body1">Onboarding message:</Typography>
<Box <CodeBlock>{JSON.stringify(onboardParticipantMsg, null, 2)} </CodeBlock>
sx={{
backgroundColor: "#48474F",
padding: 3,
wordWrap: "break-word",
mt: 1,
borderRadius: 1,
}}
>
<pre
style={{
whiteSpace: "pre-wrap",
margin: 0,
backgroundColor: "#48474F",
color: "#FBFBFB",
}}
>
{JSON.stringify(onboardParticipantMsg, null, 2)}{" "}
</pre>
</Box>
<Box <Box
sx={{ sx={{

View File

@ -10,6 +10,7 @@ import { utf8ToHex } from "@walletconnect/encoding";
import { useWalletConnectContext } from "../context/WalletConnectContext"; import { useWalletConnectContext } from "../context/WalletConnectContext";
import { ENABLE_KYC, HASHED_SUBSCRIBER_ID_KEY } from "../constants"; import { ENABLE_KYC, HASHED_SUBSCRIBER_ID_KEY } from "../constants";
import { Layout } from "../layout/Layout"; import { Layout } from "../layout/Layout";
import { CodeBlock } from "../components/CodeBlock";
const SignWithNitroKey = () => { const SignWithNitroKey = () => {
const { session, signClient, isSessionLoading } = useWalletConnectContext(); const { session, signClient, isSessionLoading } = useWalletConnectContext();
@ -142,26 +143,7 @@ const SignWithNitroKey = () => {
</Select> </Select>
{Boolean(ethAddress) && Boolean(cosmosAddress) && ( {Boolean(ethAddress) && Boolean(cosmosAddress) && (
<Box <CodeBlock>{canonicalStringify(message, null, 2)} </CodeBlock>
sx={{
backgroundColor: "#48474F",
padding: 3,
wordWrap: "break-word",
mt: 1,
borderRadius: 1,
}}
>
<pre
style={{
whiteSpace: "pre-wrap",
margin: 0,
backgroundColor: "#48474F",
color: "#FBFBFB",
}}
>
{canonicalStringify(message, null, 2)}{" "}
</pre>
</Box>
)} )}
<Box> <Box>
<LoadingButton <LoadingButton

View File

@ -1,25 +1,34 @@
import React, { useEffect, useMemo, useState } from 'react'; import React, { useEffect, useMemo, useState } from "react";
import { enqueueSnackbar } from 'notistack'; import { enqueueSnackbar } from "notistack";
import { useNavigate } from 'react-router-dom'; import { useNavigate } from "react-router-dom";
import { MsgCreateValidator } from 'cosmjs-types/cosmos/staking/v1beta1/tx'; import { MsgCreateValidator } from "cosmjs-types/cosmos/staking/v1beta1/tx";
import { Box, Link, MenuItem, Select, TextField, Typography } from '@mui/material'; import {
import { fromBech32, toBech32 } from '@cosmjs/encoding'; Box,
import { LoadingButton } from '@mui/lab'; Link,
import { EncodeObject, encodePubkey } from '@cosmjs/proto-signing'; MenuItem,
import { Registry } from '@cerc-io/registry-sdk'; Select,
TextField,
Typography,
} from "@mui/material";
import { fromBech32, toBech32 } from "@cosmjs/encoding";
import { LoadingButton } from "@mui/lab";
import { EncodeObject, encodePubkey } from "@cosmjs/proto-signing";
import { Registry } from "@cerc-io/registry-sdk";
import { useWalletConnectContext } from '../context/WalletConnectContext'; import { useWalletConnectContext } from "../context/WalletConnectContext";
import { Participant } from '../types'; import { Participant } from "../types";
import { Layout } from "../layout/Layout";
import { CodeBlock } from "../components/CodeBlock";
const Validator = () => { const Validator = () => {
const { session, signClient, isSessionLoading } = useWalletConnectContext(); const { session, signClient, isSessionLoading } = useWalletConnectContext();
const navigate = useNavigate(); const navigate = useNavigate();
const [cosmosAddress, setCosmosAddress] = useState(''); const [cosmosAddress, setCosmosAddress] = useState("");
const [isLoading, setIsLoading] = useState(false); const [isLoading, setIsLoading] = useState(false);
const [moniker, setMoniker] = useState(''); const [moniker, setMoniker] = useState("");
const [pubKey, setPubKey] = useState(''); const [pubKey, setPubKey] = useState("");
const [participant, setParticipant] = useState<Participant | null>(null); const [participant, setParticipant] = useState<Participant | null>(null);
const [isError, setIsError] = useState(false); const [isError, setIsError] = useState(false);
@ -40,10 +49,13 @@ const Validator = () => {
} }
const fetchParticipant = async () => { const fetchParticipant = async () => {
const registry = new Registry(process.env.REACT_APP_REGISTRY_GQL_ENDPOINT!); const registry = new Registry(
process.env.REACT_APP_REGISTRY_GQL_ENDPOINT!,
);
try { try {
const fetchedParticipant = await registry.getParticipantByAddress(cosmosAddress); const fetchedParticipant =
await registry.getParticipantByAddress(cosmosAddress);
if (fetchedParticipant) { if (fetchedParticipant) {
setParticipant(fetchedParticipant); setParticipant(fetchedParticipant);
} else { } else {
@ -65,7 +77,7 @@ const Validator = () => {
const msgCreateValidator: MsgCreateValidator = useMemo(() => { const msgCreateValidator: MsgCreateValidator = useMemo(() => {
const encodedPubKey = encodePubkey({ const encodedPubKey = encodePubkey({
type: "tendermint/PubKeyEd25519", type: "tendermint/PubKeyEd25519",
value: pubKey.length === 44 ? pubKey : '', value: pubKey.length === 44 ? pubKey : "",
}); });
return { return {
@ -82,8 +94,10 @@ const Validator = () => {
rate: "100000000000000000", // 0.1 rate: "100000000000000000", // 0.1
}, },
minSelfDelegation: "1", minSelfDelegation: "1",
delegatorAddress: '', delegatorAddress: "",
validatorAddress: cosmosAddress && toBech32('laconicvaloper', fromBech32(cosmosAddress).data), validatorAddress:
cosmosAddress &&
toBech32("laconicvaloper", fromBech32(cosmosAddress).data),
pubkey: encodedPubKey, pubkey: encodedPubKey,
value: { value: {
amount: process.env.REACT_APP_STAKING_AMOUNT!, amount: process.env.REACT_APP_STAKING_AMOUNT!,
@ -93,7 +107,7 @@ const Validator = () => {
}, [cosmosAddress, pubKey, moniker]); }, [cosmosAddress, pubKey, moniker]);
const msgCreateValidatorEncodeObject: EncodeObject = { const msgCreateValidatorEncodeObject: EncodeObject = {
typeUrl: '/cosmos.staking.v1beta1.MsgCreateValidator', typeUrl: "/cosmos.staking.v1beta1.MsgCreateValidator",
value: MsgCreateValidator.toJSON(msgCreateValidator), value: MsgCreateValidator.toJSON(msgCreateValidator),
}; };
@ -108,10 +122,15 @@ const Validator = () => {
} }
setIsLoading(true); setIsLoading(true);
enqueueSnackbar("View and sign the message from your Laconic Wallet", { variant: "info" }); enqueueSnackbar("View and sign the message from your Laconic Wallet", {
variant: "info",
});
try { try {
const params = { transactionMessage: msgCreateValidatorEncodeObject, signer: cosmosAddress }; const params = {
transactionMessage: msgCreateValidatorEncodeObject,
signer: cosmosAddress,
};
const response = await signClient!.request<{ code: number }>({ const response = await signClient!.request<{ code: number }>({
topic: session!.topic, topic: session!.topic,
chainId: `cosmos:${process.env.REACT_APP_LACONICD_CHAIN_ID}`, chainId: `cosmos:${process.env.REACT_APP_LACONICD_CHAIN_ID}`,
@ -124,7 +143,9 @@ const Validator = () => {
if (response.code !== 0) { if (response.code !== 0) {
throw new Error("Transaction not sent"); throw new Error("Transaction not sent");
} else { } else {
navigate("/validator-success", { state: { validatorAddress: msgCreateValidator.validatorAddress, } }); navigate("/validator-success", {
state: { validatorAddress: msgCreateValidator.validatorAddress },
});
} }
} catch (error) { } catch (error) {
console.error("Error sending transaction", error); console.error("Error sending transaction", error);
@ -134,16 +155,15 @@ const Validator = () => {
} }
}; };
const replacer = (key: string, value: any): any => { const replacer = (_key: string, value: any): any => {
if (value instanceof Uint8Array) { if (value instanceof Uint8Array) {
return Buffer.from(value).toString('hex'); return Buffer.from(value).toString("hex");
} }
return value; return value;
}; };
return ( return (
<Box sx={{ display: "flex", flexDirection: "column", marginTop: 6, gap: 1 }}> <Layout title="Create a validator">
<Typography variant="h5">Create a validator</Typography>
<Typography variant="body1">Select Laconic account:</Typography> <Typography variant="body1">Select Laconic account:</Typography>
<Select <Select
sx={{ marginBottom: 2 }} sx={{ marginBottom: 2 }}
@ -167,23 +187,14 @@ const Validator = () => {
<Typography>No participant found</Typography> <Typography>No participant found</Typography>
)} )}
<Box {participant && (
sx={{ <CodeBlock>
backgroundColor: participant ? "lightgray" : "white", Laconic Address: {participant.cosmosAddress} <br />
padding: 3, Nitro Address: {participant.nitroAddress} <br />
wordWrap: "break-word", Role: {participant.role} <br />
marginBottom: 3, KYC ID: {participant.kycId} <br />
}} </CodeBlock>
> )}
{participant && (
<pre style={{ whiteSpace: "pre-wrap", margin: 0 }}>
Laconic Address: {participant.cosmosAddress} <br />
Nitro Address: {participant.nitroAddress} <br />
Role: {participant.role} <br />
KYC ID: {participant.kycId} <br />
</pre>
)}
</Box>
{participant?.role === "validator" && ( {participant?.role === "validator" && (
<> <>
@ -200,12 +211,15 @@ const Validator = () => {
setMoniker(e.target.value); setMoniker(e.target.value);
}} }}
error={!isMonikerValid && isError} error={!isMonikerValid && isError}
helperText={!isMonikerValid && isError ? "Moniker is required" : ""} helperText={
!isMonikerValid && isError ? "Moniker is required" : ""
}
/> />
</Box> </Box>
<Typography sx={{ marginTop: 3}}> <Typography sx={{ marginTop: 3 }}>
Fetch your validator public key using the following command (refer&nbsp; Fetch your validator public key using the following command
(refer&nbsp;
<Link <Link
href="https://git.vdb.to/cerc-io/testnet-laconicd-stack/src/branch/main/testnet-onboarding-validator.md#join-as-testnet-validator" href="https://git.vdb.to/cerc-io/testnet-laconicd-stack/src/branch/main/testnet-onboarding-validator.md#join-as-testnet-validator"
target="_blank" target="_blank"
@ -213,14 +227,12 @@ const Validator = () => {
> >
this guide this guide
</Link> </Link>
) )
</Typography> </Typography>
<Box sx={{ backgroundColor: "lightgray", padding: 3, wordWrap: "break-word" }}> <CodeBlock>
<pre style={{ whiteSpace: "pre-wrap", margin: 0 }}> {`laconic-so deployment --dir testnet-laconicd-deployment exec laconicd "laconicd cometbft show-validator" | jq -r .key`}
{`laconic-so deployment --dir testnet-laconicd-deployment exec laconicd "laconicd cometbft show-validator" | jq -r .key`} </CodeBlock>
</pre>
</Box>
<Box sx={{ maxWidth: "600px" }}> <Box sx={{ maxWidth: "600px" }}>
<TextField <TextField
@ -235,15 +247,17 @@ const Validator = () => {
setPubKey(e.target.value); setPubKey(e.target.value);
}} }}
error={!isPubKeyValid && isError} error={!isPubKeyValid && isError}
helperText={!isPubKeyValid && isError ? "Public key must be 44 characters" : ""} helperText={
!isPubKeyValid && isError
? "Public key must be 44 characters"
: ""
}
/> />
</Box> </Box>
<Typography>Send transaction to chain</Typography> <Typography>Send transaction to chain</Typography>
<Box sx={{ backgroundColor: "lightgray", padding: 3, wordWrap: "break-word" }}> <CodeBlock>
<pre style={{ whiteSpace: "pre-wrap", margin: 0 }}> {JSON.stringify(msgCreateValidator, replacer, 2)}
{JSON.stringify(msgCreateValidator, replacer, 2)} </CodeBlock>
</pre>
</Box>
<Box marginTop={1} marginBottom={1}> <Box marginTop={1} marginBottom={1}>
<LoadingButton <LoadingButton
variant="contained" variant="contained"
@ -258,7 +272,7 @@ const Validator = () => {
)} )}
</> </>
)} )}
</Box> </Layout>
); );
}; };

View File

@ -1,26 +1,21 @@
import React from 'react'; import React from "react";
import { useLocation } from 'react-router-dom'; import { useLocation } from "react-router-dom";
import { Box, Link, Typography } from '@mui/material'; import { Link, Typography } from "@mui/material";
import { Layout } from "../layout/Layout";
import { CodeBlock } from "../components/CodeBlock";
const ValidatorSuccess = () => { const ValidatorSuccess = () => {
const location = useLocation(); const location = useLocation();
const { validatorAddress } = location.state as { const { validatorAddress } = location.state as {
validatorAddress?: string validatorAddress?: string;
}; };
return ( return (
<Box <Layout title="Validator created successfully">
sx={{ <Typography sx={{ marginTop: 3 }}>
display: "flex", You can view your validator details using the following command
flexDirection: "column", (Refer&nbsp;
marginTop: 6,
gap: 1,
}}
>
<Typography variant="h5">Validator created successfully</Typography>
<Typography sx={{ marginTop: 3}}>
You can view your validator details using the following command (Refer&nbsp;
<Link <Link
href="https://git.vdb.to/cerc-io/testnet-laconicd-stack/src/branch/main/testnet-onboarding-validator.md#join-as-testnet-validator" href="https://git.vdb.to/cerc-io/testnet-laconicd-stack/src/branch/main/testnet-onboarding-validator.md#join-as-testnet-validator"
target="_blank" target="_blank"
@ -28,14 +23,12 @@ const ValidatorSuccess = () => {
> >
this guide this guide
</Link> </Link>
) )
</Typography> </Typography>
<Box sx={{ backgroundColor: "lightgray", padding: 2, wordWrap: "break-word", marginTop: 2, fontSize: 14}}> <CodeBlock>
<pre style={{ whiteSpace: "pre-wrap", margin: 0 }}> {`laconic-so deployment --dir testnet-laconicd-deployment exec laconicd "laconicd query staking validators --output json" | jq '.validators[] | select(.operator_address == "${validatorAddress}")'`}
{`laconic-so deployment --dir testnet-laconicd-deployment exec laconicd "laconicd query staking validators --output json" | jq '.validators[] | select(.operator_address == "${validatorAddress}")'`} </CodeBlock>
</pre> </Layout>
</Box>
</Box>
); );
}; };