diff --git a/client/public/locales/en/translation.json b/client/public/locales/en/translation.json index c64b5bc8af..fc6872bb39 100644 --- a/client/public/locales/en/translation.json +++ b/client/public/locales/en/translation.json @@ -71,7 +71,8 @@ "orange": "Orange", "purple": "Purple", "red": "Red", - "teal": "Teal" + "teal": "Teal", + "yellow": "Yellow" }, "composed": { "add": "Add {{what}}", @@ -243,7 +244,6 @@ "businessServices": "Business services", "canceled": "Canceled", "category": "Category", - "cloudNativeReady": "Cloud-native ready", "color": "Color", "comments": "Comments", "completed": "Completed", @@ -311,7 +311,6 @@ "member(s)": "Member(s)", "memberCount": "Member count", "message": "Message", - "modernizable": "Modernizable", "migrationWave": "Migration wave", "migrationWaves": "Migration waves", "migrationWave(s)": "Migration wave(s)", @@ -322,7 +321,6 @@ "notConnected": "Not connected", "notReviewed": "Not reviewed", "notStarted": "Not started", - "notYetAssessed": "Not yet assessed", "notYetReviewed": "Not yet reviewed", "other": "Other", "owner": "Owner", @@ -384,7 +382,6 @@ "unassessed": "Unassessed", "unassigned": "Not yet assigned", "unknown": "Unknown", - "unsuitableForContainers": "Unsuitable for containers", "uploadApplicationFile": "Upload your application file", "uploadYamlFile": "Upload your YAML file", "url": "URL", diff --git a/client/public/locales/es/translation.json b/client/public/locales/es/translation.json index 5bfde9d167..12f96255f9 100644 --- a/client/public/locales/es/translation.json +++ b/client/public/locales/es/translation.json @@ -55,7 +55,8 @@ "green": "Verde", "orange": "Naranja", "purple": "Morado", - "red": "Rojo" + "red": "Rojo", + "yellow": "Amarillo" }, "composed": { "add": "Agregar {{what}}", @@ -194,7 +195,6 @@ "businessServices": "Servicios de negocio", "canceled": "Cancelado", "category": "Categoría", - "cloudNativeReady": "Listo para Cloud-native", "color": "Color", "comments": "Comentarios", "completed": "Completado", @@ -247,13 +247,11 @@ "member(s)": "Miembro(s)", "memberCount": "Número de miembros", "message": "Mensaje", - "modernizable": "Modernizable", "name": "Nombre", "northboundDependencies": "Dependencias del norte", "notAvailable": "No disponible", "notReviewed": "No revisado", "notStarted": "No empezado", - "notYetAssessed": "No evaluado todavía", "notYetReviewed": "No revisado todavía", "other": "Otro", "owner": "Propietario", @@ -304,7 +302,6 @@ "trivialButMigratable": "Trivial pero migrable", "unassessed": "No evaluado", "unknown": "Desconocido", - "unsuitableForContainers": "No apto para contenedores", "uploadApplicationFile": "Carga tu archivo con aplicaciones", "user": "Usuario", "version": "Versión", diff --git a/client/src/app/api/models.ts b/client/src/app/api/models.ts index d4ad356db9..968e9b678a 100644 --- a/client/src/app/api/models.ts +++ b/client/src/app/api/models.ts @@ -709,8 +709,8 @@ export interface Assessment description: string; status: AssessmentStatus; risk: Risk; - stakeholders: Ref[]; - stakeholderGroups: Ref[]; + stakeholders?: Ref[]; + stakeholderGroups?: Ref[]; } export interface CategorizedTag { category: TagCategory; @@ -723,11 +723,6 @@ export interface AssessmentRisk { applicationId: number; risk: Risk; } -export interface AssessmentRisk { - assessmentId: number; - applicationId: number; - risk: Risk; -} export interface AssessmentQuestionRisk { category: string; diff --git a/client/src/app/api/rest.ts b/client/src/app/api/rest.ts index 18e803352f..e9b06be9e9 100644 --- a/client/src/app/api/rest.ts +++ b/client/src/app/api/rest.ts @@ -205,6 +205,9 @@ export const getApplicationSummaryCSV = (id: string): AxiosPromise => { }); }; +export const getAssessments = (): Promise => + axios.get(ASSESSMENTS).then((response) => response.data); + export const getAssessmentsByItemId = ( isArchetype: boolean, itemId?: number | string diff --git a/client/src/app/pages/applications/applications-table-assessment/applications-table-assessment.tsx b/client/src/app/pages/applications/applications-table-assessment/applications-table-assessment.tsx index 5d82ba8841..c9bfc6cbf7 100644 --- a/client/src/app/pages/applications/applications-table-assessment/applications-table-assessment.tsx +++ b/client/src/app/pages/applications/applications-table-assessment/applications-table-assessment.tsx @@ -220,8 +220,8 @@ export const ApplicationsTable: React.FC = () => { await Promise.all( application.assessments.map(async (assessment) => { await deleteAssessment({ - id: assessment.id, - name: application.name, + assessmentId: assessment.id, + applicationName: application.name, }); }) ); diff --git a/client/src/app/pages/assessment-management/import-questionnaire-form/import-questionnaire-form.tsx b/client/src/app/pages/assessment-management/import-questionnaire-form/import-questionnaire-form.tsx index abb3b72634..4e4e58f677 100644 --- a/client/src/app/pages/assessment-management/import-questionnaire-form/import-questionnaire-form.tsx +++ b/client/src/app/pages/assessment-management/import-questionnaire-form/import-questionnaire-form.tsx @@ -56,12 +56,13 @@ export const ImportQuestionnaireForm: React.FC< trigger, } = methods; - const onHandleSuccessfullQuestionnaireCreation = ( - response: Questionnaire - ) => { + const onHandleSuccessfulQuestionnaireCreation = (response: Questionnaire) => { onSaved(response); pushNotification({ - title: t("toastr.success.questionnaireCreated"), + title: t("toastr.success.createWhat", { + type: t("terms.questionnaire"), + what: response.name, + }), variant: "success", }); onSaved(); @@ -75,7 +76,7 @@ export const ImportQuestionnaireForm: React.FC< }; const { mutate: createQuestionnaire } = useCreateQuestionnaireMutation( - onHandleSuccessfullQuestionnaireCreation, + onHandleSuccessfulQuestionnaireCreation, onHandleFailedQuestionnaireCreation ); diff --git a/client/src/app/pages/assessment/components/assessment-actions/components/assessment-actions-table.tsx b/client/src/app/pages/assessment/components/assessment-actions/components/assessment-actions-table.tsx index d60261190d..eb40ab7f79 100644 --- a/client/src/app/pages/assessment/components/assessment-actions/components/assessment-actions-table.tsx +++ b/client/src/app/pages/assessment/components/assessment-actions/components/assessment-actions-table.tsx @@ -38,6 +38,7 @@ const AssessmentActionsTable: React.FC = ({ { createAssessment(); }); @@ -257,8 +259,10 @@ const DynamicAssessmentActionsRow: FunctionComponent< variant="plain" onClick={() => { deleteAssessment({ - id: assessment.id, - name: assessment.name, + assessmentId: assessment.id, + applicationName: application?.name, + applicationId: application?.id, + archetypeId: archetype?.id, }); }} > diff --git a/client/src/app/pages/assessment/components/assessment-wizard/assessment-wizard.tsx b/client/src/app/pages/assessment/components/assessment-wizard/assessment-wizard.tsx index 2c9c2a8589..c47806ab2e 100644 --- a/client/src/app/pages/assessment/components/assessment-wizard/assessment-wizard.tsx +++ b/client/src/app/pages/assessment/components/assessment-wizard/assessment-wizard.tsx @@ -10,6 +10,7 @@ import { Assessment, AssessmentStatus, Question, + Ref, Section, } from "@app/api/models"; import { CustomWizardFooter } from "../custom-wizard-footer"; @@ -26,6 +27,7 @@ import { import { AxiosError } from "axios"; import { assessmentsByItemIdQueryKey, + useDeleteAssessmentMutation, useUpdateAssessmentMutation, } from "@app/queries/assessments"; import { useQueryClient } from "@tanstack/react-query"; @@ -34,6 +36,8 @@ import { Paths } from "@app/Paths"; import { yupResolver } from "@hookform/resolvers/yup"; import { AssessmentStakeholdersForm } from "../assessment-stakeholders-form/assessment-stakeholders-form"; import useIsArchetype from "@app/hooks/useIsArchetype"; +import { useFetchStakeholderGroups } from "@app/queries/stakeholdergoups"; +import { useFetchStakeholders } from "@app/queries/stakeholders"; export const SAVE_ACTION_KEY = "saveAction"; @@ -66,6 +70,8 @@ export const AssessmentWizard: React.FC = ({ }) => { const isArchetype = useIsArchetype(); const queryClient = useQueryClient(); + const { stakeholderGroups } = useFetchStakeholderGroups(); + const { stakeholders } = useFetchStakeholders(); const onHandleUpdateAssessmentSuccess = () => { queryClient.invalidateQueries([ @@ -77,12 +83,14 @@ export const AssessmentWizard: React.FC = ({ onHandleUpdateAssessmentSuccess ); + const { mutate: deleteAssessmentMutation } = useDeleteAssessmentMutation(); + const { t } = useTranslation(); const [currentStep, setCurrentStep] = useState(0); - const [isConfirmDialogOpen, setIsConfirmDialogOpen] = - React.useState(false); + const [assessmentToCancel, setAssessmentToCancel] = + React.useState(null); const history = useHistory(); @@ -124,13 +132,11 @@ export const AssessmentWizard: React.FC = ({ }, [assessment]); useEffect(() => { + console.log("asssessment.stakeholders", assessment?.stakeholders); methods.reset({ - stakeholders: - assessment?.stakeholders.map((stakeholder) => stakeholder.name) || [], + stakeholders: assessment?.stakeholders?.map((sh) => sh.name).sort() ?? [], stakeholderGroups: - assessment?.stakeholderGroups.map( - (stakeholderGroup) => stakeholderGroup.name - ) || [], + assessment?.stakeholderGroups?.map((sg) => sg.name).sort() ?? [], // comments: initialComments, questions: initialQuestions, [SAVE_ACTION_KEY]: SAVE_ACTION_VALUE.SAVE_AS_DRAFT, @@ -146,11 +152,9 @@ export const AssessmentWizard: React.FC = ({ defaultValues: useMemo(() => { return { stakeholders: - assessment?.stakeholders.map((stakeholder) => stakeholder.name) || [], + assessment?.stakeholders?.map((sh) => sh.name).sort() ?? [], stakeholderGroups: - assessment?.stakeholderGroups.map( - (stakeholderGroup) => stakeholderGroup.name - ) || [], + assessment?.stakeholderGroups?.map((sg) => sg.name).sort() ?? [], // comments: initialComments, questions: initialQuestions, [SAVE_ACTION_KEY]: SAVE_ACTION_VALUE.SAVE_AS_DRAFT, @@ -273,6 +277,25 @@ export const AssessmentWizard: React.FC = ({ const assessmentStatus: AssessmentStatus = "started"; const payload: Assessment = { ...assessment, + stakeholders: + values.stakeholders === undefined + ? undefined + : (values.stakeholders + .map((name) => stakeholders.find((s) => s.name === name)) + .map((sh) => + !sh ? undefined : { id: sh.id, name: sh.name } + ) + .filter(Boolean) as Ref[]), + + stakeholderGroups: + values.stakeholderGroups === undefined + ? undefined + : (values.stakeholderGroups + .map((name) => stakeholderGroups.find((s) => s.name === name)) + .map((sg) => + !sg ? undefined : { id: sg.id, name: sg.name } + ) + .filter(Boolean) as Ref[]), sections, status: assessmentStatus, }; @@ -317,6 +340,25 @@ export const AssessmentWizard: React.FC = ({ const payload: Assessment = { ...assessment, + stakeholders: + values.stakeholders === undefined + ? undefined + : (values.stakeholders + .map((name) => stakeholders.find((s) => s.name === name)) + .map((sh) => + !sh ? undefined : { id: sh.id, name: sh.name } + ) + .filter(Boolean) as Ref[]), + + stakeholderGroups: + values.stakeholderGroups === undefined + ? undefined + : (values.stakeholderGroups + .map((name) => stakeholderGroups.find((s) => s.name === name)) + .map((sg) => + !sg ? undefined : { id: sg.id, name: sg.name } + ) + .filter(Boolean) as Ref[]), sections, status: assessmentStatus, }; @@ -364,6 +406,25 @@ export const AssessmentWizard: React.FC = ({ const payload: Assessment = { ...assessment, + stakeholders: + values.stakeholders === undefined + ? undefined + : (values.stakeholders + .map((name) => stakeholders.find((s) => s.name === name)) + .map((sh) => + !sh ? undefined : { id: sh.id, name: sh.name } + ) + .filter(Boolean) as Ref[]), + + stakeholderGroups: + values.stakeholderGroups === undefined + ? undefined + : (values.stakeholderGroups + .map((name) => stakeholderGroups.find((s) => s.name === name)) + .map((sg) => + !sg ? undefined : { id: sg.id, name: sg.name } + ) + .filter(Boolean) as Ref[]), sections, status: assessmentStatus, }; @@ -509,11 +570,11 @@ export const AssessmentWizard: React.FC = ({ onNext={() => setCurrentStep((current) => current + 1)} onBack={() => setCurrentStep((current) => current - 1)} onClose={() => { - setIsConfirmDialogOpen(true); + assessment && setAssessmentToCancel(assessment); }} onGoToStep={(step) => setCurrentStep(step.id as number)} /> - {isConfirmDialogOpen && ( + {assessmentToCancel && ( = ({ confirmBtnVariant={ButtonVariant.primary} confirmBtnLabel={t("actions.continue")} cancelBtnLabel={t("actions.cancel")} - onCancel={() => setIsConfirmDialogOpen(false)} - onClose={() => setIsConfirmDialogOpen(false)} + onCancel={() => setAssessmentToCancel(null)} + onClose={() => setAssessmentToCancel(null)} onConfirm={() => { - setIsConfirmDialogOpen(false); if (isArchetype) { + assessmentToCancel.status === "empty" && + deleteAssessmentMutation({ + assessmentId: assessmentToCancel.id, + applicationName: assessmentToCancel.application?.name, + applicationId: assessmentToCancel.application?.id, + archetypeId: assessmentToCancel.archetype?.id, + }); + history.push( formatPath(Paths.archetypeAssessmentActions, { archetypeId: assessment?.archetype?.id, }) ); } else { + assessmentToCancel.status === "empty" && + deleteAssessmentMutation({ + assessmentId: assessmentToCancel.id, + applicationName: assessmentToCancel.application?.name, + applicationId: assessmentToCancel.application?.id, + archetypeId: assessmentToCancel.archetype?.id, + }); history.push( formatPath(Paths.applicationAssessmentActions, { applicationId: assessment?.application?.id, }) ); } + setAssessmentToCancel(null); }} /> )} diff --git a/client/src/app/pages/assessment/components/questionnaire-form/multi-input-selection/multi-input-selection.tsx b/client/src/app/pages/assessment/components/questionnaire-form/multi-input-selection/multi-input-selection.tsx index 5a74d2d966..d2b08203d3 100644 --- a/client/src/app/pages/assessment/components/questionnaire-form/multi-input-selection/multi-input-selection.tsx +++ b/client/src/app/pages/assessment/components/questionnaire-form/multi-input-selection/multi-input-selection.tsx @@ -27,18 +27,17 @@ export const MultiInputSelection: React.FC = ({ const isArchetype = useIsArchetype(); const { t } = useTranslation(); - return ( {sortedOptions.map((option, i) => ( - + ( { diff --git a/client/src/app/pages/assessment/components/questionnaire-form/questionnaire-form.tsx b/client/src/app/pages/assessment/components/questionnaire-form/questionnaire-form.tsx index f352741111..4604f4ddf1 100644 --- a/client/src/app/pages/assessment/components/questionnaire-form/questionnaire-form.tsx +++ b/client/src/app/pages/assessment/components/questionnaire-form/questionnaire-form.tsx @@ -12,7 +12,7 @@ import { import HelpIcon from "@patternfly/react-icons/dist/esm/icons/help-icon"; import { MultiInputSelection } from "./multi-input-selection"; import { Question, QuestionHeader, QuestionBody } from "./question"; -import { getCommentFieldName } from "../../form-utils"; +import { getCommentFieldName, getQuestionFieldName } from "../../form-utils"; import { HookFormPFTextInput } from "@app/components/HookFormPFFields"; import { useFormContext } from "react-hook-form"; import { Section } from "@app/api/models"; @@ -54,7 +54,7 @@ export const QuestionnaireForm: React.FC = ({ {sortedQuestions.map((question) => ( - + @@ -74,7 +74,10 @@ export const QuestionnaireForm: React.FC = ({ - + diff --git a/client/src/app/pages/assessment/form-utils.test.ts b/client/src/app/pages/assessment/form-utils.test.ts index dbcaf02921..2e6d74a649 100644 --- a/client/src/app/pages/assessment/form-utils.test.ts +++ b/client/src/app/pages/assessment/form-utils.test.ts @@ -26,11 +26,11 @@ describe("Application assessment - form utils", () => { it("getQuestionFieldName: fullName", () => { const fieldName = getQuestionFieldName(question, true); - expect(fieldName).toBe("questions.question-Question 321"); + expect(fieldName).toBe("questions.question-1"); }); it("getQuestionFieldName: singleName", () => { const fieldName = getQuestionFieldName(question, false); - expect(fieldName).toBe("question-Question 321"); + expect(fieldName).toBe("question-1"); }); }); diff --git a/client/src/app/pages/assessment/form-utils.ts b/client/src/app/pages/assessment/form-utils.ts index 1f2f54deb5..43fcb26f5d 100644 --- a/client/src/app/pages/assessment/form-utils.ts +++ b/client/src/app/pages/assessment/form-utils.ts @@ -15,6 +15,6 @@ export const getCommentFieldName = (section: Section, fullName: boolean) => { }; export const getQuestionFieldName = (question: Question, fullName: boolean) => { - const fieldName = `question-${question.text}`; + const fieldName = `question-${question.order}`; return fullName ? `${QUESTIONS_KEY}.${fieldName}` : fieldName; }; diff --git a/client/src/app/pages/reports/components/landscape/donut.tsx b/client/src/app/pages/reports/components/landscape/donut.tsx index b3612229b1..5c0f4bacce 100644 --- a/client/src/app/pages/reports/components/landscape/donut.tsx +++ b/client/src/app/pages/reports/components/landscape/donut.tsx @@ -11,7 +11,7 @@ export interface IDonutProps { total: number; color: string; riskLabel: string; - riskDescription: string; + riskDescription?: string; } export const Donut: React.FC = ({ diff --git a/client/src/app/pages/reports/components/landscape/landscape.tsx b/client/src/app/pages/reports/components/landscape/landscape.tsx index c21b229fda..7c500cd922 100644 --- a/client/src/app/pages/reports/components/landscape/landscape.tsx +++ b/client/src/app/pages/reports/components/landscape/landscape.tsx @@ -7,7 +7,7 @@ import { ConditionalRender } from "@app/components/ConditionalRender"; import { StateError } from "@app/components/StateError"; import { RISK_LIST } from "@app/Constants"; -import { AssessmentRisk } from "@app/api/models"; +import { Assessment, AssessmentRisk } from "@app/api/models"; import { ApplicationSelectionContext } from "../../application-selection-context"; import { NoApplicationSelectedEmptyState } from "../no-application-selected-empty-state"; @@ -48,7 +48,11 @@ const extractLandscapeData = ( return { low, medium, high, unassessed }; }; -export const Landscape: React.FC = () => { +interface ILandscapeProps { + assessments: Assessment[]; +} + +export const Landscape: React.FC = ({ assessments }) => { const { t } = useTranslation(); // Context @@ -89,11 +93,11 @@ export const Landscape: React.FC = () => { @@ -101,17 +105,17 @@ export const Landscape: React.FC = () => { value={landscapeData.medium} total={applications.length} color={RISK_LIST["yellow"].hexColor} - riskLabel={t("terms.mediumRisk")} - riskDescription={t("terms.modernizable")} + riskLabel={t("colors.yellow")} + // riskDescription={} /> @@ -119,8 +123,8 @@ export const Landscape: React.FC = () => { value={landscapeData.unassessed} total={applications.length} color={RISK_LIST["unknown"].hexColor} - riskLabel={t("terms.unassessed")} - riskDescription={t("terms.notYetAssessed")} + riskLabel={`${t("terms.unassessed")}/${t("terms.unknown")}`} + // riskDescription={} /> diff --git a/client/src/app/pages/reports/reports.tsx b/client/src/app/pages/reports/reports.tsx index 5b5aa4b2ea..3a085a639d 100644 --- a/client/src/app/pages/reports/reports.tsx +++ b/client/src/app/pages/reports/reports.tsx @@ -9,9 +9,15 @@ import { CardExpandableContent, CardHeader, CardTitle, + Flex, + FlexItem, + MenuToggle, + MenuToggleElement, PageSection, PageSectionVariants, Popover, + Select, + SelectOption, Split, SplitItem, Stack, @@ -31,11 +37,27 @@ import { Landscape } from "./components/landscape"; import { AdoptionPlan } from "./components/adoption-plan"; import { IdentifiedRisksTable } from "./components/identified-risks-table"; import { useFetchApplications } from "@app/queries/applications"; +import { useFetchAssessments } from "@app/queries/assessments"; +import { Ref } from "@app/api/models"; + +const ALL_QUESTIONNAIRES = "All questionnaires"; export const Reports: React.FC = () => { // i18 const { t } = useTranslation(); + const [isQuestionnaireSelectOpen, setIsQuestionnaireSelectOpen] = + React.useState(false); + + const [selectedQuestionnaire, setSelectedQuestionnaire] = + React.useState("All questionnaires"); + + const { + assessments, + isFetching: isAssessmentsFetching, + fetchError: assessmentsFetchError, + } = useFetchAssessments(); + // Cards const [isAdoptionCandidateTable, setIsAdoptionCandidateTable] = useState(true); @@ -66,6 +88,48 @@ export const Reports: React.FC = () => { ); } + const toggleQuestionnaire = (toggleRef: React.Ref) => ( + { + setIsQuestionnaireSelectOpen(!isQuestionnaireSelectOpen); + }} + isExpanded={isQuestionnaireSelectOpen} + > + {selectedQuestionnaire} + + ); + + const onSelectQuestionnaire = ( + _event: React.MouseEvent | undefined, + value: string | number | undefined + ) => { + setSelectedQuestionnaire(value as string); + setIsQuestionnaireSelectOpen(false); + }; + + const answeredQuestionnaires: Ref[] = + isAssessmentsFetching || assessmentsFetchError + ? [] + : assessments + .reduce((questionnaires: Ref[], assessment) => { + if ( + !questionnaires + .map((ref) => ref.id) + .includes(assessment.questionnaire.id) + ) { + assessment.questionnaire && + questionnaires.push(assessment.questionnaire); + } + return questionnaires; + }, []) + .sort((a, b) => { + if (a.name > b.name) return 1; + if (b.name > a.name) return -1; + return 0; + }); + return ( <> {pageHeaderSection} @@ -78,13 +142,56 @@ export const Reports: React.FC = () => { - - {t("terms.currentLandscape")} - + + + + + {t("terms.currentLandscape")} + + + + + + + - + + assessment.questionnaire.name === + selectedQuestionnaire + ) + } + /> diff --git a/client/src/app/queries/assessments.ts b/client/src/app/queries/assessments.ts index a443d337f1..853b38b88a 100644 --- a/client/src/app/queries/assessments.ts +++ b/client/src/app/queries/assessments.ts @@ -4,17 +4,32 @@ import { createAssessment, deleteAssessment, getAssessmentById, + getAssessments, getAssessmentsByItemId, updateAssessment, } from "@app/api/rest"; import { AxiosError } from "axios"; import { Assessment, InitialAssessment } from "@app/api/models"; import { QuestionnairesQueryKey } from "./questionnaires"; +import { ARCHETYPE_QUERY_KEY } from "./archetypes"; export const assessmentsQueryKey = "assessments"; export const assessmentQueryKey = "assessment"; export const assessmentsByItemIdQueryKey = "assessmentsByItemId"; +export const useFetchAssessments = () => { + const { isLoading, data, error } = useQuery({ + queryKey: [assessmentsQueryKey], + queryFn: getAssessments, + onError: (error: AxiosError) => console.log("error, ", error), + }); + return { + assessments: data || [], + isFetching: isLoading, + fetchError: error, + }; +}; + export const useCreateAssessmentMutation = ( isArchetype: boolean, onSuccess: (name: string) => void, @@ -57,21 +72,35 @@ export const useUpdateAssessmentMutation = ( }; export const useDeleteAssessmentMutation = ( - onSuccess: (name: string) => void, - onError: (err: AxiosError) => void + onSuccess?: (applicationName: string) => void, + onError?: (err: AxiosError) => void ) => { const queryClient = useQueryClient(); return useMutation({ - mutationFn: (args: { name: string; id: number }) => - deleteAssessment(args.id), - onSuccess: (_, args) => { - onSuccess(args.name); + mutationFn: (args: { + assessmentId: number; + applicationName?: string; + applicationId?: number; + archetypeId?: number; + }) => { + const deletedAssessment = deleteAssessment(args.assessmentId); + + queryClient.invalidateQueries([assessmentQueryKey, args?.assessmentId]); + queryClient.invalidateQueries([ARCHETYPE_QUERY_KEY, args?.archetypeId]); queryClient.invalidateQueries([ assessmentsByItemIdQueryKey, - args.id, - QuestionnairesQueryKey, + args?.archetypeId, + ]); + queryClient.invalidateQueries([ + assessmentsByItemIdQueryKey, + args?.applicationId, ]); + + return deletedAssessment; + }, + onSuccess: (_, args) => { + onSuccess && onSuccess(args?.applicationName || "Unknown"); }, onError: onError, }); diff --git a/hack/README.md b/hack/README.md new file mode 100644 index 0000000000..f4820996fe --- /dev/null +++ b/hack/README.md @@ -0,0 +1,17 @@ +# Scripts and samples of import data + +This folder is a place to keep sample data and scripts to help developers with +creating/importing various pieces of data. + +## Adding a base set of data to an empty instance + +For creating a base set of data in a tackle-hub instance, a good place to start +is [tackle-hub's hack folder](https://github.com/konveyor/tackle2-hub/tree/main/hack). +If you have hub running or port forwarded on port :9002, a basic set of data can be +added to the instance by: + +```sh +> git clone https://github.com/konveyor/tackle2-hub.git +> cd tackle2-hub/hack +> HOST=localhost:9002 ./add/all.sh +``` diff --git a/hack/import-applications/sample-applications.csv b/hack/import-applications/sample-applications.csv new file mode 100644 index 0000000000..80f64fdab1 --- /dev/null +++ b/hack/import-applications/sample-applications.csv @@ -0,0 +1,6 @@ +Record Type 1,Application Name,Description,Comments,Business Service,Dependency,Dependency Direction,Binary Group,Binary Artifact,Binary Version,Binary Packaging,Repository Type,Repository URL,Repository Branch,Repository Path,Tag Category 1,Tag 1,Tag Category 2,Tag 2,Tag Category 3,Tag 3,Tag Category 4,Tag 4,Tag Category 5,Tag 5,Tag Category 6,Tag 6,Tag Category 7,Tag 7,Tag Category 8,Tag 8,Tag Category 9,Tag 9,Tag Category 10,Tag 10,Tag Category 11,Tag 11,Tag Category 12,Tag 12,Tag Category 13,Tag 13,Tag Category 14,Tag 14,Tag Category 15,Tag 15,Tag Category 16,Tag 16,Tag Category 17,Tag 17,Tag Category 18,Tag 18,Tag Category 19,Tag 19,Tag Category 20,Tag 20 +1,Customers,Legacy Customers management service,,Retail,,,corp.acme.demo,customers-tomcat,0.0.1-SNAPSHOT,war,git,https://git-acme.local/customers.git,,,Operating System,RHEL 8,Database,Oracle,Language,Java,Runtime,Tomcat,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,, +1,Inventory,Inventory service,,Retail,,,corp.acme.demo,inventory,0.1.1-SNAPSHOT,war,git,https://git-acme.local/inventory.git,,,Operating System,RHEL 8,Database,Postgresql,Language,Java,Runtime,Quarkus,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,, +1,Gateway,API Gateway,,Retail,,,corp.acme.demo,gateway,0.1.1-SNAPSHOT,war,git,https://git-acme.local/gateway.git,,,Operating System,RHEL 8,,,Language,Java,Runtime,Spring Boot,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,, +2,Gateway,,,,Inventory,southbound +2,Gateway,,,,Customers,southbound diff --git a/client/public/questionnaire-data.yaml b/hack/import-questionnaire/cloud-native.yaml similarity index 73% rename from client/public/questionnaire-data.yaml rename to hack/import-questionnaire/cloud-native.yaml index 6af53ee020..b699d8a916 100644 --- a/client/public/questionnaire-data.yaml +++ b/hack/import-questionnaire/cloud-native.yaml @@ -1,75 +1,77 @@ name: Cloud Native description: | Questionnaire that includes the Twelve Factor Application principles. + sections: - - name: Application technologies 1 + - order: 1 + name: Application technologies 1 questions: - - formulation: What is the main technology in your application? + - text: What is the main technology in your application? explanation: What would you describe as the main framework used to build your application. answers: - - choice: Unknown + - text: Unknown rationale: This is a problem because of the uncertainty. mitigation: Gathering more information about this is required. risk: unknown - - choice: Quarkus + - text: Quarkus risk: green - autoanswer_if_tags_present: + autoAnswerFor: - category: Technology tag: Quarkus - autotag: + applyTags: - category: Technology tag: Quarkus - - choice: Spring Boot + - text: Spring Boot risk: green - autoanswer_if_tags_present: + autoAnswerFor: - category: Technology tag: Spring Boot - autotag: + applyTags: - category: Technology tag: Spring Boot - - choice: Java EE + - text: Java EE rationale: This might not be the most cloud friendly technology. mitigation: Maybe start thinking about migrating to Quarkus or Jakarta EE. risk: yellow - autoanswer_if_tags_present: + autoAnswerFor: - category: Technology tag: Java EE - autotag: + applyTags: - category: Technology tag: Java EE - - choice: J2EE + - text: J2EE rationale: This is obsolete. mitigation: Maybe start thinking about migrating to Quarkus or Jakarta EE. risk: red - autoanswer_if_tags_present: + autoAnswerFor: - category: Technology tag: J2EE - autotag: + applyTags: - category: Technology tag: J2EE - - formulation: "What version of Java EE does the application use?" + - text: "What version of Java EE does the application use?" explanation: "What version of the Java EE specification is your application using?" answers: - - choice: Below 5. + - text: Below 5. rationale: This technology stack is obsolete. mitigation: Consider migrating to at least Java EE 7. risk: red - - choice: 5 or 6 + - text: 5 or 6 rationale: This is a mostly outdated stack. mitigation: Consider migrating to at least Java EE 7. risk: yellow - - choice: "7" + - text: "7" risk: green - include_if_tags_present: + includeFor: - category: Technology tag: Java EE - - formulation: Does your application use any caching mechanism? + - text: Does your application use any caching mechanism? answers: - - choice: Yes + - text: Yes rationale: This could be problematic in containers and Kubernetes. mitigation: Review the clustering mechanism to check compatibility and support for container environments. risk: yellow - autoanswer_if_tags_present: + autoAnswerFor: - category: Caching tag: Infinispan - category: Caching @@ -78,97 +80,99 @@ sections: tag: eXtreme Scale - category: Caching tag: Coherence - - choice: No + - text: No risk: green - - choice: Unknown + - text: Unknown rationale: This is a problem because of the uncertainty. mitigation: Gathering more information about this is required. risk: unknown - - formulation: What implementation of JAX-WS does your application use? + - text: What implementation of JAX-WS does your application use? answers: - - choice: Apache Axis + - text: Apache Axis rationale: This version is obsolete mitigation: Consider migrating to Apache CXF risk: red - - choice: Apache CXF + - text: Apache CXF risk: green - - choice: Unknown + - text: Unknown rationale: This is a problem because of the uncertainty. mitigation: Gathering more information about this is required. risk: unknown - skip_if_tags_present: + excludeFor: - category: Technology tag: Spring Boot - category: Technology tag: Quarkus - - name: Application technologies + + - order: 2 + name: Application technologies questions: - - formulation: What is the main technology in your application? + - text: What is the main technology in your application? explanation: What would you describe as the main framework used to build your application. answers: - - choice: Unknown + - text: Unknown rationale: This is a problem because of the uncertainty. mitigation: Gathering more information about this is required. risk: unknown - - choice: Quarkus + - text: Quarkus risk: green - autoanswer_if_tags_present: + autoAnswerFor: - category: Technology tag: Quarkus - autotag: + applyTags: - category: Technology tag: Quarkus - - choice: Spring Boot + - text: Spring Boot risk: green - autoanswer_if_tags_present: + autoAnswerFor: - category: Technology tag: Spring Boot - autotag: + applyTags: - category: Technology tag: Spring Boot - - choice: Java EE + - text: Java EE rationale: This might not be the most cloud friendly technology. mitigation: Maybe start thinking about migrating to Quarkus or Jakarta EE. risk: yellow - autoanswer_if_tags_present: + autoAnswerFor: - category: Technology tag: Java EE - autotag: + applyTags: - category: Technology tag: Java EE - - choice: J2EE + - text: J2EE rationale: This is obsolete. mitigation: Maybe start thinking about migrating to Quarkus or Jakarta EE. risk: red - autoanswer_if_tags_present: + autoAnswerFor: - category: Technology tag: J2EE - autotag: + applyTags: - category: Technology tag: J2EE - - formulation: "What version of Java EE does the application use?" + - text: "What version of Java EE does the application use?" explanation: "What version of the Java EE specification is your application using?" answers: - - choice: Below 5. + - text: Below 5. rationale: This technology stack is obsolete. mitigation: Consider migrating to at least Java EE 7. risk: red - - choice: 5 or 6 + - text: 5 or 6 rationale: This is a mostly outdated stack. mitigation: Consider migrating to at least Java EE 7. risk: yellow - - choice: "7" + - text: "7" risk: green - include_if_tags_present: + includeFor: - category: Technology tag: Java EE - - formulation: Does your application use any caching mechanism? + - text: Does your application use any caching mechanism? answers: - - choice: Yes + - text: Yes rationale: This could be problematic in containers and Kubernetes. mitigation: Review the clustering mechanism to check compatibility and support for container environments. risk: yellow - autoanswer_if_tags_present: + autoAnswerFor: - category: Caching tag: Infinispan - category: Caching @@ -177,34 +181,36 @@ sections: tag: eXtreme Scale - category: Caching tag: Coherence - - choice: No + - text: No risk: green - - choice: Unknown + - text: Unknown rationale: This is a problem because of the uncertainty. mitigation: Gathering more information about this is required. risk: unknown - - formulation: What implementation of JAX-WS does your application use? + - text: What implementation of JAX-WS does your application use? answers: - - choice: Apache Axis + - text: Apache Axis rationale: This version is obsolete mitigation: Consider migrating to Apache CXF risk: red - - choice: Apache CXF + - text: Apache CXF risk: green - - choice: Unknown + - text: Unknown rationale: This is a problem because of the uncertainty. mitigation: Gathering more information about this is required. risk: unknown - skip_if_tags_present: + excludeFor: - category: Technology tag: Spring Boot - category: Technology tag: Quarkus + thresholds: - red: 1% - yellow: 30% - unknown: 15% -risk_messages: + red: 1 + yellow: 30 + unknown: 15 + +riskMessages: red: Application requires deep changes in architecture or lifecycle yellow: Application is Cloud friendly but requires some minor changes green: Application is Cloud Native diff --git a/hack/import-questionnaire/demo-questionnaire-remove-question.yml b/hack/import-questionnaire/demo-questionnaire-remove-question.yml new file mode 100644 index 0000000000..0349544f2b --- /dev/null +++ b/hack/import-questionnaire/demo-questionnaire-remove-question.yml @@ -0,0 +1,103 @@ +name: Test questionnaire 2 +description: This is a sample questionnaire in YAML format +required: true +sections: + - order: 1 + name: Section 1 + questions: + - order: 1 + text: What is your favorite color? + explanation: Please select your favorite color. + includeFor: + - category: Category1 + tag: Tag1 + excludeFor: [] + answers: + - order: 1 + text: Red + risk: red + rationale: Red is a nice color. + mitigation: No mitigation needed. + applyTags: [] + autoAnswerFor: [{ category: Category1, tag: Tag1 }] + selected: false + - order: 2 + text: Blue + risk: green + rationale: Blue is a calming color. + mitigation: No mitigation needed. + applyTags: [] + autoAnswerFor: [] + selected: false + - order: 2 + text: What is your favorite sport? + explanation: Please select your favorite sport. + excludeFor: + - category: Category1 + tag: Tag1 + answers: + - order: 1 + text: Soccer + risk: red + rationale: There are other sports? + mitigation: Beware of crunching tackles. High risk of injury. + applyTags: [] + autoAnswerFor: [] + selected: false + - order: 2 + text: Cycling + risk: red + rationale: Correct. + mitigation: High risk of decapitation by car. + applyTags: [] + autoAnswerFor: [] + selected: false + - order: 3 + text: Climbing + risk: yellow + rationale: Climbing is fun. + mitigation: Slight bit of mitigation needed. + applyTags: [] + autoAnswerFor: [] + selected: false + - order: 4 + text: Swimming + risk: yellow + rationale: Swimming is fun, too. + mitigation: Slight bit of mitigation needed. Drowning can be a problem. + applyTags: [] + autoAnswerFor: [] + selected: false + - order: 5 + text: Running + risk: red + rationale: Oof. + mitigation: Some mitigation required. High risk of injury. Don't run with scissors (or at all). + applyTags: [] + autoAnswerFor: [] + selected: false + - order: 3 + text: What is your favorite shape? + explanation: Please select your favorite shape. + includeFor: + - category: Category1 + tag: Tag1 + excludeFor: [] + answers: + - order: 1 + text: Circle + risk: red + selected: false + - order: 2 + text: Square + risk: green + selected: false +thresholds: + red: 5 + yellow: 10 + unknown: 15 +riskMessages: + red: High risk + yellow: Medium risk + green: Low risk + unknown: Unknown risk diff --git a/hack/import-questionnaire/demo-questionnaire.yml b/hack/import-questionnaire/demo-questionnaire.yml new file mode 100644 index 0000000000..1a5883942f --- /dev/null +++ b/hack/import-questionnaire/demo-questionnaire.yml @@ -0,0 +1,83 @@ +name: Test questionnaire +description: This is a sample questionnaire in YAML format +required: true +sections: + - order: 1 + name: Section 1 + questions: + - order: 1 + text: What is your favorite color? + explanation: Please select your favorite color. + includeFor: + - category: Category1 + tag: Tag1 + excludeFor: [] + answers: + - order: 1 + text: Red + risk: red + rationale: Red is a nice color. + mitigation: No mitigation needed. + applyTags: [] + selected: false + - order: 2 + text: Blue + risk: green + rationale: Blue is a calming color. + mitigation: No mitigation needed. + applyTags: [] + autoAnswerFor: [] + selected: false + - order: 2 + text: What is your favorite sport? + explanation: Please select your favorite sport. + answers: + - order: 1 + text: Soccer + risk: red + rationale: There are other sports? + mitigation: Beware of crunching tackles. High risk of injury. + applyTags: [] + autoAnswerFor: [] + selected: false + - order: 2 + text: Cycling + risk: red + rationale: Correct. + mitigation: High risk of decapitation by car. + applyTags: [] + autoAnswerFor: [] + selected: false + - order: 3 + text: Climbing + risk: yellow + rationale: Climbing is fun. + mitigation: Slight bit of mitigation needed. + applyTags: [] + autoAnswerFor: [] + selected: false + - order: 4 + text: Swimming + risk: yellow + rationale: Swimming is fun, too. + mitigation: Slight bit of mitigation needed. Drowning can be a problem. + applyTags: [] + autoAnswerFor: [] + selected: false + - order: 5 + text: Running + risk: red + rationale: Oof. + mitigation: Some mitigation required. High risk of injury. Don't run with scissors (or at all). + applyTags: [] + autoAnswerFor: [] + selected: false +thresholds: + red: 5 + yellow: 10 + unknown: 15 +riskMessages: + red: High risk + yellow: Medium risk + green: Low risk + unknown: Unknown risk