import React, { useEffect, useState } from "react"; import { Link } from "react-router-dom"; import { Trans, useTranslation } from "react-i18next"; import { Controller, useFormContext } from "react-hook-form"; import moment from "moment"; import { ActionGroup, AlertVariant, Button, ButtonVariant, Card, CardBody, ExpandableSection, FormGroup, Split, SplitItem, Text, TextInput, ToolbarItem, } from "@patternfly/react-core"; import GlobalRequestResult from "keycloak-admin/lib/defs/globalRequestResult"; import ClientRepresentation from "keycloak-admin/lib/defs/clientRepresentation"; import { convertToFormValues, toUpperCase } from "../util"; import { FormAccess } from "../components/form-access/FormAccess"; import { ScrollForm } from "../components/scroll-form/ScrollForm"; import { HelpItem } from "../components/help-enabler/HelpItem"; import { KeycloakDataTable } from "../components/table-toolbar/KeycloakDataTable"; import { FineGrainOpenIdConnect } from "./advanced/FineGrainOpenIdConnect"; import { OpenIdConnectCompatibilityModes } from "./advanced/OpenIdConnectCompatibilityModes"; import { AdvancedSettings } from "./advanced/AdvancedSettings"; import { TimeSelector } from "../components/time-selector/TimeSelector"; import { useAdminClient } from "../context/auth/AdminClient"; import { useAlerts } from "../components/alert/Alerts"; import { useConfirmDialog } from "../components/confirm-dialog/ConfirmDialog"; import { AddHostDialog } from "./advanced/AddHostDialog"; import { FineGrainSamlEndpointConfig } from "./advanced/FineGrainSamlEndpointConfig"; import { AuthenticationOverrides } from "./advanced/AuthenticationOverrides"; import { useRealm } from "../context/realm-context/RealmContext"; type AdvancedProps = { save: () => void; client: ClientRepresentation; }; export const AdvancedTab = ({ save, client: { id, registeredNodes, attributes, protocol, authenticationFlowBindingOverrides, }, }: AdvancedProps) => { const { t } = useTranslation("clients"); const adminClient = useAdminClient(); const { realm } = useRealm(); const { addAlert } = useAlerts(); const revocationFieldName = "notBefore"; const openIdConnect = "openid-connect"; const { getValues, setValue, register, control, reset } = useFormContext(); const [expanded, setExpanded] = useState(false); const [selectedNode, setSelectedNode] = useState(""); const [addNodeOpen, setAddNodeOpen] = useState(false); const [key, setKey] = useState(0); const refresh = () => setKey(new Date().getTime()); const [nodes, setNodes] = useState(registeredNodes || {}); const setNotBefore = (time: number) => { setValue(revocationFieldName, time); save(); }; const parseResult = (result: GlobalRequestResult, prefixKey: string) => { const successCount = result.successRequests?.length || 0; const failedCount = result.failedRequests?.length || 0; if (successCount === 0 && failedCount === 0) { addAlert(t("noAdminUrlSet"), AlertVariant.warning); } else if (failedCount > 0) { addAlert( t(prefixKey + "Success", { successNodes: result.successRequests }), AlertVariant.success ); addAlert( t(prefixKey + "Fail", { failedNodes: result.failedRequests }), AlertVariant.danger ); } else { addAlert( t(prefixKey + "Success", { successNodes: result.successRequests }), AlertVariant.success ); } }; const resetFields = (names: string[]) => { const values: { [name: string]: string } = {}; for (const name of names) { values[`attributes.${name}`] = attributes ? attributes[name.replace(/-/g, ".")] || "" : ""; } reset(values); }; const push = async () => { const result = ((await adminClient.clients.pushRevocation({ id: id!, })) as unknown) as GlobalRequestResult; parseResult(result, "notBeforePush"); }; const testCluster = async () => { const result = await adminClient.clients.testNodesAvailable({ id: id! }); parseResult(result, "testCluster"); }; const [toggleDeleteNodeConfirm, DeleteNodeConfirm] = useConfirmDialog({ titleKey: "clients:deleteNode", messageKey: t("deleteNodeBody", { node: selectedNode, }), continueButtonLabel: "common:delete", continueButtonVariant: ButtonVariant.danger, onConfirm: async () => { try { await adminClient.clients.deleteClusterNode({ id: id!, node: selectedNode, }); setNodes({ ...Object.keys(nodes).reduce((object: any, key) => { if (key !== selectedNode) { object[key] = nodes[key]; } return object; }, {}), }); refresh(); addAlert(t("deleteNodeSuccess"), AlertVariant.success); } catch (error) { addAlert( t("deleteNodeFail", { error: error.response?.data?.error || error }), AlertVariant.danger ); } }, }); useEffect(() => { register(revocationFieldName); }, [register]); const formatDate = () => { const date = getValues(revocationFieldName); if (date > 0) { return moment(date * 1000).format("LLL"); } else { return t("none"); } }; const sections = [ t("revocation"), t("clustering"), protocol === openIdConnect ? t("fineGrainOpenIdConnectConfiguration") : t("fineGrainSamlEndpointConfig"), t("advancedSettings"), t("authenticationOverrides"), ]; if (protocol === openIdConnect) { sections.splice(3, 0, t("openIdConnectCompatibilityModes")); } return ( In order to successfully push setup url on {t("settings")} tab } > } > ( )} /> { nodes[node] = moment.now() / 1000; refresh(); }} onClose={() => setAddNodeOpen(false)} /> setExpanded(!expanded)} isExpanded={expanded} > Promise.resolve( Object.entries(nodes || {}).map((entry) => { return { host: entry[0], registration: entry[1] }; }) ) } toolbarItem={ <> } actions={[ { title: t("common:delete"), onRowClick: (node) => { setSelectedNode(node.host); toggleDeleteNodeConfirm(); }, }, ]} columns={[ { name: "host", displayKey: "clients:nodeHost", }, { name: "registration", displayKey: "clients:lastRegistration", cellFormatters: [ (value) => value ? moment(parseInt(value.toString()) * 1000).format( "LLL" ) : "", ], }, ]} /> {protocol === openIdConnect && ( <> {t("clients-help:fineGrainOpenIdConnectConfiguration")} convertToFormValues(attributes, "attributes", setValue) } /> )} {protocol !== openIdConnect && ( <> {t("clients-help:fineGrainSamlEndpointConfig")} convertToFormValues(attributes, "attributes", setValue) } /> )} {protocol === openIdConnect && ( {t("clients-help:openIdConnectCompatibilityModes")} resetFields(["exclude-session-state-from-auth-response"]) } /> )} {t("clients-help:advancedSettings" + toUpperCase(protocol!))} { resetFields([ "saml-assertion-lifespan", "access-token-lifespan", "tls-client-certificate-bound-access-tokens", "pkce-code-challenge-method", ]); }} /> {t("clients-help:authenticationOverrides")} { setValue( "authenticationFlowBindingOverrides.browser", authenticationFlowBindingOverrides?.browser ); setValue( "authenticationFlowBindingOverrides.direct_grant", authenticationFlowBindingOverrides?.direct_grant ); }} /> ); };