diff --git a/cypress/integration/realm_settings_test.spec.ts b/cypress/integration/realm_settings_test.spec.ts index 66aaaf1d17..e068db18e1 100644 --- a/cypress/integration/realm_settings_test.spec.ts +++ b/cypress/integration/realm_settings_test.spec.ts @@ -54,6 +54,12 @@ describe("Realm settings", () => { cy.getId("rs-providers-tab").click(); cy.getId("provider-name-link").contains("test_hmac-generated").click(); + sidebarPage.goToRealmSettings(); + + cy.getId("rs-keys-tab").click(); + cy.getId("rs-providers-tab").click(); + cy.getId("provider-name-link").contains("test_rsa-generated").click(); + cy.wait(["@keysFetch"]); return this; diff --git a/src/realm-settings/KeysProvidersTab.tsx b/src/realm-settings/KeysProvidersTab.tsx index 81075497a9..84b7206b1b 100644 --- a/src/realm-settings/KeysProvidersTab.tsx +++ b/src/realm-settings/KeysProvidersTab.tsx @@ -42,7 +42,7 @@ import { JavaKeystoreModal } from "./JavaKeystoreModal"; import { HMACGeneratedModal } from "./key-providers/hmac-generated/HMACGeneratedModal"; import { ECDSAGeneratedModal } from "./key-providers/ecdsa-generated/ECDSAGeneratedModal"; import { RSAModal } from "./RSAModal"; -import { RSAGeneratedModal } from "./RSAGeneratedModal"; +import { RSAGeneratedModal } from "./key-providers/rsa-generated/RSAGeneratedModal"; type ComponentData = KeyMetadataRepresentation & { id?: string; diff --git a/src/realm-settings/key-providers/rsa-generated/RSAGeneratedForm.tsx b/src/realm-settings/key-providers/rsa-generated/RSAGeneratedForm.tsx new file mode 100644 index 0000000000..bc2981566c --- /dev/null +++ b/src/realm-settings/key-providers/rsa-generated/RSAGeneratedForm.tsx @@ -0,0 +1,403 @@ +import React, { useState } from "react"; +import { + ActionGroup, + AlertVariant, + Button, + FormGroup, + PageSection, + Select, + SelectOption, + SelectVariant, + Switch, + TextInput, + ValidatedOptions, +} from "@patternfly/react-core"; +import { useTranslation } from "react-i18next"; +import { Controller, useForm } from "react-hook-form"; + +import type ComponentRepresentation from "keycloak-admin/lib/defs/componentRepresentation"; +import { HelpItem } from "../../../components/help-enabler/HelpItem"; +import { useServerInfo } from "../../../context/server-info/ServerInfoProvider"; +import { useAdminClient, useFetch } from "../../../context/auth/AdminClient"; +import { useParams, useRouteMatch } from "react-router-dom"; +import { FormAccess } from "../../../components/form-access/FormAccess"; +import { ViewHeader } from "../../../components/view-header/ViewHeader"; +import { convertToFormValues } from "../../../util"; +import { useAlerts } from "../../../components/alert/Alerts"; + +type RSAGeneratedFormProps = { + handleModalToggle?: () => void; + refresh?: () => void; + editMode?: boolean; + providerType?: string; +}; + +export interface MatchParams { + providerType: string; +} + +export const RSAGeneratedForm = ({ + editMode, + providerType, + handleModalToggle, + refresh, +}: RSAGeneratedFormProps) => { + const { t } = useTranslation("realm-settings"); + const serverInfo = useServerInfo(); + const [isKeySizeDropdownOpen, setIsKeySizeDropdownOpen] = useState(false); + const [isEllipticCurveDropdownOpen, setIsEllipticCurveDropdownOpen] = + useState(false); + + const adminClient = useAdminClient(); + const { addAlert } = useAlerts(); + + const { id } = useParams<{ id: string }>(); + + const providerId = + useRouteMatch("/:providerType?")?.params.providerType; + + const save = async (component: ComponentRepresentation) => { + try { + if (id) { + await adminClient.components.update( + { id }, + { + ...component, + parentId: component.parentId, + providerId: providerType, + providerType: "org.keycloak.keys.KeyProvider", + } + ); + addAlert(t("saveProviderSuccess"), AlertVariant.success); + } else { + await adminClient.components.create({ + ...component, + parentId: component.parentId, + providerId: providerType, + providerType: "org.keycloak.keys.KeyProvider", + }); + handleModalToggle?.(); + addAlert(t("saveProviderSuccess"), AlertVariant.success); + refresh?.(); + } + } catch (error) { + addAlert( + t("saveProviderError", { + error: error.response?.data?.errorMessage || error, + }), + AlertVariant.danger + ); + } + }; + + const form = useForm({ mode: "onChange" }); + + const setupForm = (component: ComponentRepresentation) => { + form.reset(); + Object.entries(component).map(([key, value]) => { + if ( + key === "config" && + component.config?.secretSize && + component.config?.active && + component.config?.algorithm + ) { + form.setValue("config.secretSize", value.secretSize[0]); + + form.setValue("config.active", value.active[0]); + + form.setValue("config.algorithm", value.active[0]); + + convertToFormValues(value, "config", form.setValue); + } + form.setValue(key, value); + }); + }; + + useFetch( + async () => { + if (editMode) return await adminClient.components.findOne({ id: id }); + }, + (result) => { + if (result) { + setupForm(result); + } + }, + [] + ); + + const allComponentTypes = + serverInfo.componentTypes?.["org.keycloak.keys.KeyProvider"] ?? []; + + const rsaGeneratedKeySizeOptions = + allComponentTypes[5].properties[4].options!; + + const rsaGeneratedAlgorithmOptions = + allComponentTypes[5].properties[3].options; + + return ( + + {editMode && ( + + } + fieldId="id" + isRequired + validated={ + form.errors.name ? ValidatedOptions.error : ValidatedOptions.default + } + helperTextInvalid={t("common:required")} + > + + + )} + + } + fieldId="name" + isRequired + validated={ + form.errors.name ? ValidatedOptions.error : ValidatedOptions.default + } + helperTextInvalid={t("common:required")} + > + {!editMode && ( + { + return ( + onChange(value)} + data-testid="display-name-input" + /> + ); + }} + /> + )} + {editMode && ( + <> + + + )} + + + } + > + ( + { + onChange([value.toString()]); + }} + /> + )} + /> + + + } + > + { + return ( + { + onChange([value.toString()]); + }} + /> + ); + }} + /> + + + } + > + ( + + )} + /> + + + } + > + ( + + )} + /> + + + + + + + ); +}; + +export const RSAGeneratedSettings = () => { + const { t } = useTranslation("realm-settings"); + const providerId = useRouteMatch( + "/:realm/realm-settings/keys/:id?/:providerType?/settings" + )?.params.providerType; + return ( + <> + + + + + + ); +}; diff --git a/src/realm-settings/key-providers/rsa-generated/RSAGeneratedModal.tsx b/src/realm-settings/key-providers/rsa-generated/RSAGeneratedModal.tsx new file mode 100644 index 0000000000..ce9a4423ab --- /dev/null +++ b/src/realm-settings/key-providers/rsa-generated/RSAGeneratedModal.tsx @@ -0,0 +1,36 @@ +import React from "react"; +import { Modal, ModalVariant } from "@patternfly/react-core"; +import { useTranslation } from "react-i18next"; +import { RSAGeneratedForm } from "./RSAGeneratedForm"; + +type RSAGeneratedModalProps = { + providerType: string; + handleModalToggle: () => void; + refresh: () => void; + open: boolean; +}; + +export const RSAGeneratedModal = ({ + providerType, + handleModalToggle, + open, + refresh, +}: RSAGeneratedModalProps) => { + const { t } = useTranslation("realm-settings"); + + return ( + + + + ); +}; diff --git a/src/route-config.ts b/src/route-config.ts index 2580386c44..e3b048991c 100644 --- a/src/route-config.ts +++ b/src/route-config.ts @@ -38,6 +38,7 @@ import { AddOpenIdConnect } from "./identity-providers/add/AddOpenIdConnect"; import { DetailSettings } from "./identity-providers/add/DetailSettings"; import { AESGeneratedSettings } from "./realm-settings/key-providers/aes-generated/AESGeneratedForm"; import { HMACGeneratedSettings } from "./realm-settings/key-providers/hmac-generated/HMACGeneratedForm"; +import { RSAGeneratedSettings } from "./realm-settings/key-providers/rsa-generated/RSAGeneratedForm"; import { ECDSAGeneratedSettings } from "./realm-settings/key-providers/ecdsa-generated/ECDSAGeneratedForm"; export type RouteDef = BreadcrumbsRoute & { @@ -204,6 +205,12 @@ export const routes: RoutesFn = (t: TFunction) => [ breadcrumb: t("realm-settings:editProvider"), access: "view-realm", }, + { + path: "/:realm/realm-settings/keys/:id?/rsa-generated/settings", + component: RSAGeneratedSettings, + breadcrumb: t("realm-settings:editProvider"), + access: "view-realm", + }, { path: "/:realm/authentication", component: AuthenticationSection,