Skip to content

Commit 14c2237

Browse files
authored
[ACM-17421] Add modal for VM actions (#4359)
* [ACM-17421] Add modal for VM actions Signed-off-by: zlayne <[email protected]> * fix lint & translation Signed-off-by: zlayne <[email protected]> * fix tests Signed-off-by: zlayne <[email protected]> * remove obsolete snapshots Signed-off-by: zlayne <[email protected]> * vm action modal test Signed-off-by: zlayne <[email protected]> * fix lint error Signed-off-by: zlayne <[email protected]> * Add vm modal error tests cases Signed-off-by: zlayne <[email protected]> * fix lint error Signed-off-by: zlayne <[email protected]> --------- Signed-off-by: zlayne <[email protected]>
1 parent 5c984af commit 14c2237

File tree

13 files changed

+556
-514
lines changed

13 files changed

+556
-514
lines changed

frontend/public/locales/en/translation.json

+2-5
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,7 @@
1010
"{{0}} values": "{{0}} values",
1111
"{{0}} within the last:": "{{0}} within the last:",
1212
"{{action}} {{resourceKind}}": "{{action}} {{resourceKind}}",
13+
"{{action}} VirtualMachine?": "{{action}} VirtualMachine?",
1314
"{{count}} cluster cannot be edited ": "{{count}} cluster cannot be edited ",
1415
"{{count}} cluster cannot be edited _plural": "{{count}} clusters cannot be edited ",
1516
"{{count}} clusters with unknown status": "{{count}} cluster with unknown status",
@@ -518,6 +519,7 @@
518519
"Are you sure that you want to continue?": "Are you sure that you want to continue?",
519520
"Are you sure that you want to delete {{resourceName}}?": "Are you sure that you want to delete {{resourceName}}?",
520521
"Are you sure that you want to delete saved search {{savedSearchName}}?": "Are you sure that you want to delete saved search {{savedSearchName}}?",
522+
"Are you sure you want to {{action}} {{vmName}} in namespace {{vmNamespace}}?": "Are you sure you want to {{action}} {{vmName}} in namespace {{vmNamespace}}?",
521523
"Argo": "Argo",
522524
"Argo application content": "Argo application content",
523525
"Argo application steps": "Argo application steps",
@@ -2103,7 +2105,6 @@
21032105
"Paste": "Paste",
21042106
"Path": "Path",
21052107
"Pathname": "Pathname",
2106-
"Pause {{resourceKind}}": "Pause {{resourceKind}}",
21072108
"Pause VirtualMachine": "Pause VirtualMachine",
21082109
"pending": "pending",
21092110
"Pending": "Pending",
@@ -2350,7 +2351,6 @@
23502351
"Resources from these managed hubs and managed clusters will be missing in search results. Resolve cluster issues to see all cluster resources.": "Resources from these managed hubs and managed clusters will be missing in search results. Resolve cluster issues to see all cluster resources.",
23512352
"Resources with \"failed\" or \"pending\" status.": "Resources with \"failed\" or \"pending\" status.",
23522353
"Response action": "Response action",
2353-
"Restart {{resourceKind}}": "Restart {{resourceKind}}",
23542354
"Restart VirtualMachine": "Restart VirtualMachine",
23552355
"Restarts": "Restarts",
23562356
"Restore defaults": "Restore defaults",
@@ -2520,7 +2520,6 @@
25202520
"Standalone control plane": "Standalone control plane",
25212521
"Standard": "Standard",
25222522
"Standards": "Standards",
2523-
"Start {{resourceKind}}": "Start {{resourceKind}}",
25242523
"Start time": "Start time",
25252524
"Start VirtualMachine": "Start VirtualMachine",
25262525
"Starting CSV": "Starting CSV",
@@ -2626,7 +2625,6 @@
26262625
"status.upgradefailed.alert.title": "The cluster failed to upgrade",
26272626
"status.upgradefailed.message": "The cluster upgrade is in a failure state.",
26282627
"Stay": "Stay",
2629-
"Stop {{resourceKind}}": "Stop {{resourceKind}}",
26302628
"Stop VirtualMachine": "Stop VirtualMachine",
26312629
"Storage class mapping {{num}}": "Storage class mapping {{num}}",
26322630
"Storage mapping": "Storage mapping",
@@ -3149,7 +3147,6 @@
31493147
"unknown: {{count}} cluster_plural": "unknown: {{count}} clusters",
31503148
"unknown: {{count}} policy": "unknown: {{count}} policy",
31513149
"unknown: {{count}} policy_plural": "unknown: {{count}} policies",
3152-
"Unpause {{resourceKind}}": "Unpause {{resourceKind}}",
31533150
"Unpause VirtualMachine": "Unpause VirtualMachine",
31543151
"Unprocessable entity": "Unprocessable entity",
31553152
"update": "Update",
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,182 @@
1+
/* Copyright Contributors to the Open Cluster Management project */
2+
import { cleanup, render, screen, waitFor } from '@testing-library/react'
3+
import userEvent from '@testing-library/user-event'
4+
import { RecoilRoot } from 'recoil'
5+
import { fetchRetry } from '../../../resources/utils/resource-request'
6+
import { VMActionModal } from './VMActionModal'
7+
8+
jest.mock('../../../resources/utils/resource-request', () => ({
9+
getBackendUrl: jest.fn(() => ''),
10+
fetchRetry: jest.fn(({ url }) => {
11+
if (url === '/apis/subresources.kubevirt.io/v1/namespaces/testVMNamespace/virtualmachines/testVM/noop') {
12+
return Promise.reject(new Error())
13+
} else if (
14+
url === '/apis/subresources.kubevirt.io/v1/namespaces/testVMNamespace/virtualmachines/testVM/unauthorized'
15+
) {
16+
return Promise.reject(new Error('Error: Unauthorized'))
17+
}
18+
return Promise.resolve()
19+
}),
20+
}))
21+
22+
describe('VMActionModal', () => {
23+
afterEach(cleanup)
24+
test('renders VMActionModal correctly and successfully calls start action on hub vm', async () => {
25+
const abortController = new AbortController()
26+
const { getByTestId } = render(
27+
<RecoilRoot>
28+
<VMActionModal
29+
open={true}
30+
close={() => {}}
31+
action={'start'}
32+
method={'PUT'}
33+
item={{
34+
name: 'testVM',
35+
namespace: 'testVMNamespace',
36+
cluster: 'test-cluster',
37+
_hubClusterResource: 'true',
38+
}}
39+
/>
40+
</RecoilRoot>
41+
)
42+
await waitFor(() => expect(screen.queryByText('start VirtualMachine?')).toBeInTheDocument())
43+
await waitFor(() =>
44+
expect(
45+
screen.queryByText('Are you sure you want to start testVM in namespace testVMNamespace?')
46+
).toBeInTheDocument()
47+
)
48+
49+
// verify click launch button
50+
const confirmButton = getByTestId('vm-modal-confirm')
51+
expect(confirmButton).toBeTruthy()
52+
userEvent.click(confirmButton)
53+
54+
expect(fetchRetry).toHaveBeenCalledWith({
55+
data: {
56+
body: {},
57+
managedCluster: 'test-cluster',
58+
vmName: 'testVM',
59+
vmNamespace: 'testVMNamespace',
60+
},
61+
disableRedirectUnauthorizedLogin: true,
62+
headers: {
63+
Accept: '*/*',
64+
},
65+
method: 'PUT',
66+
retries: 0,
67+
signal: abortController.signal,
68+
url: '/apis/subresources.kubevirt.io/v1/namespaces/testVMNamespace/virtualmachines/testVM/start',
69+
})
70+
})
71+
72+
test('renders VMActionModal correctly and successfully calls start action on managed cluster vm', async () => {
73+
const abortController = new AbortController()
74+
const { getByTestId } = render(
75+
<RecoilRoot>
76+
<VMActionModal
77+
open={true}
78+
close={() => {}}
79+
action={'start'}
80+
method={'PUT'}
81+
item={{
82+
name: 'testVM',
83+
namespace: 'testVMNamespace',
84+
cluster: 'test-cluster',
85+
}}
86+
/>
87+
</RecoilRoot>
88+
)
89+
await waitFor(() => expect(screen.queryByText('start VirtualMachine?')).toBeInTheDocument())
90+
await waitFor(() =>
91+
expect(
92+
screen.queryByText('Are you sure you want to start testVM in namespace testVMNamespace?')
93+
).toBeInTheDocument()
94+
)
95+
96+
// verify click launch button
97+
const confirmButton = getByTestId('vm-modal-confirm')
98+
expect(confirmButton).toBeTruthy()
99+
userEvent.click(confirmButton)
100+
101+
expect(fetchRetry).toHaveBeenCalledWith({
102+
data: {
103+
body: {},
104+
managedCluster: 'test-cluster',
105+
vmName: 'testVM',
106+
vmNamespace: 'testVMNamespace',
107+
},
108+
disableRedirectUnauthorizedLogin: true,
109+
headers: {
110+
Accept: '*/*',
111+
},
112+
method: 'PUT',
113+
retries: 0,
114+
signal: abortController.signal,
115+
url: '/virtualmachines/start',
116+
})
117+
})
118+
119+
test('renders VMActionModal correctly and returns action error', async () => {
120+
const { getByTestId } = render(
121+
<RecoilRoot>
122+
<VMActionModal
123+
open={true}
124+
close={() => {}}
125+
action={'noop'}
126+
method={'PUT'}
127+
item={{
128+
name: 'testVM',
129+
namespace: 'testVMNamespace',
130+
cluster: 'test-cluster',
131+
_hubClusterResource: 'true',
132+
}}
133+
/>
134+
</RecoilRoot>
135+
)
136+
await waitFor(() => expect(screen.queryByText('noop VirtualMachine?')).toBeInTheDocument())
137+
await waitFor(() =>
138+
expect(
139+
screen.queryByText('Are you sure you want to noop testVM in namespace testVMNamespace?')
140+
).toBeInTheDocument()
141+
)
142+
143+
// verify click launch button
144+
const confirmButton = getByTestId('vm-modal-confirm')
145+
expect(confirmButton).toBeTruthy()
146+
userEvent.click(confirmButton)
147+
148+
expect(fetchRetry).toThrow()
149+
})
150+
151+
test('renders VMActionModal correctly and returns unauthorized error', async () => {
152+
const { getByTestId } = render(
153+
<RecoilRoot>
154+
<VMActionModal
155+
open={true}
156+
close={() => {}}
157+
action={'unauthorized'}
158+
method={'PUT'}
159+
item={{
160+
name: 'testVM',
161+
namespace: 'testVMNamespace',
162+
cluster: 'test-cluster',
163+
_hubClusterResource: 'true',
164+
}}
165+
/>
166+
</RecoilRoot>
167+
)
168+
await waitFor(() => expect(screen.queryByText('unauthorized VirtualMachine?')).toBeInTheDocument())
169+
await waitFor(() =>
170+
expect(
171+
screen.queryByText('Are you sure you want to unauthorized testVM in namespace testVMNamespace?')
172+
).toBeInTheDocument()
173+
)
174+
175+
// verify click launch button
176+
const confirmButton = getByTestId('vm-modal-confirm')
177+
expect(confirmButton).toBeTruthy()
178+
userEvent.click(confirmButton)
179+
180+
expect(fetchRetry).toThrow()
181+
})
182+
})
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,136 @@
1+
/* Copyright Contributors to the Open Cluster Management project */
2+
// Copyright (c) 2021 Red Hat, Inc.
3+
// Copyright Contributors to the Open Cluster Management project
4+
import { ButtonVariant, ModalVariant } from '@patternfly/react-core'
5+
import { useContext } from 'react'
6+
import { TFunction } from 'react-i18next'
7+
import { useTranslation } from '../../../lib/acm-i18next'
8+
import { fetchRetry, getBackendUrl } from '../../../resources/utils'
9+
import { AcmButton, AcmModal, AcmToastContext, IAlertContext } from '../../../ui-components'
10+
import { searchClient } from '../../Search/search-sdk/search-client'
11+
12+
export interface IVMActionModalProps {
13+
open: boolean
14+
close: () => void
15+
action: string
16+
method: 'PUT' | 'GET' | 'POST' | 'PATCH' | 'DELETE'
17+
item: {
18+
name: string
19+
namespace: string
20+
cluster: string
21+
_hubClusterResource?: string
22+
}
23+
}
24+
25+
export const ClosedVMActionModalProps: IVMActionModalProps = {
26+
open: false,
27+
close: () => {},
28+
action: '',
29+
method: 'PUT',
30+
item: {
31+
name: '',
32+
namespace: '',
33+
cluster: '',
34+
},
35+
}
36+
37+
export function handleVMActions(
38+
action: string,
39+
method: 'PUT' | 'GET' | 'POST' | 'PATCH' | 'DELETE',
40+
item: any,
41+
body: any,
42+
refetchVM: () => void, // Callback fn to refetch the vm after action
43+
toast: IAlertContext,
44+
t: TFunction
45+
) {
46+
const abortController = new AbortController()
47+
48+
const subResourceKind = action.toLowerCase().includes('pause') ? 'virtualmachineinstances' : 'virtualmachines'
49+
const path = item?._hubClusterResource
50+
? `/apis/subresources.kubevirt.io/v1/namespaces/${item.namespace}/${subResourceKind}/${item.name}/${action.toLowerCase()}`
51+
: `/${subResourceKind}/${action.toLowerCase()}`
52+
53+
fetchRetry({
54+
method: method,
55+
url: `${getBackendUrl()}${path}`,
56+
data: {
57+
managedCluster: item.cluster,
58+
vmName: item.name,
59+
vmNamespace: item.namespace,
60+
body: body || {},
61+
},
62+
signal: abortController.signal,
63+
retries: process.env.NODE_ENV === 'production' ? 2 : 0,
64+
headers: { Accept: '*/*' },
65+
disableRedirectUnauthorizedLogin: true,
66+
})
67+
.then(() => {
68+
// Wait 5 seconds to allow search collector to catch up & refetch search results to update table.
69+
setTimeout(refetchVM, 5000)
70+
})
71+
.catch((err) => {
72+
console.error(`VirtualMachine: ${item.name} ${action} error. ${err}`)
73+
74+
let errMessage: string = err?.message ?? t('An unexpected error occurred.')
75+
if (errMessage.includes(':')) errMessage = errMessage.split(':').slice(1).join(':')
76+
if (errMessage.trim() === 'Unauthorized') errMessage = t('Unauthorized to execute this action.')
77+
toast.addAlert({
78+
title: t('Error triggering action {{action}} on VirtualMachine {{name}}', {
79+
name: item.name,
80+
action,
81+
}),
82+
message: errMessage,
83+
type: 'danger',
84+
})
85+
})
86+
}
87+
88+
export const VMActionModal = (props: IVMActionModalProps) => {
89+
const { t } = useTranslation()
90+
const toast = useContext(AcmToastContext)
91+
const { open, close, action, method, item } = props
92+
93+
return (
94+
<AcmModal
95+
id={'vm-action-modal'}
96+
variant={ModalVariant.medium}
97+
isOpen={open}
98+
title={t('{{action}} VirtualMachine?', { action })}
99+
titleIconVariant={'warning'}
100+
onClose={close}
101+
actions={[
102+
<AcmButton
103+
id="vm-modal-confirm"
104+
// isDisabled={loadingAccessRequest || !canDelete}
105+
key="confirm"
106+
variant={ButtonVariant.danger}
107+
onClick={() => {
108+
handleVMActions(
109+
action,
110+
method,
111+
item,
112+
{},
113+
() => searchClient.refetchQueries({ include: ['searchResultItems'] }),
114+
toast,
115+
t
116+
)
117+
close()
118+
}}
119+
>
120+
{action}
121+
</AcmButton>,
122+
<AcmButton key="cancel" variant={ButtonVariant.secondary} onClick={close}>
123+
{t('Cancel')}
124+
</AcmButton>,
125+
]}
126+
>
127+
<div style={{ paddingTop: '1rem' }}>
128+
{t('Are you sure you want to {{action}} {{vmName}} in namespace {{vmNamespace}}?', {
129+
action: action.toLowerCase(),
130+
vmName: item.name,
131+
vmNamespace: item.namespace,
132+
})}
133+
</div>
134+
</AcmModal>
135+
)
136+
}

0 commit comments

Comments
 (0)