From 6f2fecb38c8b96175f90d02089907eb54f747db8 Mon Sep 17 00:00:00 2001 From: Oleg Chendighelean Date: Fri, 4 Oct 2024 18:11:20 +0100 Subject: [PATCH] Move network fields validation scheme to @umami/components pkg --- apps/desktop/package.json | 1 + .../network/UpsertNetworkModal.test.tsx | 66 ++++-- .../settings/network/UpsertNetworkModal.tsx | 22 +- .../Menu/NetworkMenu/EditNetworkMenu.test.tsx | 87 ++++---- .../Menu/NetworkMenu/EditNetworkMenu.tsx | 25 +-- .../AddressPill/AddressPillText.test.tsx | 2 +- .../AddressPill/AddressPillText.tsx | 0 .../src/{ => components}/AddressPill/index.ts | 0 .../{ => components}/AddressPill/testUtils.ts | 3 +- .../src/{ => components}/AddressPill/types.ts | 0 .../AddressPill/useAddressKind.test.ts | 2 +- .../AddressPill/useAddressKind.ts | 0 .../AddressPill/useAddressPill.test.tsx | 6 +- .../AddressPill/useAddressPill.ts | 0 .../DynamicDisclosure.test.tsx | 2 +- .../DynamicDisclosure/DynamicDisclosure.tsx | 0 .../DynamicDisclosure/index.ts | 0 .../DynamicDisclosure/useMultiForm.test.tsx | 2 +- .../DynamicDisclosure/useMultiForm.tsx | 0 .../MnemonicAutocomplete.test.tsx | 2 +- .../MnemonicAutocomplete.tsx | 0 .../MnemonicAutocomplete/index.ts | 0 .../ReactIdenticon}/ReactIdenticon.tsx | 0 .../src/components/ReactIdenticon/index.ts | 1 + packages/components/src/components/index.ts | 4 + packages/components/src/index.ts | 6 +- packages/components/src/testUtils.tsx | 2 +- packages/components/src/utils/index.ts | 1 + .../components/src/utils/validationSchemes.ts | 21 ++ pnpm-lock.yaml | 205 ++---------------- 30 files changed, 168 insertions(+), 292 deletions(-) rename packages/components/src/{ => components}/AddressPill/AddressPillText.test.tsx (97%) rename packages/components/src/{ => components}/AddressPill/AddressPillText.tsx (100%) rename packages/components/src/{ => components}/AddressPill/index.ts (100%) rename packages/components/src/{ => components}/AddressPill/testUtils.ts (73%) rename packages/components/src/{ => components}/AddressPill/types.ts (100%) rename packages/components/src/{ => components}/AddressPill/useAddressKind.test.ts (99%) rename packages/components/src/{ => components}/AddressPill/useAddressKind.ts (100%) rename packages/components/src/{ => components}/AddressPill/useAddressPill.test.tsx (94%) rename packages/components/src/{ => components}/AddressPill/useAddressPill.ts (100%) rename packages/components/src/{ => components}/DynamicDisclosure/DynamicDisclosure.test.tsx (99%) rename packages/components/src/{ => components}/DynamicDisclosure/DynamicDisclosure.tsx (100%) rename packages/components/src/{ => components}/DynamicDisclosure/index.ts (100%) rename packages/components/src/{ => components}/DynamicDisclosure/useMultiForm.test.tsx (99%) rename packages/components/src/{ => components}/DynamicDisclosure/useMultiForm.tsx (100%) rename packages/components/src/{ => components}/MnemonicAutocomplete/MnemonicAutocomplete.test.tsx (98%) rename packages/components/src/{ => components}/MnemonicAutocomplete/MnemonicAutocomplete.tsx (100%) rename packages/components/src/{ => components}/MnemonicAutocomplete/index.ts (100%) rename packages/components/src/{ => components/ReactIdenticon}/ReactIdenticon.tsx (100%) create mode 100644 packages/components/src/components/ReactIdenticon/index.ts create mode 100644 packages/components/src/components/index.ts create mode 100644 packages/components/src/utils/index.ts create mode 100644 packages/components/src/utils/validationSchemes.ts diff --git a/apps/desktop/package.json b/apps/desktop/package.json index c26fd4894c..78441d70c6 100644 --- a/apps/desktop/package.json +++ b/apps/desktop/package.json @@ -154,6 +154,7 @@ }, "packageManager": "pnpm@9.9.0", "dependencies": { + "@hookform/resolvers": "^3.9.0", "electron-updater": "6.3.4" } } diff --git a/apps/desktop/src/views/settings/network/UpsertNetworkModal.test.tsx b/apps/desktop/src/views/settings/network/UpsertNetworkModal.test.tsx index fd1524f60c..e6e3714f7c 100644 --- a/apps/desktop/src/views/settings/network/UpsertNetworkModal.test.tsx +++ b/apps/desktop/src/views/settings/network/UpsertNetworkModal.test.tsx @@ -12,6 +12,14 @@ beforeEach(() => { }); describe("", () => { + const updatedNetwork = { + ...customNetwork, + rpcUrl: "https://rpc.com", + tzktApiUrl: "https://tzkt.com", + tzktExplorerUrl: "https://explorer.com", + buyTezUrl: "", + }; + describe("edit mode", () => { beforeEach(() => { store.dispatch(networksActions.upsertNetwork(customNetwork)); @@ -27,14 +35,6 @@ describe("", () => { const user = userEvent.setup(); render(, { store }); - const updatedNetwork = { - ...customNetwork, - rpcUrl: "https://rpc", - tzktApiUrl: "https://tzkt", - tzktExplorerUrl: "https://explorer", - buyTezUrl: "", - }; - await act(() => user.clear(screen.getByLabelText("RPC URL"))); await act(() => user.clear(screen.getByLabelText("Tzkt API URL"))); await act(() => user.clear(screen.getByLabelText("Tzkt Explorer URL"))); @@ -55,14 +55,6 @@ describe("", () => { const user = userEvent.setup(); render(, { store }); - const updatedNetwork = { - ...customNetwork, - rpcUrl: "https://rpc", - tzktApiUrl: "https://tzkt", - tzktExplorerUrl: "https://explorer", - buyTezUrl: "", - }; - await act(() => user.clear(screen.getByLabelText("RPC URL"))); await act(() => user.clear(screen.getByLabelText("Tzkt API URL"))); await act(() => user.clear(screen.getByLabelText("Tzkt Explorer URL"))); @@ -85,6 +77,48 @@ describe("", () => { }); }); + describe("URL fields validation", () => { + const urlFields = [ + { label: "RPC URL", required: true }, + { label: "Tzkt API URL", required: true }, + { label: "Tzkt Explorer URL", required: true }, + { label: "Buy Tez URL", required: false }, + ]; + + it.each(urlFields)("validates $label field", async ({ label, required }) => { + const user = userEvent.setup(); + render(, { store }); + + await user.type(screen.getByLabelText(label), "invalid-url"); + await user.tab(); + + await waitFor(() => { + expect(screen.getByText(`Enter a valid ${label}`)).toBeVisible(); + }); + + await user.clear(screen.getByLabelText(label)); + await user.tab(); + + if (required) { + await waitFor(() => { + expect(screen.getByText(`${label} is required`)).toBeVisible(); + }); + } else { + await waitFor(() => { + expect(screen.queryByText(`${label} is required`)).not.toBeInTheDocument(); + }); + } + + await user.type(screen.getByLabelText(label), "https://valid-url.com"); + await user.tab(); + + await waitFor(() => { + expect(screen.queryByText(`Enter a valid ${label}`)).not.toBeInTheDocument(); + }); + expect(screen.queryByText(`${label} is required`)).not.toBeInTheDocument(); + }); + }); + describe("create mode", () => { describe("name field", () => { it("validates uniqueness", async () => { diff --git a/apps/desktop/src/views/settings/network/UpsertNetworkModal.tsx b/apps/desktop/src/views/settings/network/UpsertNetworkModal.tsx index 419df9af0f..6023157048 100644 --- a/apps/desktop/src/views/settings/network/UpsertNetworkModal.tsx +++ b/apps/desktop/src/views/settings/network/UpsertNetworkModal.tsx @@ -9,7 +9,8 @@ import { ModalFooter, ModalHeader, } from "@chakra-ui/react"; -import { useDynamicModalContext } from "@umami/components"; +import { zodResolver } from "@hookform/resolvers/zod"; +import { getNetworkValidationScheme, useDynamicModalContext } from "@umami/components"; import { networksActions, useAvailableNetworks } from "@umami/state"; import { type Network } from "@umami/tezos"; import { useForm } from "react-hook-form"; @@ -30,7 +31,11 @@ export const UpsertNetworkModal = ({ network }: { network?: Network }) => { formState: { errors, isValid }, register, handleSubmit, - } = useForm({ mode: "onBlur", defaultValues: network }); + } = useForm({ + mode: "onBlur", + defaultValues: network, + resolver: zodResolver(getNetworkValidationScheme(availableNetworks, network)), + }); const onSubmit = (network: Network) => { dispatch(networksActions.upsertNetwork(network)); @@ -51,12 +56,7 @@ export const UpsertNetworkModal = ({ network }: { network?: Network }) => { { - if (availableNetworks.find(n => n.name === name)) { - return "Network with this name already exists"; - } - }, + setValueAs: removeTrailingSlashes, })} /> {errors.name && {errors.name.message}} @@ -67,7 +67,6 @@ export const UpsertNetworkModal = ({ network }: { network?: Network }) => { @@ -78,7 +77,6 @@ export const UpsertNetworkModal = ({ network }: { network?: Network }) => { @@ -89,7 +87,6 @@ export const UpsertNetworkModal = ({ network }: { network?: Network }) => { @@ -98,9 +95,10 @@ export const UpsertNetworkModal = ({ network }: { network?: Network }) => { )} - + Buy Tez URL + {errors.buyTezUrl && {errors.buyTezUrl.message}}