From 22794a57bbf45cac72dd69bf9838c63b240e6e22 Mon Sep 17 00:00:00 2001 From: Dan Lee <142251406+dan-aztec@users.noreply.github.com> Date: Fri, 13 Oct 2023 07:23:08 -0700 Subject: [PATCH] feat: private token box upgrade to new Token contract (#2824) https://github.com/AztecProtocol/aztec-packages/issues/2350 https://github.com/AztecProtocol/aztec-packages/issues/2823 We used a "deprecated" pattern for the `private-token` box contract, which has been superceded by Lasse's new `Token` contract. copied over the e2e test for that one with small changes, since we don't have all the e2e utils available --- .circleci/config.yml | 6 +- build_manifest.yml | 4 +- .../app/components/contract_function_form.tsx | 165 - .../src/artifacts/PrivateToken.json | 264 -- .../src/artifacts/PrivateToken.ts | 94 - .../private-token/src/contracts/Nargo.toml | 9 - .../private-token/src/contracts/src/main.nr | 101 - .../src/tests/privatetoken.sandbox.test.ts | 122 - .../{private-token => token}/.eslintrc.cjs | 0 .../boxes/{private-token => token}/.gitignore | 0 .../{private-token => token}/.prettierrc.json | 0 .../boxes/{private-token => token}/README.md | 0 .../docker-compose.yml | 4 +- .../{private-token => token}/package.json | 3 +- .../postcss.config.cjs | 0 .../boxes/{private-token => token}/run_tests | 2 +- .../src/@types/index.d.ts | 0 .../contract_function_form.module.scss | 0 .../app/components/contract_function_form.tsx | 225 ++ .../src/app/components/copy.module.scss | 0 .../src/app/components/copy.tsx | 0 .../src/app/components/dropdown.module.scss | 0 .../src/app/components/dropdown.tsx | 0 .../src/app/components/index.ts | 0 .../src/app/components/popup.module.scss | 0 .../src/app/components/popup.tsx | 0 .../src/app/components/select.module.scss | 0 .../src/app/components/select.tsx | 0 .../src/app/components/terms.tsx | 0 .../components/wallet_dropdown.module.scss | 0 .../src/app/components/wallet_dropdown.tsx | 0 .../src/app/contract.module.scss | 0 .../src/app/contract.tsx | 0 .../src/app/home.module.scss | 0 .../{private-token => token}/src/app/home.tsx | 0 .../src/app/index.css | 0 .../src/app/index.html | 0 .../src/app/index.tsx | 0 .../boxes/token/src/artifacts/Token.json | 3360 +++++++++++++++++ .../boxes/token/src/artifacts/Token.ts | 162 + .../src/assets/aztec_logo.svg | 0 .../src/assets/check.svg | 0 .../src/assets/copy.svg | 0 .../src/assets/favicon.ico | Bin .../src/assets/soehne-leicht-kursiv.ttf | Bin .../src/assets/soehne-web-buch.woff2 | Bin .../assets/soehne-web-halbfett-kursiv.woff2 | Bin .../src/assets/soehne-web-halbfett.woff2 | Bin .../src/assets/soehne-web-kraftig.woff2 | Bin .../src/assets/soehne-web-leicht.woff2 | Bin .../{private-token => token}/src/config.ts | 6 +- .../boxes/token/src/contracts/Nargo.toml | 11 + .../boxes/token/src/contracts/src/main.nr | 446 +++ .../boxes/token/src/contracts/src/types.nr | 5 + .../src/contracts/src/types/balance_set.nr | 132 + .../src/contracts/src/types/balances_map.nr | 27 + .../src/types/safe_u120_serialization.nr | 18 + .../src/contracts/src/types/token_note.nr | 143 + .../contracts/src/types/transparent_note.nr | 123 + .../src/scripts/call_contract_function.ts | 0 .../src/scripts/deploy_contract.ts | 0 .../src/scripts/index.ts | 0 .../src/scripts/util.ts | 39 +- .../src/scripts/view_contract_function.ts | 0 .../token/src/tests/token.contract.test.ts | 975 +++++ .../boxes/token/src/tests/token_simulator.ts | 95 + .../{private-token => token}/src/typings.d.ts | 0 .../tailwind.config.cjs | 0 .../tsconfig.dest.json | 0 .../{private-token => token}/tsconfig.json | 0 .../webpack.config.js | 0 yarn-project/cli/src/unbox.ts | 6 +- yarn-project/package.json | 2 +- yarn-project/tsconfig.json | 2 +- .../yarn-project-base/Dockerfile.dockerignore | 2 +- yarn-project/yarn.lock | 99 +- 76 files changed, 5818 insertions(+), 834 deletions(-) delete mode 100644 yarn-project/boxes/private-token/src/app/components/contract_function_form.tsx delete mode 100644 yarn-project/boxes/private-token/src/artifacts/PrivateToken.json delete mode 100644 yarn-project/boxes/private-token/src/artifacts/PrivateToken.ts delete mode 100644 yarn-project/boxes/private-token/src/contracts/Nargo.toml delete mode 100644 yarn-project/boxes/private-token/src/contracts/src/main.nr delete mode 100644 yarn-project/boxes/private-token/src/tests/privatetoken.sandbox.test.ts rename yarn-project/boxes/{private-token => token}/.eslintrc.cjs (100%) rename yarn-project/boxes/{private-token => token}/.gitignore (100%) rename yarn-project/boxes/{private-token => token}/.prettierrc.json (100%) rename yarn-project/boxes/{private-token => token}/README.md (100%) rename yarn-project/boxes/{private-token => token}/docker-compose.yml (92%) rename yarn-project/boxes/{private-token => token}/package.json (98%) rename yarn-project/boxes/{private-token => token}/postcss.config.cjs (100%) rename yarn-project/boxes/{private-token => token}/run_tests (91%) rename yarn-project/boxes/{private-token => token}/src/@types/index.d.ts (100%) rename yarn-project/boxes/{private-token => token}/src/app/components/contract_function_form.module.scss (100%) create mode 100644 yarn-project/boxes/token/src/app/components/contract_function_form.tsx rename yarn-project/boxes/{private-token => token}/src/app/components/copy.module.scss (100%) rename yarn-project/boxes/{private-token => token}/src/app/components/copy.tsx (100%) rename yarn-project/boxes/{private-token => token}/src/app/components/dropdown.module.scss (100%) rename yarn-project/boxes/{private-token => token}/src/app/components/dropdown.tsx (100%) rename yarn-project/boxes/{private-token => token}/src/app/components/index.ts (100%) rename yarn-project/boxes/{private-token => token}/src/app/components/popup.module.scss (100%) rename yarn-project/boxes/{private-token => token}/src/app/components/popup.tsx (100%) rename yarn-project/boxes/{private-token => token}/src/app/components/select.module.scss (100%) rename yarn-project/boxes/{private-token => token}/src/app/components/select.tsx (100%) rename yarn-project/boxes/{private-token => token}/src/app/components/terms.tsx (100%) rename yarn-project/boxes/{private-token => token}/src/app/components/wallet_dropdown.module.scss (100%) rename yarn-project/boxes/{private-token => token}/src/app/components/wallet_dropdown.tsx (100%) rename yarn-project/boxes/{private-token => token}/src/app/contract.module.scss (100%) rename yarn-project/boxes/{private-token => token}/src/app/contract.tsx (100%) rename yarn-project/boxes/{private-token => token}/src/app/home.module.scss (100%) rename yarn-project/boxes/{private-token => token}/src/app/home.tsx (100%) rename yarn-project/boxes/{private-token => token}/src/app/index.css (100%) rename yarn-project/boxes/{private-token => token}/src/app/index.html (100%) rename yarn-project/boxes/{private-token => token}/src/app/index.tsx (100%) create mode 100644 yarn-project/boxes/token/src/artifacts/Token.json create mode 100644 yarn-project/boxes/token/src/artifacts/Token.ts rename yarn-project/boxes/{private-token => token}/src/assets/aztec_logo.svg (100%) rename yarn-project/boxes/{private-token => token}/src/assets/check.svg (100%) rename yarn-project/boxes/{private-token => token}/src/assets/copy.svg (100%) rename yarn-project/boxes/{private-token => token}/src/assets/favicon.ico (100%) rename yarn-project/boxes/{private-token => token}/src/assets/soehne-leicht-kursiv.ttf (100%) rename yarn-project/boxes/{private-token => token}/src/assets/soehne-web-buch.woff2 (100%) rename yarn-project/boxes/{private-token => token}/src/assets/soehne-web-halbfett-kursiv.woff2 (100%) rename yarn-project/boxes/{private-token => token}/src/assets/soehne-web-halbfett.woff2 (100%) rename yarn-project/boxes/{private-token => token}/src/assets/soehne-web-kraftig.woff2 (100%) rename yarn-project/boxes/{private-token => token}/src/assets/soehne-web-leicht.woff2 (100%) rename yarn-project/boxes/{private-token => token}/src/config.ts (60%) create mode 100644 yarn-project/boxes/token/src/contracts/Nargo.toml create mode 100644 yarn-project/boxes/token/src/contracts/src/main.nr create mode 100644 yarn-project/boxes/token/src/contracts/src/types.nr create mode 100644 yarn-project/boxes/token/src/contracts/src/types/balance_set.nr create mode 100644 yarn-project/boxes/token/src/contracts/src/types/balances_map.nr create mode 100644 yarn-project/boxes/token/src/contracts/src/types/safe_u120_serialization.nr create mode 100644 yarn-project/boxes/token/src/contracts/src/types/token_note.nr create mode 100644 yarn-project/boxes/token/src/contracts/src/types/transparent_note.nr rename yarn-project/boxes/{private-token => token}/src/scripts/call_contract_function.ts (100%) rename yarn-project/boxes/{private-token => token}/src/scripts/deploy_contract.ts (100%) rename yarn-project/boxes/{private-token => token}/src/scripts/index.ts (100%) rename yarn-project/boxes/{private-token => token}/src/scripts/util.ts (51%) rename yarn-project/boxes/{private-token => token}/src/scripts/view_contract_function.ts (100%) create mode 100644 yarn-project/boxes/token/src/tests/token.contract.test.ts create mode 100644 yarn-project/boxes/token/src/tests/token_simulator.ts rename yarn-project/boxes/{private-token => token}/src/typings.d.ts (100%) rename yarn-project/boxes/{private-token => token}/tailwind.config.cjs (100%) rename yarn-project/boxes/{private-token => token}/tsconfig.dest.json (100%) rename yarn-project/boxes/{private-token => token}/tsconfig.json (100%) rename yarn-project/boxes/{private-token => token}/webpack.config.js (100%) diff --git a/.circleci/config.yml b/.circleci/config.yml index bc968b697fb..75ea67cb08d 100644 --- a/.circleci/config.yml +++ b/.circleci/config.yml @@ -526,7 +526,7 @@ jobs: name: "Test via adhoc script" command: ./yarn-project/boxes/blank/run_tests - boxes-private-token: + boxes-token: machine: image: ubuntu-2204:2023.07.2 resource_class: large @@ -535,7 +535,7 @@ jobs: - *setup_env - run: name: "Test via adhoc script" - command: ./yarn-project/boxes/private-token/run_tests + command: ./yarn-project/boxes/token/run_tests canary: machine: @@ -1338,7 +1338,7 @@ workflows: - aztec-sandbox-x86_64 <<: *defaults - - boxes-private-token: + - boxes-token: requires: - aztec-sandbox-x86_64 <<: *defaults diff --git a/build_manifest.yml b/build_manifest.yml index f806ea421e3..0421afc6749 100644 --- a/build_manifest.yml +++ b/build_manifest.yml @@ -131,9 +131,9 @@ boxes-blank: dependencies: - aztec-sandbox -boxes-private-token: +boxes-token: buildDir: yarn-project - projectDir: yarn-project/boxes/private-token + projectDir: yarn-project/boxes/token dependencies: - aztec-sandbox diff --git a/yarn-project/boxes/private-token/src/app/components/contract_function_form.tsx b/yarn-project/boxes/private-token/src/app/components/contract_function_form.tsx deleted file mode 100644 index 7cdb4fa680e..00000000000 --- a/yarn-project/boxes/private-token/src/app/components/contract_function_form.tsx +++ /dev/null @@ -1,165 +0,0 @@ -import { CONTRACT_ADDRESS_PARAM_NAMES, pxe } from '../../config.js'; -import { callContractFunction, deployContract, viewContractFunction } from '../../scripts/index.js'; -import { convertArgs } from '../../scripts/util.js'; -import styles from './contract_function_form.module.scss'; -import { Button, Loader } from '@aztec/aztec-ui'; -import { AztecAddress, CompleteAddress, Fr } from '@aztec/aztec.js'; -import { ContractArtifact, FunctionArtifact } from '@aztec/foundation/abi'; -import { useFormik } from 'formik'; -import * as Yup from 'yup'; - -type NoirFunctionYupSchema = { - // hack: add `any` at the end to get the array schema to typecheck - // eslint-disable-next-line @typescript-eslint/no-explicit-any - [key: string]: Yup.NumberSchema | Yup.ArraySchema | Yup.BooleanSchema | any; -}; - -type NoirFunctionFormValues = { - [key: string]: string | number | number[] | boolean; -}; - -function generateYupSchema(functionAbi: FunctionArtifact, defaultAddress: string) { - const parameterSchema: NoirFunctionYupSchema = {}; - const initialValues: NoirFunctionFormValues = {}; - for (const param of functionAbi.parameters) { - if (CONTRACT_ADDRESS_PARAM_NAMES.includes(param.name)) { - // these are hex strings instead, but yup doesn't support bigint so we convert back to bigint on execution - parameterSchema[param.name] = Yup.string().required(); - initialValues[param.name] = defaultAddress; - continue; - } - switch (param.type.kind) { - case 'field': - parameterSchema[param.name] = Yup.number().required(); - initialValues[param.name] = 100; - break; - // not really needed for private token, since we hide the nullifier helper method which has the array input - case 'array': - // eslint-disable-next-line no-case-declarations - const arrayLength = param.type.length; - parameterSchema[param.name] = Yup.array() - .of(Yup.number()) - .min(arrayLength) - .max(arrayLength) - .transform(function (value: number[], originalValue: string) { - if (typeof originalValue === 'string') { - return originalValue.split(',').map(Number); - } - return value; - }); - initialValues[param.name] = Array(arrayLength).fill( - CONTRACT_ADDRESS_PARAM_NAMES.includes(param.name) ? defaultAddress : 200, - ); - break; - case 'boolean': - parameterSchema[param.name] = Yup.boolean().required(); - initialValues[param.name] = false; - break; - } - } - return { validationSchema: Yup.object().shape(parameterSchema), initialValues }; -} - -async function handleFunctionCall( - contractAddress: AztecAddress | undefined, - artifact: ContractArtifact, - functionName: string, - args: any, - wallet: CompleteAddress, -) { - const functionAbi = artifact.functions.find(f => f.name === functionName)!; - const typedArgs: any[] = convertArgs(functionAbi, args); - - if (functionName === 'constructor' && !!wallet) { - if (functionAbi === undefined) { - throw new Error('Cannot find constructor in the ABI.'); - } - // hack: addresses are stored as string in the form to avoid bigint compatibility issues with formik - // convert those back to bigints before sending - - // for now, dont let user change the salt. requires some change to the form generation if we want to let user choose one - // since everything is currently based on parsing the contractABI, and the salt parameter is not present there - const salt = Fr.random(); - return await deployContract(wallet, artifact, typedArgs, salt, pxe); - } - - if (functionAbi.functionType === 'unconstrained') { - return await viewContractFunction(contractAddress!, artifact, functionName, typedArgs, pxe, wallet); - } else { - const txnReceipt = await callContractFunction(contractAddress!, artifact, functionName, typedArgs, pxe, wallet); - return `Transaction ${txnReceipt.status} on block number ${txnReceipt.blockNumber}`; - } -} - -interface ContractFunctionFormProps { - wallet: CompleteAddress; - contractAddress?: AztecAddress; - artifact: ContractArtifact; - functionAbi: FunctionArtifact; - defaultAddress: string; - title?: string; - buttonText?: string; - isLoading: boolean; - disabled: boolean; - onSubmit: () => void; - onSuccess: (result: any) => void; - onError: (msg: string) => void; -} - -export function ContractFunctionForm({ - wallet, - contractAddress, - artifact, - functionAbi, - defaultAddress, - buttonText = 'Submit', - isLoading, - disabled, - onSubmit, - onSuccess, - onError, -}: ContractFunctionFormProps) { - const { validationSchema, initialValues } = generateYupSchema(functionAbi, defaultAddress); - const formik = useFormik({ - initialValues: initialValues, - validationSchema: validationSchema, - onSubmit: async (values: any) => { - onSubmit(); - try { - const result = await handleFunctionCall(contractAddress, artifact, functionAbi.name, values, wallet); - onSuccess(result); - } catch (e: any) { - onError(e.message); - } - }, - }); - - return ( -
- {functionAbi.parameters.map(input => ( -
- - - {formik.touched[input.name] && formik.errors[input.name] && ( -
{formik.errors[input.name]?.toString()}
- )} -
- ))} - {isLoading ? ( - - ) : ( -