706 lines
19 KiB
TypeScript
706 lines
19 KiB
TypeScript
import * as amplitude from "@amplitude/analytics-browser";
|
|
|
|
// React
|
|
import { useEffect, useState } from "react";
|
|
|
|
// Types
|
|
import {
|
|
ChainItemType,
|
|
DisabledChainItemType,
|
|
ErrorMessage,
|
|
QueryError,
|
|
RegisteredAddresses,
|
|
TwitterProfileType,
|
|
} from "../../types";
|
|
import { checkTwitterAuthQueryParameter } from "../../utils/url";
|
|
|
|
// Styles
|
|
import styled from "styled-components";
|
|
import color from "../../styles/color";
|
|
|
|
// Components
|
|
import { Logo } from "../../components/logo";
|
|
import { SkeletonChainList } from "../../components/skeleton";
|
|
|
|
import { PrimaryButton } from "../../components/primary-button";
|
|
import { TwitterProfile } from "../../components/twitter-profile";
|
|
import { ChainList } from "../../components/chain-list";
|
|
import { useRouter } from "next/router";
|
|
import {
|
|
getKeplrFromWindow,
|
|
KeplrWallet,
|
|
sendMsgs,
|
|
simulateMsgs,
|
|
} from "../../wallets";
|
|
import { Bech32Address, ChainIdHelper } from "@keplr-wallet/cosmos";
|
|
|
|
import { AllChainsItem } from "../../components/chain-list/all-chains-item";
|
|
import { SearchInput } from "../../components/search-input";
|
|
import {
|
|
CHAIN_ALLOWLIST,
|
|
MAIN_CHAIN_ID,
|
|
REFERRAL_KEY,
|
|
RESOLVER_ADDRESS,
|
|
REST_URL,
|
|
} from "../../constants/icns";
|
|
|
|
import {
|
|
fetchTwitterInfo,
|
|
queryAddressesFromTwitterName,
|
|
queryOwnerOfTwitterName,
|
|
queryRegisteredTwitterId,
|
|
verifyTwitterAccount,
|
|
} from "../../queries";
|
|
import {
|
|
ACCOUNT_NOT_EXIST_ERROR,
|
|
ACCOUNT_NOT_EXIST_MESSAGE,
|
|
EVM_CHAIN_IN_LEDGER_ERROR,
|
|
EVM_CHAIN_IN_LEDGER_MESSAGE,
|
|
INSUFFICIENT_GAS_ERROR,
|
|
INSUFFICIENT_GAS_MESSAGE,
|
|
INVALID_REFERRAL_ERROR,
|
|
INVALID_REFERRAL_MESSAGE,
|
|
KEPLR_NOT_FOUND_ERROR,
|
|
TOO_MANY_CHAINS_IN_LEDGER_ERROR,
|
|
TOO_MANY_CHAINS_IN_LEDGER_MESSAGE,
|
|
TWITTER_LOGIN_ERROR,
|
|
VERIFICATION_THRESHOLD_ERROR,
|
|
VERIFICATION_THRESHOLD_MESSAGE,
|
|
} from "../../constants/error-message";
|
|
import { makeClaimMessage, makeSetRecordMessage } from "../../messages";
|
|
import Axios from "axios";
|
|
import { BackButton } from "../../components/back-button";
|
|
import { FinalCheckModal } from "../../components/final-check-modal";
|
|
import { ErrorModal } from "../../components/error-modal";
|
|
import { captureException } from "@sentry/nextjs";
|
|
|
|
export default function VerificationPage() {
|
|
const router = useRouter();
|
|
const [twitterAuthInfo, setTwitterAuthInfo] = useState<TwitterProfileType>();
|
|
|
|
const [isLoadingInit, setIsLoadingInit] = useState(true);
|
|
const [isLoadingRegistration, setIsLoadingRegistration] = useState(false);
|
|
|
|
const [wallet, setWallet] = useState<KeplrWallet>();
|
|
const [walletKey, setWalletKey] = useState<{
|
|
name: string;
|
|
pubKey: Uint8Array;
|
|
bech32Address: string;
|
|
isLedgerNano?: boolean;
|
|
}>();
|
|
|
|
const [chainList, setChainList] = useState<
|
|
(ChainItemType & {
|
|
isEthermintLike?: boolean;
|
|
})[]
|
|
>([]);
|
|
const [disabledChainList, setDisabledChainList] = useState<
|
|
DisabledChainItemType[]
|
|
>([]);
|
|
const [registeredChainList, setRegisteredChainList] = useState<
|
|
RegisteredAddresses[]
|
|
>([]);
|
|
const [checkedItems, setCheckedItems] = useState(new Set());
|
|
|
|
const [searchValue, setSearchValue] = useState("");
|
|
|
|
const [nftOwnerAddress, setNFTOwnerAddress] = useState("");
|
|
const [isOwner, setIsOwner] = useState(false);
|
|
|
|
const [isModalOpen, setModalOpen] = useState(false);
|
|
const [isErrorModalOpen, setErrorModalOpen] = useState(false);
|
|
const [errorMessage, setErrorMessage] = useState<ErrorMessage>();
|
|
|
|
useEffect(() => {
|
|
init();
|
|
}, []);
|
|
|
|
useEffect(() => {
|
|
if (wallet) {
|
|
window.addEventListener("keplr_keystorechange", async () => {
|
|
await init();
|
|
});
|
|
}
|
|
}, [wallet]);
|
|
|
|
useEffect(() => {
|
|
const disabledChainList = chainList
|
|
.filter((chain) => {
|
|
if (!chain.address) {
|
|
// Address can be "" if `getKey` failed.
|
|
return true;
|
|
}
|
|
|
|
if (walletKey) {
|
|
if (walletKey.isLedgerNano && chain.isEthermintLike) {
|
|
return true;
|
|
}
|
|
}
|
|
|
|
for (const registeredChain of registeredChainList) {
|
|
if (
|
|
chain.prefix === registeredChain.bech32_prefix &&
|
|
chain.address === registeredChain.address
|
|
) {
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
})
|
|
.map<DisabledChainItemType>((chain) => {
|
|
if (walletKey) {
|
|
if (walletKey.isLedgerNano && chain.isEthermintLike) {
|
|
return {
|
|
...chain,
|
|
disabled: true,
|
|
reason: new Error(
|
|
"Support for Ethereum address on Ledger is coming soon.",
|
|
),
|
|
};
|
|
}
|
|
}
|
|
|
|
return {
|
|
...chain,
|
|
disabled: true,
|
|
};
|
|
});
|
|
|
|
const filteredChainList = chainList.filter((chain) => {
|
|
return (
|
|
disabledChainList.find(
|
|
(disabled) => disabled.chainId === chain.chainId,
|
|
) == null
|
|
);
|
|
});
|
|
|
|
setChainList(filteredChainList);
|
|
setDisabledChainList(disabledChainList);
|
|
|
|
setCheckedItems(new Set(filteredChainList));
|
|
}, [registeredChainList]);
|
|
|
|
useEffect(() => {
|
|
setCheckedItems(new Set(chainList));
|
|
}, [chainList]);
|
|
|
|
const init = async () => {
|
|
if (window.location.search) {
|
|
try {
|
|
const { state, code } = checkTwitterAuthQueryParameter(
|
|
window.location.search,
|
|
);
|
|
|
|
// Initialize Wallet
|
|
const keplrWallet = await initWallet();
|
|
|
|
// Fetch Twitter Profile
|
|
const twitterInfo = await fetchTwitterInfo(state, code);
|
|
|
|
// contract check registered
|
|
const registeredQueryResponse = await queryRegisteredTwitterId(
|
|
twitterInfo.id,
|
|
);
|
|
|
|
setTwitterAuthInfo({
|
|
...twitterInfo,
|
|
isRegistered: "data" in registeredQueryResponse,
|
|
});
|
|
|
|
if ("data" in registeredQueryResponse) {
|
|
const ownerOfQueryResponse = await queryOwnerOfTwitterName(
|
|
registeredQueryResponse.data.name,
|
|
);
|
|
|
|
if (keplrWallet) {
|
|
const key = await keplrWallet.getKey(MAIN_CHAIN_ID);
|
|
setIsOwner(ownerOfQueryResponse.data.owner === key.bech32Address);
|
|
setNFTOwnerAddress(ownerOfQueryResponse.data.owner);
|
|
}
|
|
|
|
const addressesQueryResponse = await queryAddressesFromTwitterName(
|
|
registeredQueryResponse.data.name,
|
|
);
|
|
|
|
setRegisteredChainList(addressesQueryResponse.data.addresses);
|
|
} else {
|
|
setRegisteredChainList([]);
|
|
}
|
|
} catch (error) {
|
|
if (error instanceof Error) {
|
|
if (error.message === TWITTER_LOGIN_ERROR) {
|
|
setErrorMessage({ message: TWITTER_LOGIN_ERROR, path: "/" });
|
|
setErrorModalOpen(true);
|
|
}
|
|
}
|
|
|
|
console.error(error);
|
|
} finally {
|
|
setIsLoadingInit(false);
|
|
}
|
|
}
|
|
};
|
|
|
|
const initWallet = async () => {
|
|
const keplr = await getKeplrFromWindow();
|
|
|
|
if (keplr) {
|
|
const keplrWallet = new KeplrWallet(keplr);
|
|
const key = await keplrWallet.getKey(MAIN_CHAIN_ID);
|
|
|
|
await fetchChainList(keplrWallet);
|
|
setWallet(keplrWallet);
|
|
setWalletKey(key);
|
|
|
|
return keplrWallet;
|
|
} else {
|
|
setErrorMessage({ message: KEPLR_NOT_FOUND_ERROR, path: "/" });
|
|
setErrorModalOpen(true);
|
|
}
|
|
};
|
|
|
|
const fetchChainList = async (wallet: KeplrWallet) => {
|
|
const needAllowList =
|
|
CHAIN_ALLOWLIST != null && CHAIN_ALLOWLIST.trim().length !== 0;
|
|
const chainAllowList = (CHAIN_ALLOWLIST || "")
|
|
.split(",")
|
|
.map((str) => str.trim())
|
|
.filter((str) => str.length > 0)
|
|
.map((str) => ChainIdHelper.parse(str).identifier);
|
|
|
|
const chainAllowListMap = new Map<string, true | undefined>();
|
|
for (const allow of chainAllowList) {
|
|
chainAllowListMap.set(allow, true);
|
|
}
|
|
|
|
const chainInfos = (await wallet.getChainInfosWithoutEndpoints())
|
|
.filter((chainInfo) => {
|
|
if (!needAllowList) {
|
|
return true;
|
|
}
|
|
|
|
const chainIdentifier = ChainIdHelper.parse(
|
|
chainInfo.chainId,
|
|
).identifier;
|
|
|
|
return chainAllowListMap.get(chainIdentifier) === true;
|
|
})
|
|
.map((chainInfo) => {
|
|
return {
|
|
chainId: chainInfo.chainId,
|
|
chainName: chainInfo.chainName,
|
|
prefix: chainInfo.bech32Config.bech32PrefixAccAddr,
|
|
chainImageUrl: `https://raw.githubusercontent.com/chainapsis/keplr-chain-registry/main/images/${
|
|
ChainIdHelper.parse(chainInfo.chainId).identifier
|
|
}/chain.png`,
|
|
isEthermintLike: chainInfo.isEthermintLike,
|
|
};
|
|
});
|
|
|
|
const chainIds = chainInfos.map((c) => c.chainId);
|
|
const chainKeys = await Promise.allSettled(
|
|
chainIds.map((chainId) => wallet.getKey(chainId)),
|
|
);
|
|
|
|
const chainArray = [];
|
|
for (let i = 0; i < chainKeys.length; i++) {
|
|
const chainKey = chainKeys[i];
|
|
if (chainKey.status !== "fulfilled") {
|
|
console.log("Failed to get key from wallet", chainKey);
|
|
}
|
|
chainArray.push({
|
|
address:
|
|
chainKey.status === "fulfilled" ? chainKey.value.bech32Address : "",
|
|
...chainInfos[i],
|
|
});
|
|
}
|
|
|
|
// remove duplicated item
|
|
const filteredChainList = chainArray.filter((nextChain, index, self) => {
|
|
return (
|
|
index ===
|
|
self.findIndex((prevChain) => {
|
|
const isDuplicated = prevChain.prefix === nextChain.prefix;
|
|
|
|
if (isDuplicated && prevChain.chainName !== nextChain.chainName) {
|
|
console.log(
|
|
`${nextChain.chainName} has been deleted due to a duplicate name with ${prevChain.chainName}`,
|
|
);
|
|
}
|
|
|
|
return isDuplicated;
|
|
})
|
|
);
|
|
});
|
|
|
|
setChainList(filteredChainList);
|
|
};
|
|
|
|
const checkAdr36 = async () => {
|
|
if (twitterAuthInfo && wallet) {
|
|
const key = await wallet.getKey(MAIN_CHAIN_ID);
|
|
|
|
const chainIds = Array.from(checkedItems).map((chain) => {
|
|
return (chain as ChainItemType).chainId;
|
|
});
|
|
|
|
return wallet.signICNSAdr36(
|
|
MAIN_CHAIN_ID,
|
|
RESOLVER_ADDRESS,
|
|
key.bech32Address,
|
|
twitterAuthInfo.username,
|
|
chainIds,
|
|
);
|
|
}
|
|
};
|
|
|
|
const onClickRegistration = () => {
|
|
amplitude.track("click register button");
|
|
|
|
if (isOwner) {
|
|
handleRegistration();
|
|
} else {
|
|
setModalOpen(true);
|
|
}
|
|
};
|
|
|
|
const handleRegistration = async () => {
|
|
try {
|
|
setIsLoadingRegistration(true);
|
|
|
|
const { state, code } = checkTwitterAuthQueryParameter(
|
|
window.location.search,
|
|
);
|
|
const twitterInfo = await fetchTwitterInfo(state, code);
|
|
|
|
const adr36Infos = await checkAdr36();
|
|
|
|
if (wallet && walletKey && adr36Infos) {
|
|
const icnsVerificationList = await verifyTwitterAccount(
|
|
walletKey.bech32Address,
|
|
twitterInfo.accessToken,
|
|
);
|
|
|
|
icnsVerificationList.forEach((verification) => {
|
|
if (verification.status === "fulfilled") {
|
|
if (verification.value.errors.length > 0) {
|
|
verification.value.errors.forEach((error) => {
|
|
const errorMessage = error.message || error;
|
|
captureException(errorMessage);
|
|
});
|
|
}
|
|
}
|
|
|
|
if (verification.status === "rejected") {
|
|
if (verification.reason) {
|
|
const errorMessage =
|
|
verification.reason.message || verification.reason;
|
|
captureException(errorMessage);
|
|
}
|
|
}
|
|
});
|
|
|
|
const registerMsg = makeClaimMessage(
|
|
walletKey.bech32Address,
|
|
twitterInfo.username,
|
|
icnsVerificationList,
|
|
localStorage.getItem(REFERRAL_KEY) ?? undefined,
|
|
);
|
|
|
|
const addressMsgs = adr36Infos.map((adr36Info) => {
|
|
return makeSetRecordMessage(
|
|
walletKey.bech32Address,
|
|
twitterInfo.username,
|
|
adr36Info,
|
|
);
|
|
});
|
|
|
|
const aminoMsgs = twitterAuthInfo?.isRegistered
|
|
? []
|
|
: [registerMsg.amino];
|
|
const protoMsgs = twitterAuthInfo?.isRegistered
|
|
? []
|
|
: [registerMsg.proto];
|
|
|
|
for (const addressMsg of addressMsgs) {
|
|
aminoMsgs.push(addressMsg.amino);
|
|
protoMsgs.push(addressMsg.proto);
|
|
}
|
|
|
|
const chainInfo = {
|
|
chainId: MAIN_CHAIN_ID,
|
|
rest: REST_URL,
|
|
};
|
|
|
|
console.log(aminoMsgs);
|
|
|
|
const simulated = await simulateMsgs(
|
|
chainInfo,
|
|
walletKey.bech32Address,
|
|
{
|
|
proto: protoMsgs,
|
|
},
|
|
{
|
|
amount: [],
|
|
},
|
|
);
|
|
|
|
const txHash = await sendMsgs(
|
|
wallet,
|
|
chainInfo,
|
|
walletKey.bech32Address,
|
|
{
|
|
amino: aminoMsgs,
|
|
proto: protoMsgs,
|
|
},
|
|
{
|
|
amount: [],
|
|
gas: Math.floor(simulated.gasUsed * 1.5).toString(),
|
|
},
|
|
);
|
|
|
|
await router.push({
|
|
pathname: "complete",
|
|
query: {
|
|
txHash: Buffer.from(txHash).toString("hex"),
|
|
twitterUsername: twitterInfo.username,
|
|
},
|
|
});
|
|
}
|
|
} catch (error) {
|
|
if (Axios.isAxiosError(error)) {
|
|
const message = (error?.response?.data as QueryError).message;
|
|
|
|
if (message.includes(INSUFFICIENT_GAS_ERROR)) {
|
|
setErrorMessage({ message: INSUFFICIENT_GAS_MESSAGE });
|
|
setErrorModalOpen(true);
|
|
return;
|
|
}
|
|
|
|
if (message.includes(ACCOUNT_NOT_EXIST_ERROR)) {
|
|
setErrorMessage({ message: ACCOUNT_NOT_EXIST_MESSAGE });
|
|
setErrorModalOpen(true);
|
|
return;
|
|
}
|
|
|
|
if (message.includes(VERIFICATION_THRESHOLD_ERROR)) {
|
|
setErrorMessage({ message: VERIFICATION_THRESHOLD_MESSAGE });
|
|
setErrorModalOpen(true);
|
|
return;
|
|
}
|
|
|
|
if (message.includes(INVALID_REFERRAL_ERROR)) {
|
|
setErrorMessage({ message: INVALID_REFERRAL_MESSAGE });
|
|
setErrorModalOpen(true);
|
|
return;
|
|
}
|
|
|
|
const errorMessage = error.message || error;
|
|
captureException(errorMessage);
|
|
setErrorMessage({
|
|
message: (error?.response?.data as QueryError).message,
|
|
});
|
|
setErrorModalOpen(true);
|
|
return;
|
|
}
|
|
|
|
if (error instanceof Error) {
|
|
const errorMessage = error.message || error;
|
|
console.log(errorMessage);
|
|
captureException(errorMessage);
|
|
|
|
if (error.message.includes(TOO_MANY_CHAINS_IN_LEDGER_ERROR)) {
|
|
setErrorMessage({ message: TOO_MANY_CHAINS_IN_LEDGER_MESSAGE });
|
|
setErrorModalOpen(true);
|
|
return;
|
|
}
|
|
|
|
if (error.message.includes(EVM_CHAIN_IN_LEDGER_ERROR)) {
|
|
setErrorMessage({ message: EVM_CHAIN_IN_LEDGER_MESSAGE });
|
|
setErrorModalOpen(true);
|
|
return;
|
|
}
|
|
|
|
setErrorMessage({ message: error.message });
|
|
setErrorModalOpen(true);
|
|
}
|
|
} finally {
|
|
setIsLoadingRegistration(false);
|
|
}
|
|
};
|
|
|
|
const isRegisterButtonDisable = (() => {
|
|
if (!isOwner && nftOwnerAddress) {
|
|
return true;
|
|
}
|
|
|
|
const hasCheckedItem = checkedItems.size > 0;
|
|
|
|
return !hasCheckedItem;
|
|
})();
|
|
|
|
return (
|
|
<Container>
|
|
<Logo />
|
|
|
|
<MainContainer>
|
|
{isLoadingInit ? (
|
|
<SkeletonChainList />
|
|
) : (
|
|
<ContentContainer>
|
|
<BackButton />
|
|
<TwitterProfile twitterProfileInformation={twitterAuthInfo} />
|
|
<ChainListTitleContainer>
|
|
<ChainListTitle>Chain List</ChainListTitle>
|
|
<SearchInput
|
|
searchValue={searchValue}
|
|
setSearchValue={setSearchValue}
|
|
/>
|
|
</ChainListTitleContainer>
|
|
{!searchValue ? (
|
|
<AllChainsItem
|
|
chainList={chainList}
|
|
checkedItems={checkedItems}
|
|
setCheckedItems={setCheckedItems}
|
|
/>
|
|
) : null}
|
|
<ChainList
|
|
chainList={chainList.filter(
|
|
(chain) =>
|
|
chain.chainId.includes(searchValue) ||
|
|
chain.address.includes(searchValue) ||
|
|
chain.prefix.includes(searchValue),
|
|
)}
|
|
disabledChainList={disabledChainList.filter(
|
|
(chain) =>
|
|
chain.chainId.includes(searchValue) ||
|
|
chain.address.includes(searchValue) ||
|
|
chain.prefix.includes(searchValue),
|
|
)}
|
|
checkedItems={checkedItems}
|
|
setCheckedItems={setCheckedItems}
|
|
/>
|
|
|
|
{!isOwner && nftOwnerAddress ? (
|
|
<OwnerAlert>
|
|
You are not owner of this name.
|
|
<br />
|
|
Please select the account{" "}
|
|
<AddressBold>
|
|
({Bech32Address.shortenAddress(nftOwnerAddress, 28)})
|
|
</AddressBold>
|
|
</OwnerAlert>
|
|
) : null}
|
|
|
|
{chainList.length > 0 && (
|
|
<ButtonContainer disabled={isRegisterButtonDisable}>
|
|
<PrimaryButton
|
|
disabled={isRegisterButtonDisable}
|
|
onClick={onClickRegistration}
|
|
isLoading={isLoadingRegistration}
|
|
>
|
|
Register
|
|
</PrimaryButton>
|
|
</ButtonContainer>
|
|
)}
|
|
</ContentContainer>
|
|
)}
|
|
</MainContainer>
|
|
|
|
<FinalCheckModal
|
|
twitterUserName={twitterAuthInfo?.username}
|
|
walletInfo={walletKey}
|
|
isModalOpen={isModalOpen}
|
|
onCloseModal={() => setModalOpen(false)}
|
|
onClickRegisterButton={handleRegistration}
|
|
isLoadingRegistration={isLoadingRegistration}
|
|
/>
|
|
|
|
<ErrorModal
|
|
isModalOpen={isErrorModalOpen}
|
|
onCloseModal={() => setErrorModalOpen(false)}
|
|
errorMessage={errorMessage}
|
|
/>
|
|
</Container>
|
|
);
|
|
}
|
|
|
|
const Container = styled.div`
|
|
width: 100vw;
|
|
height: 100vh;
|
|
`;
|
|
|
|
const MainContainer = styled.div`
|
|
display: flex;
|
|
justify-content: center;
|
|
|
|
height: 100vh;
|
|
|
|
padding: 2.7rem 0;
|
|
|
|
color: white;
|
|
|
|
background-image: url("/images/svg/bg-asset-3.svg"),
|
|
url("/images/svg/bg-asset-3.svg"), url("/images/svg/bg-asset-3.svg"),
|
|
url("/images/svg/bg-asset-3.svg");
|
|
background-size: 3.125rem 3.125rem;
|
|
background-position: 296px 536px, 1256px 216px, 376px 776px, 1176px 856px;
|
|
background-repeat: no-repeat;
|
|
`;
|
|
|
|
export const ContentContainer = styled.div`
|
|
display: flex;
|
|
flex-direction: column;
|
|
align-items: center;
|
|
|
|
width: 40rem;
|
|
`;
|
|
|
|
export const ButtonContainer = styled.div<{ disabled?: boolean }>`
|
|
width: 11rem;
|
|
height: 3.5rem;
|
|
|
|
margin-top: 1.5rem;
|
|
|
|
background-color: ${(props) =>
|
|
props.disabled ? color.orange["300"] : color.orange["100"]};
|
|
`;
|
|
|
|
export const ChainListTitleContainer = styled.div`
|
|
display: flex;
|
|
flex-direction: row;
|
|
justify-content: space-between;
|
|
align-items: center;
|
|
|
|
width: 100%;
|
|
|
|
margin-top: 2rem;
|
|
margin-bottom: 1rem;
|
|
`;
|
|
|
|
const ChainListTitle = styled.div`
|
|
font-weight: 700;
|
|
font-size: 1.5rem;
|
|
line-height: 1.9rem;
|
|
|
|
color: ${color.white};
|
|
`;
|
|
|
|
const OwnerAlert = styled.div`
|
|
font-family: "Inter", serif;
|
|
font-style: normal;
|
|
font-weight: 500;
|
|
font-size: 0.9rem;
|
|
line-height: 1.1rem;
|
|
text-align: center;
|
|
|
|
color: ${color.grey["400"]};
|
|
|
|
padding-top: 1.25rem;
|
|
`;
|
|
|
|
const AddressBold = styled.span`
|
|
color: ${color.white};
|
|
`;
|