From 9be22ce15b9e956ebd813f630e3ae050b2c648c3 Mon Sep 17 00:00:00 2001 From: Paul <41552663+molecula451@users.noreply.github.com> Date: Mon, 4 Mar 2024 01:58:23 -0400 Subject: [PATCH 1/2] Create CODEOWNERS --- .github/CODEOWNERS | 3 +++ 1 file changed, 3 insertions(+) create mode 100644 .github/CODEOWNERS diff --git a/.github/CODEOWNERS b/.github/CODEOWNERS new file mode 100644 index 0000000..17955a8 --- /dev/null +++ b/.github/CODEOWNERS @@ -0,0 +1,3 @@ +* @rndquu +*.ts @pavlovcik +/static @pavlovcik From 4181d261a20cb14b732a3f1f595b1d44c1e2d983 Mon Sep 17 00:00:00 2001 From: Keyrxng <106303466+Keyrxng@users.noreply.github.com> Date: Mon, 4 Mar 2024 14:20:22 +0000 Subject: [PATCH 2/2] chore: remove rewards app, update permit types (#6) --- .cspell.json | 3 +- .env.example | 3 +- .github/workflows/knip.yml | 2 +- .gitignore | 2 +- README.md | 3 + build/esbuild-build.ts | 2 +- knip.ts | 2 +- .../{rewards => audit-report}/abis/cirip.json | 0 .../abis/erc20Abi.ts | 0 .../{rewards => audit-report}/abis/index.ts | 0 .../abis/nftRewardAbi.ts | 0 .../abis/permit2Abi.ts | 0 static/scripts/audit-report/audit.ts | 6 +- static/scripts/audit-report/types/audit.d.ts | 58 ++++++ static/scripts/rewards/cirip/ens-lookup.ts | 67 ------- static/scripts/rewards/cirip/fetch-ens.ts | 12 -- static/scripts/rewards/cirip/query-graph.ts | 12 -- .../rewards/cirip/query-reverse-ens.ts | 20 -- static/scripts/rewards/constants.ts | 56 ------ static/scripts/rewards/helpers.ts | 80 -------- static/scripts/rewards/index.ts | 20 -- .../scripts/rewards/invalidate-component.ts | 12 -- .../rewards/render-transaction/index.ts | 42 ---- .../render-transaction/insert-table-data.ts | 129 ------------ .../render-transaction/render-ens-name.ts | 42 ---- .../render-transaction/render-token-symbol.ts | 56 ------ .../render-transaction/render-transaction.ts | 155 --------------- .../render-transaction/set-claim-message.ts | 6 - .../rewards/render-transaction/tx-type.ts | 68 ------- .../rewards/render-transaction/utils.ts | 5 - static/scripts/rewards/the-grid.ts | 160 --------------- static/scripts/rewards/toaster.ts | 89 --------- static/scripts/rewards/web3/erc20-permit.ts | 183 ------------------ static/scripts/rewards/web3/erc721-permit.ts | 64 ------ static/scripts/rewards/web3/wallet.ts | 103 ---------- 35 files changed, 72 insertions(+), 1390 deletions(-) rename static/scripts/{rewards => audit-report}/abis/cirip.json (100%) rename static/scripts/{rewards => audit-report}/abis/erc20Abi.ts (100%) rename static/scripts/{rewards => audit-report}/abis/index.ts (100%) rename static/scripts/{rewards => audit-report}/abis/nftRewardAbi.ts (100%) rename static/scripts/{rewards => audit-report}/abis/permit2Abi.ts (100%) delete mode 100644 static/scripts/rewards/cirip/ens-lookup.ts delete mode 100644 static/scripts/rewards/cirip/fetch-ens.ts delete mode 100644 static/scripts/rewards/cirip/query-graph.ts delete mode 100644 static/scripts/rewards/cirip/query-reverse-ens.ts delete mode 100644 static/scripts/rewards/constants.ts delete mode 100644 static/scripts/rewards/helpers.ts delete mode 100644 static/scripts/rewards/index.ts delete mode 100644 static/scripts/rewards/invalidate-component.ts delete mode 100644 static/scripts/rewards/render-transaction/index.ts delete mode 100644 static/scripts/rewards/render-transaction/insert-table-data.ts delete mode 100644 static/scripts/rewards/render-transaction/render-ens-name.ts delete mode 100644 static/scripts/rewards/render-transaction/render-token-symbol.ts delete mode 100644 static/scripts/rewards/render-transaction/render-transaction.ts delete mode 100644 static/scripts/rewards/render-transaction/set-claim-message.ts delete mode 100644 static/scripts/rewards/render-transaction/tx-type.ts delete mode 100644 static/scripts/rewards/render-transaction/utils.ts delete mode 100644 static/scripts/rewards/the-grid.ts delete mode 100644 static/scripts/rewards/toaster.ts delete mode 100644 static/scripts/rewards/web3/erc20-permit.ts delete mode 100644 static/scripts/rewards/web3/erc721-permit.ts delete mode 100644 static/scripts/rewards/web3/wallet.ts diff --git a/.cspell.json b/.cspell.json index d7ec736..f1d2c04 100644 --- a/.cspell.json +++ b/.cspell.json @@ -6,11 +6,13 @@ "language": "en", "words": [ "binkey", + "Knip", "binsec", "chainlist", "cirip", "dataurl", "devpool", + "xrandomwallet", "ethersproject", "fract", "gnosisscan", @@ -37,7 +39,6 @@ "xmark", "supabase", "SUPABASE" - ], "dictionaries": ["typescript", "node", "software-terms", "html"], "import": ["@cspell/dict-typescript/cspell-ext.json", "@cspell/dict-node/cspell-ext.json", "@cspell/dict-software-terms"], diff --git a/.env.example b/.env.example index e49d79a..f7321ea 100644 --- a/.env.example +++ b/.env.example @@ -1 +1,2 @@ -MY_SECRET="MY_SECRET" +SUPABASE_URL= +SUPABASE_ANON_KEY= \ No newline at end of file diff --git a/.github/workflows/knip.yml b/.github/workflows/knip.yml index df68bda..a16c31f 100644 --- a/.github/workflows/knip.yml +++ b/.github/workflows/knip.yml @@ -27,4 +27,4 @@ jobs: comment_id: ${{ github.workflow }}-reporter command_script_name: knip-ci annotations: true - ignore_results: false \ No newline at end of file + ignore_results: false diff --git a/.gitignore b/.gitignore index 7791818..ff82a3b 100644 --- a/.gitignore +++ b/.gitignore @@ -7,4 +7,4 @@ node_modules .pnp.cjs .pnp.loader.mjs .env -static/dist \ No newline at end of file +static/out \ No newline at end of file diff --git a/README.md b/README.md index ade7a5a..c1913c7 100644 --- a/README.md +++ b/README.md @@ -1,9 +1,11 @@ # `@ubiquity/audit.ubq.fi` #### Install Dependencies + - yarn #### Build + - yarn build ### Start UI @@ -18,3 +20,4 @@ Test Uses: "REPO": "https://github.com/repo/repo", "PAT": "YOUR_PAT" } +``` diff --git a/build/esbuild-build.ts b/build/esbuild-build.ts index 94f52ed..9f61a25 100644 --- a/build/esbuild-build.ts +++ b/build/esbuild-build.ts @@ -1,7 +1,7 @@ import esbuild from "esbuild"; import * as dotenv from "dotenv"; -const typescriptEntries = ["static/scripts/rewards/index.ts", "static/scripts/audit-report/audit.ts"]; +const typescriptEntries = ["static/scripts/audit-report/audit.ts"]; const cssEntries = ["static/styles/rewards/rewards.css", "static/styles/audit-report/audit.css"]; export const entries = [...typescriptEntries, ...cssEntries]; diff --git a/knip.ts b/knip.ts index 57b5e39..a749511 100644 --- a/knip.ts +++ b/knip.ts @@ -8,4 +8,4 @@ const config: KnipConfig = { ignoreDependencies: [], }; -export default config; \ No newline at end of file +export default config; diff --git a/static/scripts/rewards/abis/cirip.json b/static/scripts/audit-report/abis/cirip.json similarity index 100% rename from static/scripts/rewards/abis/cirip.json rename to static/scripts/audit-report/abis/cirip.json diff --git a/static/scripts/rewards/abis/erc20Abi.ts b/static/scripts/audit-report/abis/erc20Abi.ts similarity index 100% rename from static/scripts/rewards/abis/erc20Abi.ts rename to static/scripts/audit-report/abis/erc20Abi.ts diff --git a/static/scripts/rewards/abis/index.ts b/static/scripts/audit-report/abis/index.ts similarity index 100% rename from static/scripts/rewards/abis/index.ts rename to static/scripts/audit-report/abis/index.ts diff --git a/static/scripts/rewards/abis/nftRewardAbi.ts b/static/scripts/audit-report/abis/nftRewardAbi.ts similarity index 100% rename from static/scripts/rewards/abis/nftRewardAbi.ts rename to static/scripts/audit-report/abis/nftRewardAbi.ts diff --git a/static/scripts/rewards/abis/permit2Abi.ts b/static/scripts/audit-report/abis/permit2Abi.ts similarity index 100% rename from static/scripts/rewards/abis/permit2Abi.ts rename to static/scripts/audit-report/abis/permit2Abi.ts diff --git a/static/scripts/audit-report/audit.ts b/static/scripts/audit-report/audit.ts index ed9f862..6e08e5e 100644 --- a/static/scripts/audit-report/audit.ts +++ b/static/scripts/audit-report/audit.ts @@ -4,7 +4,7 @@ import { createClient } from "@supabase/supabase-js"; import axios from "axios"; import { ethers } from "ethers"; import GoDB from "godb"; -import { permit2Abi } from "../rewards/abis"; +import { permit2Abi } from "./abis"; import { Chain, ChainScan, DATABASE_NAME, NULL_HASH, NULL_ID } from "./constants"; import { getCurrency, @@ -25,9 +25,9 @@ import { GoDBSchema, ObserverKeys, QuickImport, + RewardPermit, SavedData, StandardInterface, - TxData, } from "./types"; import { getTxInfo } from "./utils/getTransaction"; @@ -562,7 +562,7 @@ async function handleRPCData(data: ChainScanResult) { transferDetails: { to }, owner, signature, - } = decodedFunctionData as unknown as TxData; + } = decodedFunctionData as RewardPermit; updateQueue.add(signature, { k: signature, t: "ether", diff --git a/static/scripts/audit-report/types/audit.d.ts b/static/scripts/audit-report/types/audit.d.ts index 86c3b64..005088e 100644 --- a/static/scripts/audit-report/types/audit.d.ts +++ b/static/scripts/audit-report/types/audit.d.ts @@ -128,3 +128,61 @@ interface GoDBTableSchema { export interface GoDBSchema { [table: string]: GoDBTableSchema; } + +export type ERC20Permit = { + type: "erc20-permit"; + permit: { + permitted: { + token: string; + amount: string; + }; + nonce: string; + deadline: string; + }; + transferDetails: { + to: string; + requestedAmount: string; + }; + owner: string; + signature: string; + networkId: number; +}; + +export type ERC721Permit = { + type: "erc721-permit"; + permit: { + permitted: { + token: string; + amount: string; + }; + nonce: string; + deadline: string; + }; + transferDetails: { + to: string; + requestedAmount: string; + }; + owner: string; + signature: string; + networkId: number; + nftMetadata: { + GITHUB_ORGANIZATION_NAME: string; + GITHUB_REPOSITORY_NAME: string; + GITHUB_ISSUE_ID: string; + GITHUB_USERNAME: string; + GITHUB_CONTRIBUTION_TYPE: string; + }; + request: { + beneficiary: string; + deadline: string; + keys: string[]; + nonce: string; + values: string[]; + }; +}; + +export type RewardPermit = ERC20Permit | ERC721Permit; + +export function permitCheck(permit: RewardPermit): permit is Erc20Permit { + return permit.type === "erc20-permit"; +} diff --git a/static/scripts/rewards/cirip/ens-lookup.ts b/static/scripts/rewards/cirip/ens-lookup.ts deleted file mode 100644 index 67a772f..0000000 --- a/static/scripts/rewards/cirip/ens-lookup.ts +++ /dev/null @@ -1,67 +0,0 @@ -import { ethers } from "ethers"; -import abi from "../abis/cirip.json"; -import { fetchEns } from "./fetch-ens"; -import { queryReverseEns } from "./query-reverse-ens"; - -export const UBIQUITY_RPC_ENDPOINT = "https://rpc-pay.ubq.fi/v1/mainnet"; -export const reverseEnsInterface = new ethers.utils.Interface(abi); - -// addEventListener("fetch", event => { -// event.respondWith(handleRequest(event.request).catch(err => new Response(err.stack, { status: 500 }))); -// }); - -export async function ensLookup(addr: string) { - const _address = "/".concat(addr); // quick adapter - - // try { - const start = _address.indexOf("/0x"); - if (start == -1) throw "No ethereum address provided."; - if (_address.length <= 42 + start) { - throw "Invalid ethereum address provided."; - } - const address = _address.substring(start + 1, start + 43).toLowerCase(); - - let reverseRecord = null as null | string; - // let response = ""; - try { - reverseRecord = await queryReverseEns(address); - const responseParsed = JSON.parse(reverseRecord).result; - const _reverseRecord = ethers.utils.defaultAbiCoder.decode([ethers.utils.ParamType.from("string[]")], responseParsed); - reverseRecord = _reverseRecord[0][0]; - } catch (e) { - console.error(e); - // throw "Error contacting ethereum node. \nCause: '" + e + "'. \nResponse: " + response; - } - - const allDomains = await fetchEns(address); - - if (reverseRecord == "") { - reverseRecord = null; - } - - // if reverse record is set, validate addr owns this domain. - if (reverseRecord != null && !allDomains.includes(reverseRecord)) { - console.warn("Failed to validate! Reverse record set to " + reverseRecord + ", but user does not own this name."); - reverseRecord = null; - } - - return { - reverseRecord: reverseRecord, - domains: allDomains, - }; - // new Response(JSON.stringify(response), { - // headers: { - // "Content-Type": "application/json;charset=UTF-8", - // "Access-Control-Allow-Origin": "*", - // }, - // }); - // } catch (e) { - // return new Response("Error: " + e, { - // status: 400, - // headers: { - // "Content-Type": "text/raw;charset=UTF-8", - // "Access-Control-Allow-Origin": "*", - // }, - // }); - // } -} diff --git a/static/scripts/rewards/cirip/fetch-ens.ts b/static/scripts/rewards/cirip/fetch-ens.ts deleted file mode 100644 index ba3ba46..0000000 --- a/static/scripts/rewards/cirip/fetch-ens.ts +++ /dev/null @@ -1,12 +0,0 @@ -import { queryGraph } from "./query-graph"; - -export async function fetchEns(address: string) { - const endpoint = "https://api.thegraph.com/subgraphs/name/ensdomains/ens"; - const query = `{ - domains(where:{owner:"${address.toLowerCase()}"}) { - name - } - }`; - const res = await queryGraph(endpoint, query); - return res.data.domains.map((domain: { name: string }) => domain.name); -} diff --git a/static/scripts/rewards/cirip/query-graph.ts b/static/scripts/rewards/cirip/query-graph.ts deleted file mode 100644 index 8df99d5..0000000 --- a/static/scripts/rewards/cirip/query-graph.ts +++ /dev/null @@ -1,12 +0,0 @@ -export async function queryGraph(endpoint: string | URL | Request, query: string) { - const response = await fetch(endpoint, { - method: "POST", - headers: { - "Content-Type": "application/json", - Accept: "application/json", - }, - body: JSON.stringify({ query }), - }); - - return response.json(); -} diff --git a/static/scripts/rewards/cirip/query-reverse-ens.ts b/static/scripts/rewards/cirip/query-reverse-ens.ts deleted file mode 100644 index 5b14fd7..0000000 --- a/static/scripts/rewards/cirip/query-reverse-ens.ts +++ /dev/null @@ -1,20 +0,0 @@ -import { reverseEnsInterface, UBIQUITY_RPC_ENDPOINT } from "./ens-lookup"; - -export async function queryReverseEns(address: string) { - const data = reverseEnsInterface.encodeFunctionData("getNames", [[address.substring(2)]]); - - const response = await fetch(UBIQUITY_RPC_ENDPOINT, { - method: "POST", - headers: { - "Content-Type": "application/json", - }, - body: JSON.stringify({ - jsonrpc: "2.0", - id: "1", - method: "eth_call", - params: [{ to: "0x3671aE578E63FdF66ad4F3E12CC0c0d71Ac7510C", data: data }, "latest"], - }), - }); - - return response.text(); -} diff --git a/static/scripts/rewards/constants.ts b/static/scripts/rewards/constants.ts deleted file mode 100644 index 891a651..0000000 --- a/static/scripts/rewards/constants.ts +++ /dev/null @@ -1,56 +0,0 @@ -// type RPC = { url: string; tracking?: string; trackingDetails?: string }; -// type Network = { name?: string; rpcs: RPC[]; websiteDead?: boolean; rpcWorking?: boolean }; -// type Networks = { [key: string]: Network }; - -declare const extraRpcs: Record; // @DEV: passed in at build time check build/esbuild-build.ts - -export enum NetworkIds { - Mainnet = 1, - Goerli = 5, - Gnosis = 100, - Anvil = 31337, -} - -export enum Tokens { - DAI = "0x6b175474e89094c44da98b954eedeac495271d0f", - WXDAI = "0xe91d153e0b41518a2ce8dd3d7944fa863463a97d", -} - -export const networkNames = { - [NetworkIds.Mainnet]: "Ethereum Mainnet", - [NetworkIds.Goerli]: "Goerli Testnet", - [NetworkIds.Gnosis]: "Gnosis Chain", - [NetworkIds.Anvil]: "http://127.0.0.1:8545", -}; - -export const networkCurrencies: Record = { - [NetworkIds.Mainnet]: { symbol: "ETH", decimals: 18 }, - [NetworkIds.Goerli]: { symbol: "GoerliETH", decimals: 18 }, - [NetworkIds.Gnosis]: { symbol: "XDAI", decimals: 18 }, - [NetworkIds.Anvil]: { symbol: "XDAI", decimals: 18 }, -}; - -export function getNetworkName(networkId?: number) { - const networkName = networkNames[networkId as keyof typeof networkNames]; - if (!networkName) { - console.error(`Unknown network ID: ${networkId}`); - } - return networkName ?? "Unknown Network"; -} - -export const networkExplorers: Record = { - [NetworkIds.Mainnet]: "https://etherscan.io", - [NetworkIds.Goerli]: "https://goerli.etherscan.io", - [NetworkIds.Gnosis]: "https://gnosisscan.io", - [NetworkIds.Anvil]: "https://gnosisscan.io", -}; - -export const networkRpcs: Record = { - [NetworkIds.Mainnet]: ["https://rpc-pay.ubq.fi/v1/mainnet", ...(extraRpcs[NetworkIds.Mainnet] || [])], - [NetworkIds.Goerli]: ["https://rpc-pay.ubq.fi/v1/goerli", ...(extraRpcs[NetworkIds.Goerli] || [])], - [NetworkIds.Gnosis]: ["https://rpc.ankr.com/gnosis", ...(extraRpcs[NetworkIds.Gnosis] || [])], - [NetworkIds.Anvil]: ["http://127.0.0.1:8545", ""], -}; - -export const permit2Address = "0x000000000022D473030F116dDEE9F6B43aC78BA3"; -export const nftAddress = "0xAa1bfC0e51969415d64d6dE74f27CDa0587e645b"; diff --git a/static/scripts/rewards/helpers.ts b/static/scripts/rewards/helpers.ts deleted file mode 100644 index b9c3896..0000000 --- a/static/scripts/rewards/helpers.ts +++ /dev/null @@ -1,80 +0,0 @@ -import axios from "axios"; -import { Contract, ethers } from "ethers"; -import { erc20Abi } from "./abis"; -import { JsonRpcProvider } from "@ethersproject/providers"; -import { networkRpcs } from "./constants"; - -type DataType = { - jsonrpc: string; - id: number; - result: { - number: string; - timestamp: string; - hash: string; - }; -}; - -function verifyBlock(data: DataType) { - try { - const { jsonrpc, id, result } = data; - const { number, timestamp, hash } = result; - return jsonrpc === "2.0" && id === 1 && parseInt(number, 16) > 0 && parseInt(timestamp, 16) > 0 && hash.match(/[0-9|a-f|A-F|x]/gm)?.join("").length === 66; - } catch (error) { - return false; - } -} - -const RPC_BODY = JSON.stringify({ - jsonrpc: "2.0", - method: "eth_getBlockByNumber", - params: ["latest", false], - id: 1, -}); - -const RPC_HEADER = { - "Content-Type": "application/json", -}; - -export async function getErc20Contract(contractAddress: string, provider: JsonRpcProvider): Promise { - return new ethers.Contract(contractAddress, erc20Abi, provider); -} - -export async function getOptimalProvider(networkId: number) { - if (networkId === 31337) - return new ethers.providers.JsonRpcProvider("http://127.0.0.1:8545", { - name: "http://127.0.0.1:8545", - chainId: 31337, - ensAddress: "", - }); - - const promises = networkRpcs[networkId].map(async (baseURL: string) => { - try { - const startTime = performance.now(); - const API = axios.create({ - baseURL, - headers: RPC_HEADER, - }); - - const { data } = await API.post("", RPC_BODY); - const endTime = performance.now(); - const latency = endTime - startTime; - if (verifyBlock(data)) { - return Promise.resolve({ - latency, - baseURL, - }); - } else { - return Promise.reject(); - } - } catch (error) { - return Promise.reject(); - } - }); - - const { baseURL: optimalRPC } = await Promise.any(promises); - return new ethers.providers.JsonRpcProvider(optimalRPC, { - name: optimalRPC, - chainId: networkId, - ensAddress: "", - }); -} diff --git a/static/scripts/rewards/index.ts b/static/scripts/rewards/index.ts deleted file mode 100644 index 9aa754c..0000000 --- a/static/scripts/rewards/index.ts +++ /dev/null @@ -1,20 +0,0 @@ -import { init } from "./render-transaction/render-transaction"; -import { grid } from "./the-grid"; - -(async function appAsyncWrapper() { - try { - // display commit hash - const commit = await fetch("commit.txt"); - if (commit.ok) { - const commitHash = await commit.text(); - const buildElement = document.querySelector(`#build a`) as HTMLAnchorElement; - buildElement.innerHTML = commitHash; - buildElement.href = `https://github.com/ubiquity/pay.ubq.fi/commit/${commitHash}`; - } - init().catch(console.error); - } catch (error) { - console.error(error); - } -})().catch(console.error); - -grid(document.getElementById("grid") as HTMLElement); diff --git a/static/scripts/rewards/invalidate-component.ts b/static/scripts/rewards/invalidate-component.ts deleted file mode 100644 index 8687d32..0000000 --- a/static/scripts/rewards/invalidate-component.ts +++ /dev/null @@ -1,12 +0,0 @@ -const invalidateBtnInnerHTML = `
Void
-`; - -// parse string and turn into an html entity -function parseHtml(html: string) { - const button = document.createElement("button"); - button.id = "invalidateBtn"; - button.innerHTML = html; - return button.cloneNode(true) as HTMLButtonElement; -} - -export default parseHtml(invalidateBtnInnerHTML); diff --git a/static/scripts/rewards/render-transaction/index.ts b/static/scripts/rewards/render-transaction/index.ts deleted file mode 100644 index ee56eba..0000000 --- a/static/scripts/rewards/render-transaction/index.ts +++ /dev/null @@ -1,42 +0,0 @@ -import { networkExplorers } from "../constants"; -import { getOptimalProvider } from "../helpers"; -import { ClaimTx } from "./tx-type"; - -class AppState { - public claimTxs: ClaimTx[] = []; - private _currentIndex = 0; - - get currentIndex(): number { - return this._currentIndex; - } - - get currentTx(): ClaimTx | null { - return this.currentIndex < this.claimTxs.length ? this.claimTxs[this.currentIndex] : null; - } - - async currentNetworkRpc(): Promise { - if (!this.currentTx) { - return (await getOptimalProvider(1)).connection.url; - } - return (await getOptimalProvider(this.currentTx.networkId)).connection.url; - } - - get currentExplorerUrl(): string { - if (!this.currentTx) { - return "https://etherscan.io"; - } - return networkExplorers[this.currentTx.networkId] || "https://etherscan.io"; - } - - nextTx(): ClaimTx | null { - this._currentIndex = Math.min(this.claimTxs.length - 1, this._currentIndex + 1); - return this.currentTx; - } - - previousTx(): ClaimTx | null { - this._currentIndex = Math.max(0, this._currentIndex - 1); - return this.currentTx; - } -} - -export const app = new AppState(); diff --git a/static/scripts/rewards/render-transaction/insert-table-data.ts b/static/scripts/rewards/render-transaction/insert-table-data.ts deleted file mode 100644 index 0da3719..0000000 --- a/static/scripts/rewards/render-transaction/insert-table-data.ts +++ /dev/null @@ -1,129 +0,0 @@ -import { ethers } from "ethers"; -import { app } from "."; -import { Erc20Permit, Erc721Permit } from "./tx-type"; -import { fetchTreasury } from "../web3/erc20-permit"; -import { renderTokenSymbol } from "./render-token-symbol"; -import { networkExplorers } from "../constants"; - -export function shortenAddress(address: string): string { - return `${address.slice(0, 10)}...${address.slice(-8)}`; -} - -export async function insertErc20PermitTableData( - permit: Erc20Permit, - provider: ethers.providers.JsonRpcProvider, - symbol: string, - decimals: number, - table: Element -) { - const requestedAmountElement = document.getElementById("rewardAmount") as Element; - renderToFields(permit.transferDetails.to, app.currentExplorerUrl); - renderTokenFields(permit.permit.permitted.token, app.currentExplorerUrl); - - renderDetailsFields([ - { name: "From", value: `${permit.owner}` }, - { - name: "Expiry", - value: permit.permit.deadline.lte(Number.MAX_SAFE_INTEGER.toString()) ? new Date(permit.permit.deadline.toNumber()).toLocaleString() : undefined, - }, - { name: "Balance", value: "Loading..." }, - { name: "Allowance", value: "Loading..." }, - ]); - - renderTokenSymbol({ - requestedAmountElement, - tokenAddress: permit.permit.permitted.token, - ownerAddress: permit.owner, - amount: permit.transferDetails.requestedAmount, - explorerUrl: networkExplorers[permit.networkId], - symbol, - decimals, - }); - - // Optimistically rendered what we can so consider it loaded - table.setAttribute(`data-claim`, "ok"); - table.setAttribute(`data-contract-loaded`, "true"); - table.setAttribute(`data-claim-rendered`, "true"); - - const { balance, allowance } = await fetchTreasury(permit.permit.permitted.token, permit.owner, provider); - - renderDetailsFields([ - { name: "From", value: `${permit.owner}` }, - { - name: "Expiry", - value: permit.permit.deadline.lte(Number.MAX_SAFE_INTEGER.toString()) ? new Date(permit.permit.deadline.toNumber()).toLocaleString() : undefined, - }, - { name: "Balance", value: balance.gte(0) ? `${ethers.utils.formatUnits(balance, decimals)} ${symbol}` : "N/A" }, - { name: "Allowance", value: allowance.gte(0) ? `${ethers.utils.formatUnits(allowance, decimals)} ${symbol}` : "N/A" }, - ]); -} - -export function insertErc721PermitTableData(permit: Erc721Permit, table: Element): Element { - const requestedAmountElement = document.getElementById("rewardAmount") as Element; - renderToFields(permit.request.beneficiary, app.currentExplorerUrl); - renderTokenFields(permit.nftAddress, app.currentExplorerUrl); - const { GITHUB_REPOSITORY_NAME, GITHUB_CONTRIBUTION_TYPE, GITHUB_ISSUE_ID, GITHUB_ORGANIZATION_NAME, GITHUB_USERNAME } = permit.nftMetadata; - renderDetailsFields([ - { - name: "NFT address", - value: `${permit.nftAddress}`, - }, - { - name: "Expiry", - value: permit.request.deadline.lte(Number.MAX_SAFE_INTEGER.toString()) ? new Date(permit.request.deadline.toNumber()).toLocaleString() : undefined, - }, - { - name: "GitHub Organization", - value: `${GITHUB_ORGANIZATION_NAME}`, - }, - { - name: "GitHub Repository", - value: `${GITHUB_REPOSITORY_NAME}`, - }, - { - name: "GitHub Issue", - value: `${GITHUB_ISSUE_ID}`, - }, - { - name: "GitHub Username", - value: `${GITHUB_USERNAME}`, - }, - { name: "Contribution Type", value: GITHUB_CONTRIBUTION_TYPE.split(",").join(", ") }, - ]); - table.setAttribute(`data-claim-rendered`, "true"); - return requestedAmountElement; -} - -function renderDetailsFields(additionalDetails: { name: string; value: string | undefined }[]) { - const additionalDetailsDiv = document.getElementById("additionalDetailsTable") as Element; - let additionalDetailsHtml = ""; - for (const { name, value } of additionalDetails) { - if (!value) continue; - additionalDetailsHtml += ` -
${name}
-
${value}
- `; - } - - additionalDetailsDiv.innerHTML = additionalDetailsHtml; -} - -function renderTokenFields(tokenAddress: string, explorerUrl: string) { - const tokenFull = document.querySelector("#Token .full") as Element; - const tokenShort = document.querySelector("#Token .short") as Element; - tokenFull.innerHTML = `
${tokenAddress}
`; - tokenShort.innerHTML = `
${shortenAddress(tokenAddress)}
`; - - const tokenBoth = document.getElementById(`rewardToken`) as Element; - tokenBoth.innerHTML = `${tokenBoth.innerHTML}`; -} - -function renderToFields(receiverAddress: string, explorerUrl: string) { - const toFull = document.querySelector("#To .full") as Element; - const toShort = document.querySelector("#To .short") as Element; - toFull.innerHTML = `
${receiverAddress}
`; - toShort.innerHTML = `
${shortenAddress(receiverAddress)}
`; - - const toBoth = document.getElementById(`rewardRecipient`) as Element; - toBoth.innerHTML = `${toBoth.innerHTML}`; -} diff --git a/static/scripts/rewards/render-transaction/render-ens-name.ts b/static/scripts/rewards/render-transaction/render-ens-name.ts deleted file mode 100644 index 8be3262..0000000 --- a/static/scripts/rewards/render-transaction/render-ens-name.ts +++ /dev/null @@ -1,42 +0,0 @@ -import { ensLookup } from "../cirip/ens-lookup"; -import { app } from "./index"; - -type EnsParams = - | { - element: Element; - address: string; - tokenAddress: string; - tokenView: true; - } - | { - element: Element; - address: string; - tokenAddress?: undefined; - tokenView?: false; - }; - -export async function renderEnsName({ element, address, tokenAddress, tokenView }: EnsParams): Promise { - let href: string = ""; - try { - const resolved = await ensLookup(address); - let ensName: undefined | string; - if (resolved.reverseRecord) { - ensName = resolved.reverseRecord; - } else if (resolved.domains.length) { - const domain = resolved.domains.shift(); - if (domain) { - ensName = domain; - } - } - if (ensName) { - if (tokenView) { - href = `${app.currentExplorerUrl}/token/${tokenAddress}?a=${address}`; - } else { - href = `${app.currentExplorerUrl}/address/${address}"`; - } - element.innerHTML = `${ensName}`; - } - } catch (error) { - console.error(error); - } -} diff --git a/static/scripts/rewards/render-transaction/render-token-symbol.ts b/static/scripts/rewards/render-transaction/render-token-symbol.ts deleted file mode 100644 index 429bda7..0000000 --- a/static/scripts/rewards/render-transaction/render-token-symbol.ts +++ /dev/null @@ -1,56 +0,0 @@ -import { BigNumberish, utils } from "ethers"; -import { getErc20Contract } from "../helpers"; -import { JsonRpcProvider } from "@ethersproject/providers"; - -export const tokens = [ - { - name: "WXDAI", - address: "0xe91d153e0b41518a2ce8dd3d7944fa863463a97d", - }, - { - name: "DAI", - address: "0x6b175474e89094c44da98b954eedeac495271d0f", - }, -]; - -export function renderTokenSymbol({ - requestedAmountElement, - tokenAddress, - ownerAddress, - amount, - explorerUrl, - symbol, - decimals, -}: { - requestedAmountElement: Element; - tokenAddress: string; - ownerAddress: string; - amount: BigNumberish; - explorerUrl: string; - symbol: string; - decimals: number; -}) { - return (requestedAmountElement.innerHTML = `${utils.formatUnits( - amount, - decimals - )} ${symbol}`); -} - -export async function renderNftSymbol({ - table, - requestedAmountElement, - tokenAddress, - explorerUrl, - provider, -}: { - table: Element; - requestedAmountElement: Element; - tokenAddress: string; - explorerUrl: string; - provider: JsonRpcProvider; -}): Promise { - const contract = await getErc20Contract(tokenAddress, provider); - const symbol = await contract.symbol(); - table.setAttribute(`data-contract-loaded`, "true"); - requestedAmountElement.innerHTML = `1 ${symbol}`; -} diff --git a/static/scripts/rewards/render-transaction/render-transaction.ts b/static/scripts/rewards/render-transaction/render-transaction.ts deleted file mode 100644 index 4f3d015..0000000 --- a/static/scripts/rewards/render-transaction/render-transaction.ts +++ /dev/null @@ -1,155 +0,0 @@ -import { JsonRpcProvider } from "@ethersproject/providers"; -import { Type } from "@sinclair/typebox"; -import { Value } from "@sinclair/typebox/value"; -import { networkExplorers } from "../constants"; -import { getOptimalProvider } from "../helpers"; -import { claimButton, hideClaimButton, resetClaimButton } from "../toaster"; -import { claimErc20PermitHandler, generateInvalidatePermitAdminControl, processERC20 } from "../web3/erc20-permit"; -import { claimErc721PermitHandler } from "../web3/erc721-permit"; -import { app } from "./index"; -import { insertErc721PermitTableData } from "./insert-table-data"; -import { renderEnsName } from "./render-ens-name"; -import { renderNftSymbol } from "./render-token-symbol"; -import { setClaimMessage } from "./set-claim-message"; -import { claimTxT } from "./tx-type"; -import { removeAllEventListeners } from "./utils"; -import { handleNetwork } from "../web3/wallet"; - -let optimalRPC: JsonRpcProvider; - -export async function init() { - const table = document.getElementsByTagName(`table`)[0]; - - // decode base64 to get tx data - const urlParams = new URLSearchParams(window.location.search); - const base64encodedTxData = urlParams.get("claim"); - - if (!base64encodedTxData) { - setClaimMessage({ type: "Notice", message: `No claim data found.` }); - table.setAttribute(`data-claim`, "none"); - return false; - } - - try { - const claimTxs = Value.Decode(Type.Array(claimTxT), JSON.parse(atob(base64encodedTxData))); - app.claimTxs = claimTxs; - optimalRPC = await getOptimalProvider(app.currentTx?.networkId ?? app.claimTxs[0].networkId); - - handleNetwork(app.currentTx?.networkId ?? app.claimTxs[0].networkId).catch(console.error); - } catch (error) { - console.error(error); - setClaimMessage({ type: "Error", message: `Invalid claim data passed in URL` }); - table.setAttribute(`data-claim`, "error"); - return false; - } - - let isDetailsVisible = false; - - table.setAttribute(`data-details-visible`, isDetailsVisible.toString()); - - const additionalDetails = document.getElementById(`additionalDetails`) as Element; - additionalDetails.addEventListener("click", () => { - isDetailsVisible = !isDetailsVisible; - table.setAttribute(`data-details-visible`, isDetailsVisible.toString()); - }); - - const rewardsCount = document.getElementById("rewardsCount"); - if (rewardsCount) { - if (!app.claimTxs || app.claimTxs.length <= 1) { - // already hidden - } else { - rewardsCount.innerHTML = `${app.currentIndex + 1}/${app.claimTxs.length} reward`; - - const nextTxButton = document.getElementById("nextTx"); - if (nextTxButton) { - nextTxButton.addEventListener("click", () => { - claimButton.element = removeAllEventListeners(claimButton.element) as HTMLButtonElement; - app.nextTx(); - rewardsCount.innerHTML = `${app.currentIndex + 1}/${app.claimTxs.length} reward`; - table.setAttribute(`data-claim`, "none"); - renderTransaction(optimalRPC, true).catch(console.error); - }); - } - - const prevTxButton = document.getElementById("previousTx"); - if (prevTxButton) { - prevTxButton.addEventListener("click", () => { - claimButton.element = removeAllEventListeners(claimButton.element) as HTMLButtonElement; - app.previousTx(); - rewardsCount.innerHTML = `${app.currentIndex + 1}/${app.claimTxs.length} reward`; - table.setAttribute(`data-claim`, "none"); - renderTransaction(optimalRPC, true).catch(console.error); - }); - } - - setPagination(nextTxButton, prevTxButton); - } - } - - renderTransaction(optimalRPC).catch(console.error); -} - -function setPagination(nextTxButton: Element | null, prevTxButton: Element | null) { - if (!nextTxButton || !prevTxButton) return; - if (app.claimTxs.length > 1) { - prevTxButton.classList.remove("hide-pagination"); - nextTxButton.classList.remove("hide-pagination"); - - prevTxButton.classList.add("show-pagination"); - nextTxButton.classList.add("show-pagination"); - } -} - -type Success = boolean; -export async function renderTransaction(provider: JsonRpcProvider, nextTx?: boolean): Promise { - const table = document.getElementsByTagName(`table`)[0]; - resetClaimButton(); - - if (nextTx) { - app.nextTx(); - if (!app.claimTxs || app.claimTxs.length <= 1) { - // already hidden - } else { - setPagination(document.getElementById("nextTx"), document.getElementById("previousTx")); - - const rewardsCount = document.getElementById("rewardsCount") as Element; - rewardsCount.innerHTML = `${app.currentIndex + 1}/${app.claimTxs.length} reward`; - table.setAttribute(`data-claim`, "none"); - } - } - - if (!app.currentTx) { - hideClaimButton(); - return false; - } - - if (app.currentTx.type === "erc20-permit") { - await processERC20(app.currentTx.permit.permitted.token, provider, app.currentTx, table); - - // insert tx data into table - const toElement = document.getElementById(`rewardRecipient`) as Element; - renderEnsName({ element: toElement, address: app.currentTx.transferDetails.to }).catch(console.error); - - generateInvalidatePermitAdminControl(app.currentTx).catch(console.error); - - claimButton.element.addEventListener("click", claimErc20PermitHandler(app.currentTx, optimalRPC)); - } else if (app.currentTx.type === "erc721-permit") { - const requestedAmountElement = insertErc721PermitTableData(app.currentTx, table); - table.setAttribute(`data-claim`, "ok"); - - renderNftSymbol({ - tokenAddress: app.currentTx.nftAddress, - explorerUrl: networkExplorers[app.currentTx.networkId], - table, - requestedAmountElement, - provider, - }).catch(console.error); - - const toElement = document.getElementById(`rewardRecipient`) as Element; - renderEnsName({ element: toElement, address: app.currentTx.request.beneficiary }).catch(console.error); - - claimButton.element.addEventListener("click", claimErc721PermitHandler(app.currentTx, provider)); - } - - return true; -} diff --git a/static/scripts/rewards/render-transaction/set-claim-message.ts b/static/scripts/rewards/render-transaction/set-claim-message.ts deleted file mode 100644 index 51d38d1..0000000 --- a/static/scripts/rewards/render-transaction/set-claim-message.ts +++ /dev/null @@ -1,6 +0,0 @@ -export function setClaimMessage({ type, message }: { type: string; message: string }): void { - const claimMessageType = document.querySelector(`table > thead th`) as Element; - const claimMessageBody = document.querySelector(`table > thead td`) as Element; - claimMessageType.innerHTML = `
${type}
`; - claimMessageBody.innerHTML = `
${message}
`; -} diff --git a/static/scripts/rewards/render-transaction/tx-type.ts b/static/scripts/rewards/render-transaction/tx-type.ts deleted file mode 100644 index 23df134..0000000 --- a/static/scripts/rewards/render-transaction/tx-type.ts +++ /dev/null @@ -1,68 +0,0 @@ -import { StaticDecode, Type as T } from "@sinclair/typebox"; -import { BigNumber } from "ethers"; - -const bigNumberT = T.Transform(T.Union([T.RegExp(/^\d+$/), T.Number()])) - .Decode((value) => BigNumber.from(value)) - .Encode((value) => value.toString()); - -// const networkIdT = T.Transform(T.Union([T.RegExp(/^0x\d+$/), T.Number()])) -// .Decode(value => (typeof value === "number" ? "0x" + value.toString(16) : value)) -// .Encode(value => value); - -const networkIdT = T.Number(); - -const addressT = T.Transform(T.RegExp(/^0x[a-fA-F0-9]{40}$/)) - .Decode((value) => value.toLowerCase()) - .Encode((value) => value); - -const signatureT = T.Transform(T.RegExp(/^0x[a-fA-F0-9]+$/)) - .Decode((value) => value.toLowerCase()) - .Encode((value) => value); - -const erc20PermitT = T.Object({ - type: T.Literal("erc20-permit"), - permit: T.Object({ - permitted: T.Object({ - token: addressT, - amount: bigNumberT, - }), - nonce: bigNumberT, - deadline: bigNumberT, - }), - transferDetails: T.Object({ - to: addressT, - requestedAmount: bigNumberT, - }), - owner: addressT, - signature: signatureT, - networkId: networkIdT, -}); - -export type Erc20Permit = StaticDecode; - -const erc721Permit = T.Object({ - type: T.Literal("erc721-permit"), - request: T.Object({ - beneficiary: addressT, - deadline: bigNumberT, - keys: T.Array(T.String()), - nonce: bigNumberT, - values: T.Array(T.String()), - }), - nftMetadata: T.Object({ - GITHUB_ORGANIZATION_NAME: T.String(), - GITHUB_REPOSITORY_NAME: T.String(), - GITHUB_ISSUE_ID: T.String(), - GITHUB_USERNAME: T.String(), - GITHUB_CONTRIBUTION_TYPE: T.String(), - }), - nftAddress: addressT, - networkId: networkIdT, - signature: signatureT, -}); - -export type Erc721Permit = StaticDecode; - -export const claimTxT = T.Union([erc20PermitT, erc721Permit]); - -export type ClaimTx = StaticDecode; diff --git a/static/scripts/rewards/render-transaction/utils.ts b/static/scripts/rewards/render-transaction/utils.ts deleted file mode 100644 index 80281a6..0000000 --- a/static/scripts/rewards/render-transaction/utils.ts +++ /dev/null @@ -1,5 +0,0 @@ -export function removeAllEventListeners(element: Element): Element { - const clone = element.cloneNode(true) as Element; - element.replaceWith(clone); - return clone; -} diff --git a/static/scripts/rewards/the-grid.ts b/static/scripts/rewards/the-grid.ts deleted file mode 100644 index bd9832e..0000000 --- a/static/scripts/rewards/the-grid.ts +++ /dev/null @@ -1,160 +0,0 @@ -export function grid(node = document.body) { - // Create canvas and WebGL context - const canvas = document.createElement("canvas"); - const devicePixelRatio = window.devicePixelRatio || 1; - canvas.width = window.innerWidth * devicePixelRatio; - canvas.height = window.innerHeight * devicePixelRatio; - node.appendChild(canvas); - - const gl = canvas.getContext("webgl") as WebGLRenderingContext; - - // Enable alpha blending - gl.enable(gl.BLEND); - gl.blendFunc(gl.SRC_ALPHA, gl.ONE_MINUS_SRC_ALPHA); - - // Define shader sources - const vertexShaderSource = ` - attribute vec2 a_position; - - void main() { - gl_Position = vec4(a_position, 0, 1); - } -`; - - // cspell:ignore mediump - const fragmentShaderSource = ` - precision mediump float; - - uniform vec2 u_resolution; - uniform float u_time; - - float rand(vec2 n) { - return fract(sin(dot(n, vec2(12.9898, 4.1414))) * 43758.5453); - } - - void main() { - vec3 color = vec3(128.0/255.0, 128.0/255.0, 128.0/255.0); // #808080 - vec2 tilePosition = mod(gl_FragCoord.xy, 24.0); - vec2 tileNumber = floor(gl_FragCoord.xy / 24.0); - - float period = rand(tileNumber) * 9.0 + 1.0; // Random value in the range [1, 10] - float phase = fract(u_time / period / 8.0); // Animation eight times slower - float opacity = (1.0 - abs(phase * 2.0 - 1.0)) * 0.125; // Limit maximum opacity to 0.25 - - vec4 backgroundColor = vec4(color, opacity); - - if (tilePosition.x > 23.0 && tilePosition.y < 1.0) { - gl_FragColor = vec4(color, 1.0); // Full opacity for the dot - } else { - gl_FragColor = backgroundColor; - } - } -`; - - // Define shader creation function - function createShader(gl: WebGLRenderingContext, type: number, source: string) { - const shader = gl.createShader(type); - if (!shader) { - console.error("An error occurred creating the shaders"); - return null; - } - gl.shaderSource(shader, source); - gl.compileShader(shader); - if (!gl.getShaderParameter(shader, gl.COMPILE_STATUS)) { - console.error("An error occurred compiling the shaders: " + gl.getShaderInfoLog(shader)); - gl.deleteShader(shader); - return null; - } - return shader; - } - - // Create vertex and fragment shaders - const vertexShader = createShader(gl, gl.VERTEX_SHADER, vertexShaderSource); - if (!vertexShader) { - console.error("An error occurred creating the vertex shader"); - return; - } - const fragmentShader = createShader(gl, gl.FRAGMENT_SHADER, fragmentShaderSource); - if (!fragmentShader) { - console.error("An error occurred creating the fragment shader"); - return; - } - - // Create program, attach shaders, and link - const program = gl.createProgram(); - if (!program) { - console.error("An error occurred creating the program"); - return; - } - - gl.attachShader(program, vertexShader); - gl.attachShader(program, fragmentShader); - gl.linkProgram(program); - - // Verify program link status - if (!gl.getProgramParameter(program, gl.LINK_STATUS)) { - console.error("Unable to initialize the shader program: " + gl.getProgramInfoLog(program)); - return; - } - - // Use the program - gl.useProgram(program); - - // Get location of time and resolution uniforms - const timeUniformLocation = gl.getUniformLocation(program, "u_time"); - const resolutionUniformLocation = gl.getUniformLocation(program, "u_resolution"); - - // Bind the position buffer and set attribute pointer - const positionBuffer = gl.createBuffer(); - gl.bindBuffer(gl.ARRAY_BUFFER, positionBuffer); - gl.bufferData(gl.ARRAY_BUFFER, new Float32Array([-1, -1, 1, -1, -1, 1, 1, 1]), gl.STATIC_DRAW); - - const positionAttributeLocation = gl.getAttribLocation(program, "a_position"); - gl.enableVertexAttribArray(positionAttributeLocation); - gl.vertexAttribPointer(positionAttributeLocation, 2, gl.FLOAT, false, 0, 0); - - // Resize function - function resizeCanvasToDisplaySize(canvas: HTMLCanvasElement) { - // Lookup the size the browser is displaying the canvas. - const displayWidth = window.innerWidth; - const displayHeight = window.innerHeight; - - // Check if the canvas is not the same size. - if (canvas.width != displayWidth || canvas.height != displayHeight) { - // Make the canvas the same size - canvas.width = displayWidth; - canvas.height = displayHeight; - - // Update WebGL viewport to match - gl.viewport(0, 0, canvas.width, canvas.height); - } - } - - // Render function - function render() { - resizeCanvasToDisplaySize(canvas); // Check and update canvas size each frame - - // Update resolution uniform - gl.uniform2f(resolutionUniformLocation, canvas.width, canvas.height); - - gl.clearColor(0.0, 0.0, 0.0, 1.0); - gl.clear(gl.COLOR_BUFFER_BIT); - - // Update time uniform - gl.uniform1f(timeUniformLocation, performance.now() / 1000.0); - - // Draw - gl.drawArrays(gl.TRIANGLE_STRIP, 0, 4); - - // Request next frame - requestAnimationFrame(render); - } - - // Handle window resize - window.addEventListener("resize", () => { - resizeCanvasToDisplaySize(canvas); - }); - - // Start the render loop - render(); -} diff --git a/static/scripts/rewards/toaster.ts b/static/scripts/rewards/toaster.ts deleted file mode 100644 index 9294e72..0000000 --- a/static/scripts/rewards/toaster.ts +++ /dev/null @@ -1,89 +0,0 @@ -export const toaster = { - create: createToast, - error: errorToast, - icons: { - success: "fa-circle-check", - error: "fa-circle-xmark", - warning: "fa-triangle-exclamation", - info: "fa-circle-info", - }, -}; - -export const claimButton = { - loading: loadingClaimButton, - reset: resetClaimButton, - element: document.getElementById("claimButton") as HTMLButtonElement, -}; - -const notifications = document.querySelector(".notifications") as HTMLUListElement; - -export function createToast(meaning: keyof typeof toaster.icons, text: string) { - const toastDetails = { - timer: 5000, - } as { - timer: number; - timeoutId?: NodeJS.Timeout; - }; - // Getting the icon and text for the toast based on the id passed - const _icon = toaster.icons[meaning]; - const toastContent = document.createElement("li"); // Creating a new 'li' element for the toast - toastContent.className = `toast .${_icon} ${meaning}`; // Setting the classes for the toast - - // Setting the inner HTML for the toast - toastContent.innerHTML = `
${text}
`; - - // attaching a click event listener to the toast to remove it when the close icon is clicked - const i = document.createElement("i"); - i.className = "fa-solid fa-xmark"; - i.onclick = () => removeToast(toastContent, toastDetails.timeoutId); - toastContent.appendChild(i); - - notifications.appendChild(toastContent); // Append the toast to the notification ul - - // Setting a timeout to remove the toast after the specified duration - toastDetails.timeoutId = setTimeout(() => removeToast(toastContent, toastDetails.timeoutId), toastDetails.timer); -} - -function removeToast(toast: HTMLElement, timeoutId?: NodeJS.Timeout) { - toast.classList.add("hide"); - if (timeoutId) { - clearTimeout(timeoutId); // Clearing the timeout for the toast - } - setTimeout(() => toast.remove(), 500); // Removing the toast after 500ms -} - -export function loadingClaimButton(triggerLoader = true) { - claimButton.element.disabled = true; - // Adding this because not all disabling should trigger loading spinner - if (triggerLoader) { - claimButton.element.classList.add("show-cl"); - claimButton.element.classList.remove("hide-cl"); - } -} - -export function resetClaimButton() { - claimButton.element.disabled = false; - claimButton.element.classList.add("hide-cl"); - claimButton.element.classList.remove("show-cl"); -} - -export function hideClaimButton() { - claimButton.element.disabled = true; - claimButton.element.classList.add("hide-cl"); - claimButton.element.classList.remove("show-cl"); -} - -type Err = { stack?: unknown; reason?: string } extends Error ? Error : { stack?: unknown; reason?: string }; - -export function errorToast(error: Err, errorMessage?: string) { - delete error.stack; - const errorData = JSON.stringify(error, null, 2); - if (errorMessage) { - toaster.create("error", errorMessage); - } else if (error?.reason) { - // parse error data to get error message - const parsedError = JSON.parse(errorData); - const _errorMessage = parsedError?.error?.message ?? parsedError?.reason; - toaster.create("error", _errorMessage); - } -} diff --git a/static/scripts/rewards/web3/erc20-permit.ts b/static/scripts/rewards/web3/erc20-permit.ts deleted file mode 100644 index fd26d9f..0000000 --- a/static/scripts/rewards/web3/erc20-permit.ts +++ /dev/null @@ -1,183 +0,0 @@ -import { BigNumber, BigNumberish, ethers } from "ethers"; -import { permit2Abi } from "../abis"; -import { permit2Address } from "../constants"; -import { getErc20Contract, getOptimalProvider } from "../helpers"; -import { Erc20Permit } from "../render-transaction/tx-type"; -import { toaster, resetClaimButton, errorToast, loadingClaimButton, claimButton } from "../toaster"; -import { renderTransaction } from "../render-transaction/render-transaction"; -import { connectWallet } from "./wallet"; -import invalidateButton from "../invalidate-component"; -import { JsonRpcProvider } from "@ethersproject/providers"; -import { tokens } from "../render-transaction/render-token-symbol"; -import { insertErc20PermitTableData } from "../render-transaction/insert-table-data"; - -export async function processERC20(tokenAddress: string, provider: JsonRpcProvider, permit: Erc20Permit, table: Element) { - let symbol = tokenAddress === tokens[0].address ? tokens[0].name : tokenAddress === tokens[1].address ? tokens[1].name : ""; - let decimals = tokenAddress === tokens[0].address ? 18 : tokenAddress === tokens[1].address ? 18 : -1; - - if (!symbol || decimals === -1) { - try { - const contract = await getErc20Contract(tokenAddress, provider); - symbol = contract.symbol(); - decimals = contract.decimals(); - } catch (err) { - throw new Error(`Error fetching symbol and decimals for token address: ${tokenAddress}`); - } - } - - await insertErc20PermitTableData(permit, provider, symbol, decimals, table); -} - -export async function fetchTreasury(contractAddr: string, owner: string, provider: JsonRpcProvider) { - try { - const contract = await getErc20Contract(contractAddr, provider); - const [balance, allowance] = await Promise.all([contract.balanceOf(owner), contract.allowance(owner, permit2Address)]); - return { balance, allowance } as { balance: BigNumber; allowance: BigNumber }; - } catch (err) { - console.log(err); - } - - return { balance: BigNumber.from(0), allowance: BigNumber.from(0) }; -} - -export function claimErc20PermitHandler(permit: Erc20Permit, provider: JsonRpcProvider) { - return async function handler() { - try { - const signer = await connectWallet(); - if (!signer) { - return; - } - - if (!(await checkPermitClaimable(permit, signer, provider))) { - return; - } - - loadingClaimButton(); - const permit2Contract = new ethers.Contract(permit2Address, permit2Abi, signer); - const tx = await permit2Contract.permitTransferFrom(permit.permit, permit.transferDetails, permit.owner, permit.signature); - toaster.create("info", `Transaction sent`); - const receipt = await tx.wait(); - toaster.create("success", `Claim Complete.`); - console.log(receipt.transactionHash); // @TODO: post to database - - claimButton.element.removeEventListener("click", handler); - renderTransaction(provider).catch(console.error); - } catch (error: unknown) { - if (error instanceof Error) { - console.log(error); - errorToast(error, error.message); - resetClaimButton(); - } - } - }; -} - -export async function checkPermitClaimable(permit: Erc20Permit, signer: ethers.providers.JsonRpcSigner | null, provider: JsonRpcProvider) { - const isClaimed = await isNonceClaimed(permit); - if (isClaimed) { - toaster.create("error", `Your reward for this task has already been claimed or invalidated.`); - return false; - } - - if (permit.permit.deadline.lt(Math.floor(Date.now() / 1000))) { - toaster.create("error", `This reward has expired.`); - return false; - } - - const { balance, allowance } = await fetchTreasury(permit.permit.permitted.token, permit.owner, provider); - const permitted = BigNumber.from(permit.permit.permitted.amount); - const isSolvent = balance.gte(permitted); - const isAllowed = allowance.gte(permitted); - - if (!isSolvent) { - toaster.create("error", `Not enough funds on funding wallet to collect this reward. Please let the financier know.`); - return false; - } - if (!isAllowed) { - toaster.create("error", `Not enough allowance on the funding wallet to collect this reward. Please let the financier know.`); - return false; - } - - if (signer) { - const user = (await signer.getAddress()).toLowerCase(); - const beneficiary = permit.transferDetails.to.toLowerCase(); - if (beneficiary !== user) { - toaster.create("warning", `This reward is not for you.`); - return false; - } - } - - return true; -} - -export async function generateInvalidatePermitAdminControl(permit: Erc20Permit) { - const signer = await connectWallet(); - if (!signer) { - return; - } - - const user = (await signer.getAddress()).toLowerCase(); - const owner = permit.owner.toLowerCase(); - if (owner !== user) { - return; - } - - const controls = document.getElementById("controls") as HTMLDivElement; - controls.appendChild(invalidateButton); - - invalidateButton.addEventListener("click", async function invalidateButtonClickHandler() { - try { - const signer = await connectWallet(); - if (!signer) { - return; - } - const isClaimed = await isNonceClaimed(permit); - if (isClaimed) { - toaster.create("error", `This reward has already been claimed or invalidated.`); - return; - } - await invalidateNonce(signer, permit.permit.nonce); - } catch (error: unknown) { - if (error instanceof Error) { - console.log(error); - errorToast(error, error.message); - return; - } - } - toaster.create("info", "Nonce invalidation transaction sent"); - }); -} - -//mimics https://github.com/Uniswap/permit2/blob/a7cd186948b44f9096a35035226d7d70b9e24eaf/src/SignatureTransfer.sol#L150 -export async function isNonceClaimed(permit: Erc20Permit): Promise { - const provider = await getOptimalProvider(permit.networkId); - - const permit2Contract = new ethers.Contract(permit2Address, permit2Abi, provider); - - const { wordPos, bitPos } = nonceBitmap(BigNumber.from(permit.permit.nonce)); - const bitmap = await permit2Contract.nonceBitmap(permit.owner, wordPos); - - const bit = BigNumber.from(1).shl(bitPos); - const flipped = BigNumber.from(bitmap).xor(bit); - - return bit.and(flipped).eq(0); -} - -export async function invalidateNonce(signer: ethers.providers.JsonRpcSigner, nonce: BigNumberish): Promise { - const permit2Contract = new ethers.Contract(permit2Address, permit2Abi, signer); - const { wordPos, bitPos } = nonceBitmap(nonce); - // mimics https://github.com/ubiquity/pay.ubq.fi/blob/c9e7ed90718fe977fd9f348db27adf31d91d07fb/scripts/solidity/test/Permit2.t.sol#L428 - const bit = BigNumber.from(1).shl(bitPos); - const sourceBitmap = await permit2Contract.nonceBitmap(await signer.getAddress(), wordPos.toString()); - const mask = sourceBitmap.or(bit); - await permit2Contract.invalidateUnorderedNonces(wordPos, mask); -} - -// mimics https://github.com/Uniswap/permit2/blob/db96e06278b78123970183d28f502217bef156f4/src/SignatureTransfer.sol#L142 -export function nonceBitmap(nonce: BigNumberish): { wordPos: BigNumber; bitPos: number } { - // wordPos is the first 248 bits of the nonce - const wordPos = BigNumber.from(nonce).shr(8); - // bitPos is the last 8 bits of the nonce - const bitPos = BigNumber.from(nonce).and(255).toNumber(); - return { wordPos, bitPos }; -} diff --git a/static/scripts/rewards/web3/erc721-permit.ts b/static/scripts/rewards/web3/erc721-permit.ts deleted file mode 100644 index 0dafff6..0000000 --- a/static/scripts/rewards/web3/erc721-permit.ts +++ /dev/null @@ -1,64 +0,0 @@ -import { JsonRpcProvider, TransactionResponse } from "@ethersproject/providers"; -import { ethers } from "ethers"; -import { nftRewardAbi } from "../abis/nftRewardAbi"; -import { renderTransaction } from "../render-transaction/render-transaction"; -import { Erc721Permit } from "../render-transaction/tx-type"; -import { claimButton, errorToast, loadingClaimButton, resetClaimButton, toaster } from "../toaster"; -import { connectWallet } from "./wallet"; - -export function claimErc721PermitHandler(permit: Erc721Permit, provider: JsonRpcProvider) { - return async function claimButtonHandler() { - const signer = await connectWallet(); - if (!signer) { - return; - } - - if ((await signer.getAddress()).toLowerCase() !== permit.request.beneficiary) { - toaster.create("warning", `This NFT is not for you.`); - resetClaimButton(); - return; - } - - if (permit.request.deadline.lt(Math.floor(Date.now() / 1000))) { - toaster.create("error", `This NFT has expired.`); - resetClaimButton(); - return; - } - - const isRedeemed = await isNonceRedeemed(permit, provider); - if (isRedeemed) { - toaster.create("error", `This NFT has already been redeemed.`); - resetClaimButton(); - return; - } - - loadingClaimButton(); - try { - const nftContract = new ethers.Contract(permit.nftAddress, nftRewardAbi, signer); - - const tx: TransactionResponse = await nftContract.safeMint(permit.request, permit.signature); - toaster.create("info", `Transaction sent. Waiting for confirmation...`); - const receipt = await tx.wait(); - toaster.create("success", `Claim Complete.`); - console.log(receipt.transactionHash); // @TODO: post to database - - claimButton.element.removeEventListener("click", claimButtonHandler); - - renderTransaction(provider, true).catch((error) => { - console.error(error); - toaster.create("error", `Error rendering transaction: ${error.message}`); - }); - } catch (error: unknown) { - if (error instanceof Error) { - console.error(error); - errorToast(error, error.message ?? error); - resetClaimButton(); - } - } - }; -} - -export async function isNonceRedeemed(nftMint: Erc721Permit, provider: JsonRpcProvider): Promise { - const nftContract = new ethers.Contract(nftMint.nftAddress, nftRewardAbi, provider); - return nftContract.nonceRedeemed(nftMint.request.nonce); -} diff --git a/static/scripts/rewards/web3/wallet.ts b/static/scripts/rewards/web3/wallet.ts deleted file mode 100644 index 34766b8..0000000 --- a/static/scripts/rewards/web3/wallet.ts +++ /dev/null @@ -1,103 +0,0 @@ -import { JsonRpcSigner } from "@ethersproject/providers"; -import { ethers } from "ethers"; -import { getNetworkName, networkCurrencies, networkExplorers, networkRpcs } from "../constants"; -import invalidateButton from "../invalidate-component"; -import { claimButton, loadingClaimButton, resetClaimButton, toaster } from "../toaster"; - -export async function connectWallet(): Promise { - try { - const provider = new ethers.providers.Web3Provider(window.ethereum, "any"); - await provider.send("eth_requestAccounts", []); - const signer = provider.getSigner(); - resetClaimButton(); - return signer; - } catch (error: unknown) { - if (error instanceof Error) { - if (error?.message?.includes("missing provider")) { - toaster.create("info", "Please use a web3 enabled browser to collect this reward."); - claimButton.element.disabled = true; - } else { - toaster.create("info", "Please connect your wallet to collect this reward."); - claimButton.element.disabled = true; - } - } - return null; - } -} - -export async function handleNetwork(desiredNetworkId: number) { - const web3provider = new ethers.providers.Web3Provider(window.ethereum); - if (!web3provider || !web3provider.provider.isMetaMask) { - toaster.create("info", "Please connect to MetaMask."); - loadingClaimButton(false); - invalidateButton.disabled = true; - } - - const currentNetworkId = (await web3provider.getNetwork()).chainId; - - // watch for network changes - window.ethereum.on("chainChanged", (newNetworkId: T | string) => handleIfOnCorrectNetwork(parseInt(newNetworkId as string, 16), desiredNetworkId)); - - // if its not on ethereum mainnet, gnosis, or goerli, display error - notOnCorrectNetwork(currentNetworkId, desiredNetworkId, web3provider); -} - -function notOnCorrectNetwork(currentNetworkId: number, desiredNetworkId: number, web3provider: ethers.providers.Web3Provider) { - if (currentNetworkId !== desiredNetworkId) { - if (desiredNetworkId == void 0) { - console.error(`You must pass in an EVM network ID in the URL query parameters using the key 'network' e.g. '?network=1'`); - } - const networkName = getNetworkName(desiredNetworkId); - if (!networkName) { - toaster.create("error", `This dApp currently does not support payouts for network ID ${desiredNetworkId}`); - } - loadingClaimButton(false); - invalidateButton.disabled = true; - switchNetwork(web3provider, desiredNetworkId).catch((error) => { - console.error(error); - toaster.create("error", `Please switch to the ${networkName} network to claim this reward.`); - }); - } -} - -function handleIfOnCorrectNetwork(currentNetworkId: number, desiredNetworkId: number) { - if (desiredNetworkId === currentNetworkId) { - // enable the button once on the correct network - resetClaimButton(); - invalidateButton.disabled = false; - } else { - loadingClaimButton(false); - invalidateButton.disabled = true; - } -} - -export async function switchNetwork(provider: ethers.providers.Web3Provider, networkId: number): Promise { - try { - await provider.send("wallet_switchEthereumChain", [{ chainId: "0x" + networkId.toString(16) }]); - return true; - } catch (error: unknown) { - // Add network if it doesn't exist. - const code = (error as { code: number }).code; - if (code == 4902) { - return await addNetwork(provider, networkId); - } - return false; - } -} - -export async function addNetwork(provider: ethers.providers.Web3Provider, networkId: number): Promise { - try { - await provider.send("wallet_addEthereumChain", [ - { - chainId: "0x" + networkId.toString(16), - chainName: getNetworkName(networkId), - rpcUrls: networkRpcs[networkId], - blockExplorerUrls: [networkExplorers[networkId]], - nativeCurrency: networkCurrencies[networkId], - }, - ]); - return true; - } catch (error: unknown) { - return false; - } -}