Skip to content
This repository has been archived by the owner on Dec 9, 2024. It is now read-only.

Commit

Permalink
fix(DataMapper): Show a confirmation dialog when DataMapper step is t…
Browse files Browse the repository at this point in the history
…o be deleted, with also offering an option to delete associated mapping file (XSLT)

Fixes: https://github.com/KaotoIO/kaoto-datamapper-integration/issues/97
  • Loading branch information
igarashitm committed Oct 14, 2024
1 parent c97b5f0 commit 7f20716
Show file tree
Hide file tree
Showing 12 changed files with 149 additions and 31 deletions.
15 changes: 12 additions & 3 deletions packages/ui/src/components/DataMapper/on-delete-datamapper.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2,8 +2,17 @@ import { DataMapperMetadataService } from '../../services/datamapper-metadata.se
import { IMetadataApi } from '../../providers';
import { IVisualizationNode } from '../../models';

export const onDeleteDataMapper = (api: IMetadataApi, vizNode: IVisualizationNode) => {
export const ACTION_ID_DELETE_STEP_AND_FILE = 'del-step-and-file';
export const ACTION_ID_DELETE_STEP_ONLY = 'del-step-only';

export const onDeleteDataMapper = async (
api: IMetadataApi,
vizNode: IVisualizationNode,
modalAnswer: string | undefined,
) => {
const metadataId = DataMapperMetadataService.getDataMapperMetadataId(vizNode);
DataMapperMetadataService.deleteMetadata(api, metadataId);
// TODO DataMapperMetadataService.deleteXsltFile(api, metadataId);
if (modalAnswer === ACTION_ID_DELETE_STEP_AND_FILE) {
await DataMapperMetadataService.deleteXsltFile(api, metadataId);
}
await DataMapperMetadataService.deleteMetadata(api, metadataId);
};
Original file line number Diff line number Diff line change
Expand Up @@ -4,13 +4,13 @@ import { FunctionComponent, PropsWithChildren, useCallback, useContext } from 'r
import { IDataTestID } from '../../../../models';
import { IVisualizationNode } from '../../../../models/visualization/base-visual-entity';
import {
ACTION_ID_CONFIRM,
ACTION_ID_CANCEL,
ActionConfirmationModalContext,
} from '../../../../providers/action-confirmation-modal.provider';
import { EntitiesContext } from '../../../../providers/entities.provider';
import { NodeInteractionAddonContext } from '../../../registers/interactions/node-interaction-addon.provider';
import { IInteractionAddonType } from '../../../registers/interactions/node-interaction-addon.model';
import { processNodeInteractionAddonRecursively } from './item-delete-helper';
import { findModalCustomizationRecursively, processNodeInteractionAddonRecursively } from './item-delete-helper';

interface ItemDeleteGroupProps extends PropsWithChildren<IDataTestID> {
vizNode: IVisualizationNode;
Expand All @@ -24,15 +24,22 @@ export const ItemDeleteGroup: FunctionComponent<ItemDeleteGroupProps> = (props)
const { getRegisteredInteractionAddons } = useContext(NodeInteractionAddonContext);

const onRemoveGroup = useCallback(async () => {
const modalCustoms = findModalCustomizationRecursively(props.vizNode, (vn) =>
getRegisteredInteractionAddons(IInteractionAddonType.ON_DELETE, vn),
);
const additionalModalText = modalCustoms.length > 0 ? modalCustoms[0].additionalText : undefined;
const buttonOptions = modalCustoms.length > 0 ? modalCustoms[0].buttonOptions : undefined;
/** Open delete confirm modal, get the confirmation */
const isDeleteConfirmed = await deleteModalContext?.actionConfirmation({
const modalAnswer = await deleteModalContext?.actionConfirmation({
title: 'Permanently delete flow?',
text: 'All steps will be lost.',
additionalModalText,
buttonOptions,
});

if (isDeleteConfirmed !== ACTION_ID_CONFIRM) return;
if (!modalAnswer || modalAnswer === ACTION_ID_CANCEL) return;

processNodeInteractionAddonRecursively(props.vizNode, (vn) =>
processNodeInteractionAddonRecursively(props.vizNode, modalAnswer, (vn) =>
getRegisteredInteractionAddons(IInteractionAddonType.ON_DELETE, vn),
);

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -5,12 +5,13 @@ import { IDataTestID } from '../../../../models';
import { IVisualizationNode } from '../../../../models/visualization/base-visual-entity';
import { EntitiesContext } from '../../../../providers/entities.provider';
import {
ACTION_ID_CANCEL,
ACTION_ID_CONFIRM,
ActionConfirmationModalContext,
} from '../../../../providers/action-confirmation-modal.provider';
import { NodeInteractionAddonContext } from '../../../registers/interactions/node-interaction-addon.provider';
import { IInteractionAddonType } from '../../../registers/interactions/node-interaction-addon.model';
import { processNodeInteractionAddonRecursively } from './item-delete-helper';
import { findModalCustomizationRecursively, processNodeInteractionAddonRecursively } from './item-delete-helper';

interface ItemDeleteStepProps extends PropsWithChildren<IDataTestID> {
vizNode: IVisualizationNode;
Expand All @@ -23,17 +24,26 @@ export const ItemDeleteStep: FunctionComponent<ItemDeleteStepProps> = (props) =>
const { getRegisteredInteractionAddons } = useContext(NodeInteractionAddonContext);

const onRemoveNode = useCallback(async () => {
if (props.loadActionConfirmationModal) {
const modalCustoms = findModalCustomizationRecursively(props.vizNode, (vn) =>
getRegisteredInteractionAddons(IInteractionAddonType.ON_DELETE, vn),
);

let modalAnswer: string | undefined = ACTION_ID_CONFIRM;
if (props.loadActionConfirmationModal || modalCustoms.length > 0) {
const additionalModalText = modalCustoms.length > 0 ? modalCustoms[0].additionalText : undefined;
const buttonOptions = modalCustoms.length > 0 ? modalCustoms[0].buttonOptions : undefined;
/** Open delete confirm modal, get the confirmation */
const isDeleteConfirmed = await deleteModalContext?.actionConfirmation({
modalAnswer = await deleteModalContext?.actionConfirmation({
title: 'Permanently delete step?',
text: 'Step and its children will be lost.',
additionalModalText,
buttonOptions,
});

if (isDeleteConfirmed !== ACTION_ID_CONFIRM) return;
if (!modalAnswer || modalAnswer === ACTION_ID_CANCEL) return;
}

processNodeInteractionAddonRecursively(props.vizNode, (vn) =>
processNodeInteractionAddonRecursively(props.vizNode, modalAnswer, (vn) =>
getRegisteredInteractionAddons(IInteractionAddonType.ON_DELETE, vn),
);

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -6,12 +6,13 @@ import { AddStepMode, IVisualizationNode } from '../../../../models/visualizatio
import { CatalogModalContext } from '../../../../providers/catalog-modal.provider';
import { EntitiesContext } from '../../../../providers/entities.provider';
import {
ACTION_ID_CANCEL,
ACTION_ID_CONFIRM,
ActionConfirmationModalContext,
} from '../../../../providers/action-confirmation-modal.provider';
import { NodeInteractionAddonContext } from '../../../registers/interactions/node-interaction-addon.provider';
import { IInteractionAddonType } from '../../../registers/interactions/node-interaction-addon.model';
import { processNodeInteractionAddonRecursively } from './item-delete-helper';
import { findModalCustomizationRecursively, processNodeInteractionAddonRecursively } from './item-delete-helper';

interface ItemReplaceStepProps extends PropsWithChildren<IDataTestID> {
vizNode: IVisualizationNode;
Expand All @@ -27,14 +28,22 @@ export const ItemReplaceStep: FunctionComponent<ItemReplaceStepProps> = (props)
const onReplaceNode = useCallback(async () => {
if (!props.vizNode || !entitiesContext) return;

if (props.loadActionConfirmationModal) {
const modalCustoms = findModalCustomizationRecursively(props.vizNode, (vn) =>
getRegisteredInteractionAddons(IInteractionAddonType.ON_DELETE, vn),
);
let modalAnswer: string | undefined = ACTION_ID_CONFIRM;
if (props.loadActionConfirmationModal || modalCustoms.length > 0) {
const additionalModalText = modalCustoms.length > 0 ? modalCustoms[0].additionalText : undefined;
const buttonOptions = modalCustoms.length > 0 ? modalCustoms[0].buttonOptions : undefined;
/** Open delete confirm modal, get the confirmation */
const isReplaceConfirmed = await replaceModalContext?.actionConfirmation({
modalAnswer = await replaceModalContext?.actionConfirmation({
title: 'Replace step?',
text: 'Step and its children will be lost.',
additionalModalText,
buttonOptions,
});

if (isReplaceConfirmed !== ACTION_ID_CONFIRM) return;
if (!modalAnswer || modalAnswer === ACTION_ID_CANCEL) return;
}

/** Find compatible components */
Expand All @@ -47,7 +56,7 @@ export const ItemReplaceStep: FunctionComponent<ItemReplaceStepProps> = (props)
const definedComponent = await catalogModalContext?.getNewComponent(catalogFilter);
if (!definedComponent) return;

processNodeInteractionAddonRecursively(props.vizNode, (vn) =>
processNodeInteractionAddonRecursively(props.vizNode, modalAnswer, (vn) =>
getRegisteredInteractionAddons(IInteractionAddonType.ON_DELETE, vn),
);

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@ import {
IInteractionAddonType,
IRegisteredInteractionAddon,
} from '../../../registers/interactions/node-interaction-addon.model';
import { ACTION_ID_CONFIRM } from '../../../../providers';

describe('item-delete-helper', () => {
describe('processNodeInteractionAddonRecursively', () => {
Expand All @@ -18,7 +19,7 @@ describe('item-delete-helper', () => {
};
addons[childVn.id] = [mockAddon];
vizNode.addChild(childVn);
processNodeInteractionAddonRecursively(vizNode, (vn) => addons[vn.id] ?? []);
processNodeInteractionAddonRecursively(vizNode, ACTION_ID_CONFIRM, (vn) => addons[vn.id] ?? []);
expect(mockAddon.callback).toHaveBeenCalled();
});
});
Expand Down
Original file line number Diff line number Diff line change
@@ -1,14 +1,39 @@
import { IVisualizationNode } from '../../../../models';
import { IRegisteredInteractionAddon } from '../../../registers/interactions/node-interaction-addon.model';
import {
IModalCustomization,
IRegisteredInteractionAddon,
} from '../../../registers/interactions/node-interaction-addon.model';

export const processNodeInteractionAddonRecursively = (
parentVizNode: IVisualizationNode,
modalAnswer: string | undefined,
getAddons: (vizNode: IVisualizationNode) => IRegisteredInteractionAddon[],
) => {
parentVizNode.getChildren()?.forEach((child) => {
processNodeInteractionAddonRecursively(child, getAddons);
processNodeInteractionAddonRecursively(child, modalAnswer, getAddons);
});
getAddons(parentVizNode).forEach((addon) => {
addon.callback(parentVizNode);
addon.callback(parentVizNode, modalAnswer);
});
};

export const findModalCustomizationRecursively = (
parentVizNode: IVisualizationNode,
getAddons: (vizNode: IVisualizationNode) => IRegisteredInteractionAddon[],
) => {
const modalCustomizations: IModalCustomization[] = [];
// going breadth-first while addon processes depth-first... do we want?
getAddons(parentVizNode).forEach((addon) => {
if (addon.modalCustomization && !modalCustomizations.includes(addon.modalCustomization)) {
modalCustomizations.push(addon.modalCustomization);
}
});
parentVizNode.getChildren()?.forEach((child) => {
findModalCustomizationRecursively(child, getAddons).forEach((custom) => {
if (!modalCustomizations.includes(custom)) {
modalCustomizations.push(custom);
}
});
});
return modalCustomizations;
};
Original file line number Diff line number Diff line change
@@ -1,9 +1,14 @@
import { FunctionComponent, PropsWithChildren, useContext, useRef } from 'react';
import { datamapperActivationFn } from './datamapper.activationfn';
import { MetadataContext } from '../../providers';
import { onDeleteDataMapper } from '../DataMapper/on-delete-datamapper';
import {
ACTION_ID_DELETE_STEP_AND_FILE,
ACTION_ID_DELETE_STEP_ONLY,
onDeleteDataMapper,
} from '../DataMapper/on-delete-datamapper';
import { NodeInteractionAddonContext } from './interactions/node-interaction-addon.provider';
import { IInteractionAddonType, IRegisteredInteractionAddon } from './interactions/node-interaction-addon.model';
import { ButtonVariant } from '@patternfly/react-core';

export const RegisterNodeInteractionAddons: FunctionComponent<PropsWithChildren> = ({ children }) => {
const metadataApi = useContext(MetadataContext)!;
Expand All @@ -12,8 +17,22 @@ export const RegisterNodeInteractionAddons: FunctionComponent<PropsWithChildren>
{
type: IInteractionAddonType.ON_DELETE,
activationFn: datamapperActivationFn,
callback: (vizNode) => {
metadataApi && onDeleteDataMapper(metadataApi, vizNode);
callback: (vizNode, modalAnswer) => {
metadataApi && onDeleteDataMapper(metadataApi, vizNode, modalAnswer);
},
modalCustomization: {
additionalText: 'Do you also want to delete the associated Kaoto DataMapper mapping file (XSLT)?',
buttonOptions: {
[ACTION_ID_DELETE_STEP_AND_FILE]: {
variant: ButtonVariant.danger,
buttonText: 'Delete both step and file',
},
[ACTION_ID_DELETE_STEP_ONLY]: {
variant: ButtonVariant.secondary,
isDanger: true,
buttonText: 'Delete the step, but keep the file',
},
},
},
},
]);
Expand Down
Original file line number Diff line number Diff line change
@@ -1,13 +1,20 @@
import { IVisualizationNode } from '../../../models';
import { ActionConfirmationButtonOption } from '../../../providers';

export enum IInteractionAddonType {
ON_DELETE = 'onDelete',
}

export interface IModalCustomization {
buttonOptions: Record<string, ActionConfirmationButtonOption>;
additionalText?: string;
}

export interface IRegisteredInteractionAddon {
type: IInteractionAddonType;
activationFn: (vizNode: IVisualizationNode) => boolean;
callback: (vizNode: IVisualizationNode) => void;
callback: (vizNode: IVisualizationNode, modalAnswer: string | undefined) => void;
modalCustomization?: IModalCustomization;
}

export interface INodeInteractionAddonContext {
Expand Down Expand Up @@ -38,7 +45,7 @@ export interface INodeInteractionAddonContext {
*
* const addons = nodeInteractionAddonContext.getRegisteredInteractionAddons(IInteractionAddonType.ON_DELETE, vizNode);
* addons.forEach((addon) => {
* addon.onDelete(vizNode);
* addon.callback(vizNode, ACTION_ID_CONFIRM);
* });
* ```
* @param type The interaction addon type enum value
Expand Down
18 changes: 16 additions & 2 deletions packages/ui/src/multiplying-architecture/KaotoBridge.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -81,6 +81,12 @@ interface KaotoBridgeProps {
*/
saveResourceContent(path: string, content: string): Promise<void>;

/**
* Delete resource using the channel API.
* @param path The path of the resource
*/
deleteResource(path: string): Promise<boolean>;

/**
* Show a Quick Pick widget and ask the user to select one or more files available in the workspace.
* @param include The filter expression for the files to include
Expand Down Expand Up @@ -109,6 +115,7 @@ export const KaotoBridge = forwardRef<EditorApi, PropsWithChildren<KaotoBridgePr
setMetadata,
getResourceContent,
saveResourceContent,
deleteResource,
askUserForFileSelection,
},
forwardedRef,
Expand All @@ -120,8 +127,15 @@ export const KaotoBridge = forwardRef<EditorApi, PropsWithChildren<KaotoBridgePr
const settingsAdapter = useContext(SettingsContext);
const catalogUrl = settingsAdapter.getSettings().catalogUrl;
const metadataApi = useMemo(
() => ({ getMetadata, setMetadata, getResourceContent, saveResourceContent, askUserForFileSelection }),
[getMetadata, setMetadata, getResourceContent, saveResourceContent, askUserForFileSelection],
() => ({
getMetadata,
setMetadata,
getResourceContent,
saveResourceContent,
deleteResource,
askUserForFileSelection,
}),
[getMetadata, setMetadata, getResourceContent, saveResourceContent, deleteResource, askUserForFileSelection],
);

/**
Expand Down
6 changes: 6 additions & 0 deletions packages/ui/src/multiplying-architecture/KaotoEditorApp.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -40,6 +40,7 @@ export class KaotoEditorApp implements Editor {
this.setMetadata = this.setMetadata.bind(this);
this.getResourceContent = this.getResourceContent.bind(this);
this.saveResourceContent = this.saveResourceContent.bind(this);
this.deleteResource = this.deleteResource.bind(this);
this.askUserForFileSelection = this.askUserForFileSelection.bind(this);
}

Expand Down Expand Up @@ -115,6 +116,10 @@ export class KaotoEditorApp implements Editor {
return this.envelopeContext.channelApi.requests.saveResourceContent(path, content);
}

async deleteResource(path: string): Promise<boolean> {
return this.envelopeContext.channelApi.requests.deleteResource(path);
}

async askUserForFileSelection(
include: string,
exclude?: string,
Expand All @@ -139,6 +144,7 @@ export class KaotoEditorApp implements Editor {
setMetadata={this.setMetadata}
getResourceContent={this.getResourceContent}
saveResourceContent={this.saveResourceContent}
deleteResource={this.deleteResource}
askUserForFileSelection={this.askUserForFileSelection}
>
<RouterProvider router={kaotoEditorRouter} />
Expand Down
6 changes: 6 additions & 0 deletions packages/ui/src/providers/metadata.provider.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -27,6 +27,12 @@ export interface IMetadataApi {
*/
saveResourceContent(path: string, content: string): Promise<void>;

/**
* Delete resource
* @param path The path of the resource
*/
deleteResource(path: string): Promise<boolean>;

/**
* Show a Quick Pick widget and ask the user to select one or more files available in the workspace.
* @param include The filter expression for the files to include
Expand Down
9 changes: 7 additions & 2 deletions packages/ui/src/services/datamapper-metadata.service.ts
Original file line number Diff line number Diff line change
Expand Up @@ -205,7 +205,12 @@ export class DataMapperMetadataService {
});
}

static deleteMetadata(api: IMetadataApi, metadataId: string) {
api.setMetadata(metadataId, undefined);
static async deleteMetadata(api: IMetadataApi, metadataId: string) {
await api.setMetadata(metadataId, undefined);
}

static async deleteXsltFile(api: IMetadataApi, metadataId: string) {
const metadata = (await api.getMetadata(metadataId)) as IDataMapperMetadata;
await api.deleteResource(metadata.xsltPath);
}
}

0 comments on commit 7f20716

Please sign in to comment.