Skip to content

Commit b57ad1b

Browse files
author
Marcus Sorensen
committed
Don't allow inadvertent deletion of hidden details via API
1 parent f7345e8 commit b57ad1b

File tree

2 files changed

+129
-19
lines changed

2 files changed

+129
-19
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);

test/integration/component/test_update_vm.py

Lines changed: 109 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,111 @@ 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+
175+
vm = self.listVmById(self.virtual_machine.id)
176+
177+
self.debug(
178+
"VirtualMachine launched with id, name, displayname: %s %s %s" \
179+
% (self.virtual_machine.id, vm.name, vm.displayname)
180+
)
181+
182+
# set up a hidden detail
183+
dbresult = self.dbclient.execute("select id from vm_instance where uuid='%s'" % vm.id)
184+
self.assertEqual(validateList(dbresult)[0], PASS, "sql query returned invalid response")
185+
vm_db_id = dbresult[0][0]
186+
self.debug("VM has database id %d" % vm_db_id)
187+
188+
self.dbclient.execute("insert into user_vm_details (vm_id, name, value, display) values (%d,'%s','HOST', 0)" % (vm_db_id, hidden_detail_name))
189+
190+
vm = self.listVmById(self.virtual_machine.id)
191+
self.debug("VirtualMachine fetched with details: %s of type %s" % (vm.details, type(vm.details)))
192+
193+
self.assertIsNone(vm.details[hidden_detail_name], "hidden detail should be hidden")
194+
195+
# add two details by appending to what was returned via API
196+
updating_vm_details = vm.details.__dict__
197+
updating_vm_details[detail1] = "foo"
198+
updating_vm_details[detail2] = "bar"
199+
200+
self.debug("Updating VM to new details: %s" % updating_vm_details)
201+
vm = self.virtual_machine.update(self.apiclient, details=[updating_vm_details])
202+
203+
self.assertIsNotNone(vm.details[detail1], "Expect " + detail1)
204+
self.assertIsNotNone(vm.details[detail2], "Expect " + detail2)
205+
self.assertIsNone(vm.details[hidden_detail_name], "hidden detail should be hidden")
206+
self.assertTrue(self.detailInDatabase(vm_db_id, hidden_detail_name), "hidden detail should still exist in db")
207+
208+
# delete one detail
209+
updating_vm_details = vm.details.__dict__
210+
del updating_vm_details["detail1"]
211+
212+
self.debug("Deleting one detail by updating details: %s" % updating_vm_details)
213+
vm = self.virtual_machine.update(self.apiclient, details=[updating_vm_details])
214+
215+
self.assertIsNone(vm.details[detail1], "Do not expect " + detail1)
216+
self.assertIsNotNone(vm.details[detail2], "Expect " + detail2)
217+
self.assertIsNone(vm.details[hidden_detail_name], "hidden detail should be hidden")
218+
self.assertTrue(self.detailInDatabase(vm_db_id, hidden_detail_name), "hidden detail should still exist in db")
219+
220+
# cleanup, ensure hidden detail is not deleted
221+
vm = self.virtual_machine.update(self.apiclient, cleanupdetails="true")
222+
self.assertIsNone(vm.details[detail1], "Do not expect " + detail1)
223+
self.assertIsNone(vm.details[detail2], "Do not expect " + detail2)
224+
self.assertIsNone(vm.details[hidden_detail_name], "hidden detail should be hidden")
225+
self.assertTrue(self.detailInDatabase(vm_db_id, hidden_detail_name), "hidden detail should still exist in db")
226+
227+
228+
def detailInDatabase(self, vm_id, detail_name):
229+
dbresult = self.dbclient.execute("select id from user_vm_details where vm_id=%s and name='%s'" % (vm_id, detail_name))
230+
self.debug("Detail %s for VM %s: %s" % (detail_name, vm_id, dbresult))
231+
if validateList(dbresult)[0] == PASS:
232+
return True
233+
return False
234+
235+
236+
def listVmById(self, id):
237+
list_vms = VirtualMachine.list(self.apiclient, id=id)
238+
self.assertEqual(
239+
isinstance(list_vms, list),
240+
True,
241+
"List VM response was not a valid list"
242+
)
243+
self.assertNotEqual(
244+
len(list_vms),
245+
0,
246+
"List VM response was empty"
247+
)
248+
return list_vms[0]
249+
155250
def tearDown(self):
156251
try:
157252
cleanup_resources(self.apiclient, self.cleanup)

0 commit comments

Comments
 (0)