Skip to content

Commit a070227

Browse files
Marcus SorensenMarcus SorensenDaanHoogland
authored
server Don't allow inadvertent deletion of hidden details via API (#7880)
* Don't allow inadvertent deletion of hidden details via API * Update VM details unit test ensuring system/hidden details not removed * Update test/integration/component/test_update_vm.py --------- Co-authored-by: Marcus Sorensen <mls@apple.com> Co-authored-by: dahn <daan.hoogland@gmail.com>
1 parent 2e1c282 commit a070227

File tree

3 files changed

+152
-26
lines changed

3 files changed

+152
-26
lines changed

server/src/main/java/com/cloud/vm/UserVmManagerImpl.java

Lines changed: 20 additions & 5 deletions
Original file line numberDiff line numberDiff line change
@@ -2756,13 +2756,18 @@ public UserVm updateVirtualMachine(UpdateVMCmd cmd) throws ResourceUnavailableEx
27562756
final List<String> userReadOnlySettings = Stream.of(QueryService.UserVMReadOnlyDetails.value().split(","))
27572757
.map(item -> (item).trim())
27582758
.collect(Collectors.toList());
2759+
List<UserVmDetailVO> existingDetails = userVmDetailsDao.listDetails(id);
27592760
if (cleanupDetails){
27602761
if (caller != null && caller.getType() == Account.Type.ADMIN) {
2761-
userVmDetailsDao.removeDetails(id);
2762+
for (final UserVmDetailVO detail : existingDetails) {
2763+
if (detail != null && detail.isDisplay()) {
2764+
userVmDetailsDao.removeDetail(id, detail.getName());
2765+
}
2766+
}
27622767
} else {
2763-
for (final UserVmDetailVO detail : userVmDetailsDao.listDetails(id)) {
2768+
for (final UserVmDetailVO detail : existingDetails) {
27642769
if (detail != null && !userDenyListedSettings.contains(detail.getName())
2765-
&& !userReadOnlySettings.contains(detail.getName())) {
2770+
&& !userReadOnlySettings.contains(detail.getName()) && detail.isDisplay()) {
27662771
userVmDetailsDao.removeDetail(id, detail.getName());
27672772
}
27682773
}
@@ -2782,15 +2787,25 @@ public UserVm updateVirtualMachine(UpdateVMCmd cmd) throws ResourceUnavailableEx
27822787
if (userReadOnlySettings.contains(detailName)) {
27832788
throw new InvalidParameterValueException("You're not allowed to add or edit the read-only setting: " + detailName);
27842789
}
2790+
if (existingDetails.stream().anyMatch(d -> Objects.equals(d.getName(), detailName) && !d.isDisplay())){
2791+
throw new InvalidParameterValueException("You're not allowed to add or edit the non-displayable setting: " + detailName);
2792+
}
27852793
}
2786-
// Add any hidden/denied or read-only detail
2787-
for (final UserVmDetailVO detail : userVmDetailsDao.listDetails(id)) {
2794+
// Add any existing user denied or read-only details. We do it here because admins would already provide these (or can delete them).
2795+
for (final UserVmDetailVO detail : existingDetails) {
27882796
if (userDenyListedSettings.contains(detail.getName()) || userReadOnlySettings.contains(detail.getName())) {
27892797
details.put(detail.getName(), detail.getValue());
27902798
}
27912799
}
27922800
}
27932801

2802+
// ensure details marked as non-displayable are maintained, regardless of admin or not
2803+
for (final UserVmDetailVO existingDetail : existingDetails) {
2804+
if (!existingDetail.isDisplay()) {
2805+
details.put(existingDetail.getName(), existingDetail.getValue());
2806+
}
2807+
}
2808+
27942809
verifyVmLimits(vmInstance, details);
27952810
vmInstance.setDetails(details);
27962811
_vmDao.saveDetails(vmInstance);

server/src/test/java/com/cloud/vm/UserVmManagerImplTest.java

Lines changed: 22 additions & 7 deletions
Original file line numberDiff line numberDiff line change
@@ -149,7 +149,7 @@ public class UserVmManagerImplTest {
149149
private EntityManager entityManager;
150150

151151
@Mock
152-
private UserVmDetailsDao userVmDetailVO;
152+
private UserVmDetailsDao userVmDetailsDao;
153153

154154
@Mock
155155
private UserVmVO userVmVoMock;
@@ -300,7 +300,7 @@ public void updateVirtualMachineTestDisplayChanged() throws ResourceUnavailableE
300300
verifyMethodsThatAreAlwaysExecuted();
301301

302302
Mockito.verify(userVmManagerImpl).updateDisplayVmFlag(false, vmId, userVmVoMock);
303-
Mockito.verify(userVmDetailVO, Mockito.times(0)).removeDetails(vmId);
303+
Mockito.verify(userVmDetailsDao, Mockito.times(0)).removeDetail(anyLong(), anyString());
304304
}
305305

306306
@Test
@@ -310,12 +310,15 @@ public void updateVirtualMachineTestCleanUpTrue() throws ResourceUnavailableExce
310310
Mockito.when(_serviceOfferingDao.findById(Mockito.anyLong(), Mockito.anyLong())).thenReturn((ServiceOfferingVO) offering);
311311
Mockito.when(updateVmCommand.isCleanupDetails()).thenReturn(true);
312312
Mockito.lenient().doNothing().when(userVmManagerImpl).updateDisplayVmFlag(false, vmId, userVmVoMock);
313-
Mockito.doNothing().when(userVmDetailVO).removeDetails(vmId);
313+
314314
Mockito.when(updateVmCommand.getUserdataId()).thenReturn(null);
315315

316+
prepareExistingDetails(vmId, "userdetail");
317+
316318
userVmManagerImpl.updateVirtualMachine(updateVmCommand);
317319
verifyMethodsThatAreAlwaysExecuted();
318-
Mockito.verify(userVmDetailVO).removeDetails(vmId);
320+
Mockito.verify(userVmDetailsDao).removeDetail(vmId, "userdetail");
321+
Mockito.verify(userVmDetailsDao, Mockito.times(0)).removeDetail(vmId, "systemdetail");
319322
Mockito.verify(userVmManagerImpl, Mockito.times(0)).updateDisplayVmFlag(false, vmId, userVmVoMock);
320323
}
321324

@@ -340,6 +343,16 @@ public void updateVirtualMachineTestCleanUpFalseAndDetailsEmpty() throws Resourc
340343
prepareAndExecuteMethodDealingWithDetails(false, false);
341344
}
342345

346+
private List<UserVmDetailVO> prepareExistingDetails(Long vmId, String... existingDetailKeys) {
347+
List<UserVmDetailVO> existingDetails = new ArrayList<>();
348+
for (String detail : existingDetailKeys) {
349+
existingDetails.add(new UserVmDetailVO(vmId, detail, "foo", true));
350+
}
351+
existingDetails.add(new UserVmDetailVO(vmId, "systemdetail", "bar", false));
352+
Mockito.when(userVmDetailsDao.listDetails(vmId)).thenReturn(existingDetails);
353+
return existingDetails;
354+
}
355+
343356
private void prepareAndExecuteMethodDealingWithDetails(boolean cleanUpDetails, boolean isDetailsEmpty) throws ResourceUnavailableException, InsufficientCapacityException {
344357
configureDoNothingForMethodsThatWeDoNotWantToTest();
345358

@@ -360,8 +373,9 @@ private void prepareAndExecuteMethodDealingWithDetails(boolean cleanUpDetails, b
360373
lenient().doNothing().when(_networkMgr).saveExtraDhcpOptions(anyString(), anyLong(), anyMap());
361374
HashMap<String, String> details = new HashMap<>();
362375
if(!isDetailsEmpty) {
363-
details.put("", "");
376+
details.put("newdetail", "foo");
364377
}
378+
prepareExistingDetails(vmId, "existingdetail");
365379
Mockito.when(updateVmCommand.getUserdataId()).thenReturn(null);
366380
Mockito.when(updateVmCommand.getDetails()).thenReturn(details);
367381
Mockito.when(updateVmCommand.isCleanupDetails()).thenReturn(cleanUpDetails);
@@ -371,14 +385,15 @@ private void prepareAndExecuteMethodDealingWithDetails(boolean cleanUpDetails, b
371385
verifyMethodsThatAreAlwaysExecuted();
372386

373387
Mockito.verify(userVmVoMock, Mockito.times(cleanUpDetails || isDetailsEmpty ? 0 : 1)).setDetails(details);
374-
Mockito.verify(userVmDetailVO, Mockito.times(cleanUpDetails ? 1: 0)).removeDetails(vmId);
388+
Mockito.verify(userVmDetailsDao, Mockito.times(cleanUpDetails ? 1 : 0)).removeDetail(vmId, "existingdetail");
389+
Mockito.verify(userVmDetailsDao, Mockito.times(0)).removeDetail(vmId, "systemdetail");
375390
Mockito.verify(userVmDao, Mockito.times(cleanUpDetails || isDetailsEmpty ? 0 : 1)).saveDetails(userVmVoMock);
376391
Mockito.verify(userVmManagerImpl, Mockito.times(0)).updateDisplayVmFlag(false, vmId, userVmVoMock);
377392
}
378393

379394
private void configureDoNothingForDetailsMethod() {
380395
Mockito.lenient().doNothing().when(userVmManagerImpl).updateDisplayVmFlag(false, vmId, userVmVoMock);
381-
Mockito.doNothing().when(userVmDetailVO).removeDetails(vmId);
396+
Mockito.doNothing().when(userVmDetailsDao).removeDetail(anyLong(), anyString());
382397
Mockito.doNothing().when(userVmDao).saveDetails(userVmVoMock);
383398
}
384399

test/integration/component/test_update_vm.py

Lines changed: 110 additions & 14 deletions
Original file line numberDiff line numberDiff line change
@@ -15,11 +15,11 @@
1515
# specific language governing permissions and limitations
1616
# under the License.
1717

18-
1918
from marvin.cloudstackTestCase import cloudstackTestCase
2019
from marvin.lib.base import Account, VirtualMachine, ServiceOffering
21-
from marvin.lib.utils import cleanup_resources
20+
from marvin.lib.utils import (validateList, cleanup_resources)
2221
from marvin.lib.common import get_zone, get_domain, get_template
22+
from marvin.codes import PASS
2323
from nose.plugins.attrib import attr
2424

2525
class TestData(object):
@@ -59,6 +59,7 @@ class TestUpdateVirtualMachine(cloudstackTestCase):
5959
def setUp(self):
6060
self.testdata = TestData().testdata
6161
self.apiclient = self.testClient.getApiClient()
62+
self.dbclient = self.testClient.getDbConnection()
6263

6364
# Get Zone, Domain and Default Built-in template
6465
self.domain = get_domain(self.apiclient)
@@ -106,18 +107,7 @@ def test_update_vm_name(self):
106107
templateid=self.template.id
107108
)
108109

109-
list_vms = VirtualMachine.list(self.apiclient, id=self.virtual_machine.id)
110-
self.assertEqual(
111-
isinstance(list_vms, list),
112-
True,
113-
"List VM response was not a valid list"
114-
)
115-
self.assertNotEqual(
116-
len(list_vms),
117-
0,
118-
"List VM response was empty"
119-
)
120-
vm = list_vms[0]
110+
vm = self.listVmById(self.virtual_machine.id)
121111

122112
self.debug(
123113
"VirtualMachine launched with id, name, displayname: %s %s %s"\
@@ -152,6 +142,112 @@ def test_update_vm_name(self):
152142
self.assertEqual(vmnew.displayname, vmnewstarted.displayname,
153143
msg="display name changed on start, displayname is %s" % vmnewstarted.displayname)
154144

145+
@attr(tags=['advanced', 'simulator', 'basic', 'sg', 'details'], required_hardware="false")
146+
def test_update_vm_details_admin(self):
147+
"""Test Update VirtualMachine Details
148+
149+
# Set up a VM
150+
# Set up hidden detail in DB for VM
151+
152+
# Validate the following:
153+
# 1. Can add two details (detail1, detail2)
154+
# 2. Can fetch new details on VM
155+
# 3. Can delete detail1
156+
# 4. Hidden detail not removed
157+
# 6. The detail2 remains
158+
# 7. Ensure cleanup parameter doesn't remove hidden details
159+
"""
160+
hidden_detail_name = "configDriveLocation"
161+
detail1 = "detail1"
162+
detail2 = "detail2"
163+
164+
# set up a VM
165+
self.virtual_machine = VirtualMachine.create(
166+
self.apiclient,
167+
self.testdata["virtual_machine"],
168+
accountid=self.account.name,
169+
zoneid=self.zone.id,
170+
domainid=self.account.domainid,
171+
serviceofferingid=self.service_offering.id,
172+
templateid=self.template.id
173+
)
174+
self.cleanup.append(self.virtual_machine)
175+
176+
vm = self.listVmById(self.virtual_machine.id)
177+
178+
self.debug(
179+
"VirtualMachine launched with id, name, displayname: %s %s %s" \
180+
% (self.virtual_machine.id, vm.name, vm.displayname)
181+
)
182+
183+
# set up a hidden detail
184+
dbresult = self.dbclient.execute("select id from vm_instance where uuid='%s'" % vm.id)
185+
self.assertEqual(validateList(dbresult)[0], PASS, "sql query returned invalid response")
186+
vm_db_id = dbresult[0][0]
187+
self.debug("VM has database id %d" % vm_db_id)
188+
189+
self.dbclient.execute("insert into user_vm_details (vm_id, name, value, display) values (%d,'%s','HOST', 0)" % (vm_db_id, hidden_detail_name))
190+
191+
vm = self.listVmById(self.virtual_machine.id)
192+
self.debug("VirtualMachine fetched with details: %s of type %s" % (vm.details, type(vm.details)))
193+
194+
self.assertIsNone(vm.details[hidden_detail_name], "hidden detail should be hidden")
195+
196+
# add two details by appending to what was returned via API
197+
updating_vm_details = vm.details.__dict__
198+
updating_vm_details[detail1] = "foo"
199+
updating_vm_details[detail2] = "bar"
200+
201+
self.debug("Updating VM to new details: %s" % updating_vm_details)
202+
vm = self.virtual_machine.update(self.apiclient, details=[updating_vm_details])
203+
204+
self.assertIsNotNone(vm.details[detail1], "Expect " + detail1)
205+
self.assertIsNotNone(vm.details[detail2], "Expect " + detail2)
206+
self.assertIsNone(vm.details[hidden_detail_name], "hidden detail should be hidden")
207+
self.assertTrue(self.detailInDatabase(vm_db_id, hidden_detail_name), "hidden detail should still exist in db")
208+
209+
# delete one detail
210+
updating_vm_details = vm.details.__dict__
211+
del updating_vm_details["detail1"]
212+
213+
self.debug("Deleting one detail by updating details: %s" % updating_vm_details)
214+
vm = self.virtual_machine.update(self.apiclient, details=[updating_vm_details])
215+
216+
self.assertIsNone(vm.details[detail1], "Do not expect " + detail1)
217+
self.assertIsNotNone(vm.details[detail2], "Expect " + detail2)
218+
self.assertIsNone(vm.details[hidden_detail_name], "hidden detail should be hidden")
219+
self.assertTrue(self.detailInDatabase(vm_db_id, hidden_detail_name), "hidden detail should still exist in db")
220+
221+
# cleanup, ensure hidden detail is not deleted
222+
vm = self.virtual_machine.update(self.apiclient, cleanupdetails="true")
223+
self.assertIsNone(vm.details[detail1], "Do not expect " + detail1)
224+
self.assertIsNone(vm.details[detail2], "Do not expect " + detail2)
225+
self.assertIsNone(vm.details[hidden_detail_name], "hidden detail should be hidden")
226+
self.assertTrue(self.detailInDatabase(vm_db_id, hidden_detail_name), "hidden detail should still exist in db")
227+
228+
229+
def detailInDatabase(self, vm_id, detail_name):
230+
dbresult = self.dbclient.execute("select id from user_vm_details where vm_id=%s and name='%s'" % (vm_id, detail_name))
231+
self.debug("Detail %s for VM %s: %s" % (detail_name, vm_id, dbresult))
232+
if validateList(dbresult)[0] == PASS:
233+
return True
234+
return False
235+
236+
237+
def listVmById(self, id):
238+
list_vms = VirtualMachine.list(self.apiclient, id=id)
239+
self.assertEqual(
240+
isinstance(list_vms, list),
241+
True,
242+
"List VM response was not a valid list"
243+
)
244+
self.assertNotEqual(
245+
len(list_vms),
246+
0,
247+
"List VM response was empty"
248+
)
249+
return list_vms[0]
250+
155251
def tearDown(self):
156252
try:
157253
cleanup_resources(self.apiclient, self.cleanup)

0 commit comments

Comments
 (0)