diff --git a/.changeset/violet-lobsters-smile.md b/.changeset/violet-lobsters-smile.md new file mode 100644 index 00000000000..8f94a0eeacb --- /dev/null +++ b/.changeset/violet-lobsters-smile.md @@ -0,0 +1,6 @@ +--- +"@wso2is/admin.actions.v1": minor +"@wso2is/i18n": patch +--- + +Add getActionByActionId api to actions ui diff --git a/features/admin.actions.v1/api/use-get-action-by-id.ts b/features/admin.actions.v1/api/use-get-action-by-id.ts new file mode 100644 index 00000000000..238697ca2c4 --- /dev/null +++ b/features/admin.actions.v1/api/use-get-action-by-id.ts @@ -0,0 +1,64 @@ +/** + * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). + * + * WSO2 LLC. licenses this file to you under the Apache License, + * Version 2.0 (the "License"); you may not use this file except + * in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, + * software distributed under the License is distributed on an + * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY + * KIND, either express or implied. See the License for the + * specific language governing permissions and limitations + * under the License. + */ + +import useRequest, { + RequestConfigInterface, + RequestErrorInterface, + RequestResultInterface +} from "@wso2is/admin.core.v1/hooks/use-request"; +import { store } from "@wso2is/admin.core.v1/store"; +import { HttpMethods } from "@wso2is/core/models"; +import isEmpty from "lodash-es/isEmpty"; +import { ActionResponseInterface } from "../models/actions"; + +/** + * Hook to get the action configurations by id. + * + * @param actionType - Type of the action. + * @param actionId - ID of the action. + * @returns SWR response object containing the data, error, isLoading, isValidating, mutate. + */ +const useGetActionById = + (actionType: string, actionId: string): RequestResultInterface => { + + const requestConfig: RequestConfigInterface = { + headers: { + Accept: "application/json", + "Content-Type": "application/json" + }, + method: HttpMethods.GET, + url: `${ store.getState().config.endpoints.actions }/${ actionType }/${ actionId }` + }; + + const shouldFetch: boolean = !isEmpty(actionId); + + const { data, error, isLoading, isValidating, mutate } = useRequest( + shouldFetch ? requestConfig : null, + { shouldRetryOnError: false } + ); + + return { + data, + error, + isLoading, + isValidating, + mutate + }; +}; + +export default useGetActionById; diff --git a/features/admin.actions.v1/api/use-get-actions-by-type.ts b/features/admin.actions.v1/api/use-get-actions-by-type.ts index 9f896992657..47e6354c37c 100644 --- a/features/admin.actions.v1/api/use-get-actions-by-type.ts +++ b/features/admin.actions.v1/api/use-get-actions-by-type.ts @@ -23,16 +23,16 @@ import useRequest, { } from "@wso2is/admin.core.v1/hooks/use-request"; import { store } from "@wso2is/admin.core.v1/store"; import { HttpMethods } from "@wso2is/core/models"; -import { ActionResponseInterface } from "../models/actions"; +import { ActionBasicResponseInterface } from "../models/actions"; /** - * Hook to get the action configurations. + * Hook to get the actions configurations by action type. * * @param actionType - Type of the action. * @param shouldFetch - Should fetch the data. * @returns SWR response object containing the data, error, isLoading, isValidating, mutate. */ -const useGetActionsByType = +const useGetActionsByType = (actionType: string, shouldFetch: boolean = true): RequestResultInterface => { const requestConfig: RequestConfigInterface = { diff --git a/features/admin.actions.v1/components/action-config-form.tsx b/features/admin.actions.v1/components/action-config-form.tsx index 1dcdc69afe6..5c77b78872a 100644 --- a/features/admin.actions.v1/components/action-config-form.tsx +++ b/features/admin.actions.v1/components/action-config-form.tsx @@ -46,6 +46,7 @@ import { Dispatch } from "redux"; import { Icon } from "semantic-ui-react"; import createAction from "../api/create-action"; import updateAction from "../api/update-action"; +import useGetActionById from "../api/use-get-action-by-id"; import useGetActionsByType from "../api/use-get-actions-by-type"; import { ActionsConstants } from "../constants/actions-constants"; import { @@ -106,6 +107,10 @@ const ActionConfigForm: FunctionComponent = ({ mutate: mutateActions } = useGetActionsByType(actionTypeApiPath); + const { + mutate: mutateAction + } = useGetActionById(actionTypeApiPath, initialValues?.id); + /** * The following useEffect is used to set the current Action Authentication Type. */ @@ -342,7 +347,7 @@ const ActionConfigForm: FunctionComponent = ({ .then(() => { handleSuccess(ActionsConstants.UPDATE); setIsAuthenticationUpdateFormState(false); - mutateActions(); + mutateAction(); }) .catch((error: AxiosError) => { handleError(error, ActionsConstants.UPDATE); diff --git a/features/admin.actions.v1/models/actions.ts b/features/admin.actions.v1/models/actions.ts index f61dd236c4e..b931e439983 100644 --- a/features/admin.actions.v1/models/actions.ts +++ b/features/admin.actions.v1/models/actions.ts @@ -16,6 +16,7 @@ * under the License. */ +import { HttpMethod } from "@asgardeo/auth-react"; import { FeatureStatusLabel } from "@wso2is/admin.feature-gate.v1/models/feature-status"; import { ReactNode } from "react"; @@ -102,6 +103,31 @@ interface AuthenticationInterface { properties: Partial; } +/** + * Link Relation type. + */ +enum Relation { + SELF = "self" +} + +/** + * Link Interface. + */ +interface LinkInterface { + /** + * Url of the endpoint. + */ + href: string; + /** + * Http method. + */ + method: HttpMethod + /** + * Relation to the resource. + */ + rel: Relation; +} + /** * Authentication Properties. */ @@ -129,9 +155,9 @@ export interface AuthenticationPropertiesInterface { } /** - * Action Basic Response. + * Action Base Response. */ -export interface ActionBasicResponseInterface { +export interface ActionBaseResponseInterface { /** * ID of the Action. */ @@ -154,10 +180,20 @@ export interface ActionBasicResponseInterface { status: ActionStatus; } +/** + * Action Basic Response. + */ +export interface ActionBasicResponseInterface extends ActionBaseResponseInterface { + /** + * Links of the Action. + */ + links: LinkInterface[]; +} + /** * Action Response. */ -export interface ActionResponseInterface extends ActionBasicResponseInterface { +export interface ActionResponseInterface extends ActionBaseResponseInterface { /** * Endpoint configuration of the Action. */ diff --git a/features/admin.actions.v1/pages/action-configuration-page.tsx b/features/admin.actions.v1/pages/action-configuration-page.tsx index e50798c4ee3..be900c321ff 100644 --- a/features/admin.actions.v1/pages/action-configuration-page.tsx +++ b/features/admin.actions.v1/pages/action-configuration-page.tsx @@ -44,6 +44,7 @@ import { Dispatch } from "redux"; import { Checkbox, CheckboxProps, Grid } from "semantic-ui-react"; import changeActionStatus from "../api/change-action-status"; import deleteAction from "../api/delete-action"; +import useGetActionById from "../api/use-get-action-by-id"; import useGetActionsByType from "../api/use-get-actions-by-type"; import ActionConfigForm from "../components/action-config-form"; import { ActionsConstants } from "../constants/actions-constants"; @@ -97,20 +98,29 @@ const ActionConfigurationPage: FunctionComponent actions?.[0]?.id || null, [ actions ]); + + const { + data: action, + error: actionFetchRequestError, + isLoading: isActionLoading, + mutate: mutateAction + } = useGetActionById(actionTypeApiPath, actionId); + + const isLoading: boolean = isActionsLoading || !actions || !Array.isArray(actions) || isActionLoading; const actionInitialValues: ActionConfigFormPropertyInterface = useMemo(() => { - if (actions) { + if (action) { return { - authenticationType: actions[0]?.endpoint?.authentication?.type.toString(), - endpointUri: actions[0]?.endpoint?.uri, - id: actions[0]?.id, - name: actions[0]?.name + authenticationType: action?.endpoint?.authentication?.type.toString(), + endpointUri: action?.endpoint?.uri, + id: action?.id, + name: action?.name }; } else { return null; } - }, [ actions ]); + }, [ action ]); useEffect(() => { if (actions?.length >= 1) { @@ -122,7 +132,7 @@ const ActionConfigurationPage: FunctionComponent { if (isActionsLoading || !actionsFetchRequestError) { @@ -132,23 +142,51 @@ const ActionConfigurationPage: FunctionComponent({ - description: t("actions:notification.error.fetch.description", + description: t("actions:notification.error.fetchByType.description", { description: actionsFetchRequestError.response.data.description }), level: AlertLevels.ERROR, - message: t("actions:notification.error.fetch.message") + message: t("actions:notification.error.fetchByType.message") }) ); } else { dispatch( addAlert({ - description: t("actions:notification.genericError.fetch.description"), + description: t("actions:notification.genericError.fetchByType.description"), level: AlertLevels.ERROR, - message: t("actions:notification.genericError.fetch.message") + message: t("actions:notification.genericError.fetchByType.message") }) ); } }, [ isActionsLoading, actionsFetchRequestError ]); + /** + * The following useEffect is used to handle if any error occurs while fetching the Action by Id. + */ + useEffect(() => { + if (isActionLoading || !actionFetchRequestError) { + return; + } + + if (actionFetchRequestError.response?.data?.description) { + dispatch( + addAlert({ + description: t("actions:notification.error.fetchById.description", + { description: actionFetchRequestError.response.data.description }), + level: AlertLevels.ERROR, + message: t("actions:notification.error.fetchById.message") + }) + ); + } else { + dispatch( + addAlert({ + description: t("actions:notification.genericError.fetchById.description"), + level: AlertLevels.ERROR, + message: t("actions:notification.genericError.fetchById.message") + }) + ); + } + }, [ isActionLoading, actionFetchRequestError ]); + /** * Handles the back button click event. */ @@ -258,7 +296,7 @@ const ActionConfigurationPage: FunctionComponent { - mutateActions(); + mutateAction(); setIsSubmitting(false); }); }; diff --git a/modules/i18n/src/models/namespaces/actions-ns.ts b/modules/i18n/src/models/namespaces/actions-ns.ts index 60681266652..6dd77156847 100644 --- a/modules/i18n/src/models/namespaces/actions-ns.ts +++ b/modules/i18n/src/models/namespaces/actions-ns.ts @@ -155,7 +155,11 @@ export interface actionsNS { description: string; message: string; }; - fetch: { + fetchById: { + description: string; + message: string; + }; + fetchByType: { description: string; message: string; }; @@ -185,7 +189,11 @@ export interface actionsNS { description: string; message: string; }; - fetch: { + fetchById: { + description: string; + message: string; + }; + fetchByType: { description: string; message: string; }; diff --git a/modules/i18n/src/translations/en-US/portals/actions.ts b/modules/i18n/src/translations/en-US/portals/actions.ts index 1b84736cdbc..f01e41f244b 100644 --- a/modules/i18n/src/translations/en-US/portals/actions.ts +++ b/modules/i18n/src/translations/en-US/portals/actions.ts @@ -162,13 +162,17 @@ export const actions: actionsNS = { description: "{{description}}", message: "Error deleting the action." }, - fetch: { + fetchById: { description: "{{description}}", - message: "Error fetching the action." + message: "Error retrieving the action." + }, + fetchByType: { + description: "{{description}}", + message: "Error retrieving actions." }, typesFetch: { description: "{{description}}", - message: "Error fetching the action types." + message: "Error retrieving the action types." }, update: { description: "{{description}}", @@ -192,12 +196,16 @@ export const actions: actionsNS = { description: "Couldn't delete the action.", message: "Something went wrong." }, - fetch: { - description: "Couldn't fetch the action.", + fetchById: { + description: "Couldn't retrieve the action.", + message: "Something went wrong." + }, + fetchByType: { + description: "Couldn't retrieve actions.", message: "Something went wrong." }, typesFetch: { - description: "Couldn't fetch the action types.", + description: "Couldn't retrieve the action types.", message: "Something went wrong." }, update: {