mirror of
https://github.com/apache/cloudstack.git
synced 2025-10-26 08:42:29 +01:00
Fix Vijay's test to use the marvin integratin libararies
Changes to marvin_request and base libraries to accomadate POST requests. Additional tests for sending userdata through GET Userdata tests - send size>2k data via GET and POST. Signed-off-by: Prasanna Santhanam <tsp@apache.org>
This commit is contained in:
parent
b0caae6b33
commit
4e9e7937ed
@ -1,108 +0,0 @@
|
|||||||
#!/usr/bin/env python
|
|
||||||
|
|
||||||
import marvin
|
|
||||||
from marvin import cloudstackTestCase
|
|
||||||
from marvin.cloudstackTestCase import *
|
|
||||||
from marvin.integration.lib.base import *
|
|
||||||
|
|
||||||
import unittest
|
|
||||||
import hashlib
|
|
||||||
import random
|
|
||||||
import os
|
|
||||||
import string
|
|
||||||
|
|
||||||
class TestDeployVmWithUserData(cloudstackTestCase):
|
|
||||||
"""
|
|
||||||
This test deploys a virtual machine into a user account
|
|
||||||
using the small service offering and builtin template
|
|
||||||
"""
|
|
||||||
def setUp(self):
|
|
||||||
password = "password"
|
|
||||||
|
|
||||||
self.apiClient = self.testClient.getApiClient() #Get ourselves an API client
|
|
||||||
|
|
||||||
self.acct = createAccount.createAccountCmd() #The createAccount command
|
|
||||||
self.acct.accounttype = 0 #We need a regular user. admins have accounttype=1
|
|
||||||
self.acct.firstname = 'firstname'
|
|
||||||
self.acct.lastname = 'lastname'
|
|
||||||
self.acct.password = password
|
|
||||||
self.acct.username = 'user1'
|
|
||||||
self.acct.email = 'user1@user.com'
|
|
||||||
self.acct.account = 'user1'
|
|
||||||
self.acct.domainid = 1 #The default ROOT domain
|
|
||||||
self.acctResponse = self.apiClient.createAccount(self.acct)
|
|
||||||
|
|
||||||
self.debug("Successfully created account: %s, user: %s, id: \
|
|
||||||
%s"%(self.acctResponse.account.account, \
|
|
||||||
self.acctResponse.account.username, \
|
|
||||||
self.acctResponse.account.id))
|
|
||||||
|
|
||||||
# Generate userdata of 2500 bytes. This is larger than the 2048 bytes limit.
|
|
||||||
# CS however allows for upto 4K bytes in the code. So this must succeed.
|
|
||||||
# Overall, the query length must not exceed 4K, for then the json decoder
|
|
||||||
# will fail this operation at the marvin client side itself.
|
|
||||||
user_data = ''.join(random.choice(string.ascii_uppercase + string.digits) for x in range(2500))
|
|
||||||
|
|
||||||
self.virtual_machine = {
|
|
||||||
"displayname": "Test VM",
|
|
||||||
"username": "root",
|
|
||||||
"password": "password",
|
|
||||||
"ssh_port": 22,
|
|
||||||
"hypervisor": 'VMware',
|
|
||||||
"privateport": 22,
|
|
||||||
"publicport": 22,
|
|
||||||
"protocol": 'TCP',
|
|
||||||
}
|
|
||||||
#self.virtual_machine["userdata"] = base64.b64encode(user_data)
|
|
||||||
self.virtual_machine["userdata"] = user_data
|
|
||||||
|
|
||||||
def test_DeployVm(self):
|
|
||||||
"""
|
|
||||||
Let's start by defining the attributes of our VM that we will be
|
|
||||||
deploying on CloudStack. We will be assuming a single zone is available
|
|
||||||
and is configured and all templates are Ready
|
|
||||||
"""
|
|
||||||
deployVmCmd = deployVirtualMachine.deployVirtualMachineCmd()
|
|
||||||
deployVmCmd.zoneid = 1
|
|
||||||
deployVmCmd.account = self.acct.account
|
|
||||||
deployVmCmd.domainid = self.acct.domainid
|
|
||||||
deployVmCmd.templateid = 7
|
|
||||||
deployVmCmd.serviceofferingid = 1
|
|
||||||
|
|
||||||
# Userdata is passed in the virtual_machine dictionary.
|
|
||||||
deployVmResponse = VirtualMachine.create(
|
|
||||||
self.apiClient,
|
|
||||||
self.virtual_machine,
|
|
||||||
accountid=self.acct.account,
|
|
||||||
domainid=self.acct.domainid,
|
|
||||||
serviceofferingid=deployVmCmd.serviceofferingid,
|
|
||||||
templateid=deployVmCmd.templateid,
|
|
||||||
zoneid=deployVmCmd.zoneid
|
|
||||||
)
|
|
||||||
|
|
||||||
# At this point our VM is expected to be Running. Let's find out what
|
|
||||||
# listVirtualMachines tells us about VMs in this account
|
|
||||||
|
|
||||||
listVmCmd = listVirtualMachines.listVirtualMachinesCmd()
|
|
||||||
listVmCmd.id = deployVmResponse.id
|
|
||||||
listVmResponse = self.apiClient.listVirtualMachines(listVmCmd)
|
|
||||||
|
|
||||||
self.assertNotEqual(len(listVmResponse), 0, "Check if the list API \
|
|
||||||
returns a non-empty response")
|
|
||||||
|
|
||||||
vm = listVmResponse[0]
|
|
||||||
|
|
||||||
self.assertEqual(vm.id, deployVmResponse.id, "Check if the VM returned \
|
|
||||||
is the same as the one we deployed")
|
|
||||||
|
|
||||||
|
|
||||||
self.assertEqual(vm.state, "Running", "Check if VM has reached \
|
|
||||||
a state of running")
|
|
||||||
|
|
||||||
def tearDown(self):
|
|
||||||
"""
|
|
||||||
Delete the account created. This will clear the VM belonging to that account as well.
|
|
||||||
"""
|
|
||||||
deleteAcct = deleteAccount.deleteAccountCmd()
|
|
||||||
deleteAcct.id = self.acctResponse.account.id
|
|
||||||
self.apiClient.deleteAccount(deleteAcct)
|
|
||||||
144
test/integration/smoke/test_deploy_vm_with_userdata.py
Normal file
144
test/integration/smoke/test_deploy_vm_with_userdata.py
Normal file
@ -0,0 +1,144 @@
|
|||||||
|
# Licensed to the Apache Software Foundation (ASF) under one
|
||||||
|
# or more contributor license agreements. See the NOTICE file
|
||||||
|
# distributed with this work for additional information
|
||||||
|
# regarding copyright ownership. The ASF licenses this file
|
||||||
|
# to you under the Apache License, Version 2.0 (the
|
||||||
|
# "License"); you may not use this file except in compliance
|
||||||
|
# with the License. You may obtain a copy of the License at
|
||||||
|
#
|
||||||
|
# http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
#
|
||||||
|
# Unless required by applicable law or agreed to in writing,
|
||||||
|
# software distributed under the License is distributed on an
|
||||||
|
# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
|
||||||
|
# KIND, either express or implied. See the License for the
|
||||||
|
# specific language governing permissions and limitations
|
||||||
|
# under the License.
|
||||||
|
|
||||||
|
from marvin.cloudstackTestCase import cloudstackTestCase
|
||||||
|
from marvin.integration.lib.base import *
|
||||||
|
from marvin.integration.lib.common import get_template, get_zone, list_virtual_machines, cleanup_resources
|
||||||
|
from nose.plugins.attrib import attr
|
||||||
|
|
||||||
|
import random
|
||||||
|
import string
|
||||||
|
|
||||||
|
class Services:
|
||||||
|
def __init__(self):
|
||||||
|
self.services = {
|
||||||
|
"account": {
|
||||||
|
"email": "test@test.com",
|
||||||
|
"firstname": "Test",
|
||||||
|
"lastname": "User",
|
||||||
|
"username": "test",
|
||||||
|
"password": "password",
|
||||||
|
},
|
||||||
|
"virtual_machine": {
|
||||||
|
"displayname": "Test VM",
|
||||||
|
"username": "root",
|
||||||
|
"password": "password",
|
||||||
|
"ssh_port": 22,
|
||||||
|
"hypervisor": 'XenServer',
|
||||||
|
"privateport": 22,
|
||||||
|
"publicport": 22,
|
||||||
|
"protocol": 'TCP',
|
||||||
|
},
|
||||||
|
"ostype": 'CentOS 5.3 (64-bit)',
|
||||||
|
"service_offering": {
|
||||||
|
"name": "Tiny Instance",
|
||||||
|
"displaytext": "Tiny Instance",
|
||||||
|
"cpunumber": 1,
|
||||||
|
"cpuspeed": 100,
|
||||||
|
"memory": 256,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
class TestDeployVmWithUserData(cloudstackTestCase):
|
||||||
|
"""Tests for UserData
|
||||||
|
"""
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def setUpClass(cls):
|
||||||
|
cls.apiClient = super(TestDeployVmWithUserData, cls).getClsTestClient().getApiClient()
|
||||||
|
cls.services = Services().services
|
||||||
|
cls.zone = get_zone(cls.apiClient, cls.services)
|
||||||
|
if cls.zone.localstorageenabled:
|
||||||
|
#For devcloud since localstroage is enabled
|
||||||
|
cls.services["service_offering"]["storagetype"] = "local"
|
||||||
|
cls.service_offering = ServiceOffering.create(
|
||||||
|
cls.apiClient,
|
||||||
|
cls.services["service_offering"]
|
||||||
|
)
|
||||||
|
cls.account = Account.create(cls.apiClient, services=cls.services["account"])
|
||||||
|
cls.template = get_template(
|
||||||
|
cls.apiClient,
|
||||||
|
cls.zone.id,
|
||||||
|
cls.services["ostype"]
|
||||||
|
)
|
||||||
|
cls.debug("Successfully created account: %s, id: \
|
||||||
|
%s" % (cls.account.name,\
|
||||||
|
cls.account.id))
|
||||||
|
cls.cleanup = [cls.account]
|
||||||
|
|
||||||
|
# Generate userdata of 2500 bytes. This is larger than the 2048 bytes limit.
|
||||||
|
# CS however allows for upto 4K bytes in the code. So this must succeed.
|
||||||
|
# Overall, the query length must not exceed 4K, for then the json decoder
|
||||||
|
# will fail this operation at the marvin client side itcls.
|
||||||
|
user_data = ''.join(random.choice(string.ascii_uppercase + string.digits) for x in range(2500))
|
||||||
|
cls.services["virtual_machine"]["userdata"] = user_data
|
||||||
|
|
||||||
|
@attr(tags=["simulator", "devcloud", "basic", "advanced"])
|
||||||
|
def test_deployvm_userdata_post(self):
|
||||||
|
"""Test userdata as POST, size > 2k
|
||||||
|
"""
|
||||||
|
deployVmResponse = VirtualMachine.create(
|
||||||
|
self.apiClient,
|
||||||
|
services=self.services["virtual_machine"],
|
||||||
|
accountid=self.account.name,
|
||||||
|
domainid=self.account.domainid,
|
||||||
|
serviceofferingid=self.service_offering.id,
|
||||||
|
templateid=self.template.id,
|
||||||
|
zoneid=self.zone.id,
|
||||||
|
method='POST'
|
||||||
|
)
|
||||||
|
vms = list_virtual_machines(
|
||||||
|
self.apiClient,
|
||||||
|
account=self.account.name,
|
||||||
|
domainid=self.account.domainid
|
||||||
|
)
|
||||||
|
self.assert_(len(vms) > 0, "There are no Vms deployed in the account %s" % self.account.name)
|
||||||
|
vm = vms[0]
|
||||||
|
self.assert_(vm.id == str(deployVmResponse.id), "Vm deployed is different from the test")
|
||||||
|
self.assert_(vm.state == "Running", "VM is not in Running state")
|
||||||
|
|
||||||
|
@attr(tags=["simulator", "devcloud", "basic", "advanced"])
|
||||||
|
def test_deployvm_userdata(self):
|
||||||
|
"""Test userdata as GET, size > 2k
|
||||||
|
"""
|
||||||
|
deployVmResponse = VirtualMachine.create(
|
||||||
|
self.apiClient,
|
||||||
|
services=self.services["virtual_machine"],
|
||||||
|
accountid=self.account.name,
|
||||||
|
domainid=self.account.domainid,
|
||||||
|
serviceofferingid=self.service_offering.id,
|
||||||
|
templateid=self.template.id,
|
||||||
|
zoneid=self.zone.id
|
||||||
|
)
|
||||||
|
vms = list_virtual_machines(
|
||||||
|
self.apiClient,
|
||||||
|
account=self.account.name,
|
||||||
|
domainid=self.account.domainid
|
||||||
|
)
|
||||||
|
self.assert_(len(vms) > 0, "There are no Vms deployed in the account %s" % self.account.name)
|
||||||
|
vm = vms[0]
|
||||||
|
self.assert_(vm.id == str(deployVmResponse.id), "Vm deployed is different from the test")
|
||||||
|
self.assert_(vm.state == "Running", "VM is not in Running state")
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def tearDownClass(cls):
|
||||||
|
try:
|
||||||
|
#Cleanup resources used
|
||||||
|
cleanup_resources(cls.apiClient, cls.cleanup)
|
||||||
|
except Exception as e:
|
||||||
|
raise Exception("Warning: Exception during cleanup : %s" % e)
|
||||||
@ -110,16 +110,15 @@ class cloudConnection(object):
|
|||||||
self.logging.info("Computed Signature by Marvin: %s" % signature)
|
self.logging.info("Computed Signature by Marvin: %s" % signature)
|
||||||
return signature
|
return signature
|
||||||
|
|
||||||
def request(self, command, auth=True, payload={}, data={}):
|
def request(self, command, auth=True, payload={}, method='GET'):
|
||||||
"""
|
"""
|
||||||
Makes requests using auth or over integration port
|
Makes requests using auth or over integration port
|
||||||
@param command: cloudstack API command name
|
@param command: cloudstack API command name
|
||||||
eg: deployVirtualMachineCommand
|
eg: deployVirtualMachineCommand
|
||||||
@param auth: Authentication (apikey,secretKey) => True
|
@param auth: Authentication (apikey,secretKey) => True
|
||||||
else False for integration.api.port
|
else False for integration.api.port
|
||||||
@param payload: GET param data composed as a dictionary
|
@param payload: request data composed as a dictionary
|
||||||
of key,value pairs
|
@param method: GET/POST via HTTP
|
||||||
@param data: POST data as a dictionary
|
|
||||||
@return:
|
@return:
|
||||||
"""
|
"""
|
||||||
payload["command"] = command
|
payload["command"] = command
|
||||||
@ -131,9 +130,8 @@ class cloudConnection(object):
|
|||||||
payload["signature"] = signature
|
payload["signature"] = signature
|
||||||
|
|
||||||
try:
|
try:
|
||||||
if data:
|
if method == 'POST':
|
||||||
response = requests.get(self.baseurl, params=payload,
|
response = requests.post(self.baseurl, params=payload)
|
||||||
data=data)
|
|
||||||
else:
|
else:
|
||||||
response = requests.get(self.baseurl, params=payload)
|
response = requests.get(self.baseurl, params=payload)
|
||||||
except ConnectionError, c:
|
except ConnectionError, c:
|
||||||
@ -193,25 +191,19 @@ class cloudConnection(object):
|
|||||||
i = i + 1
|
i = i + 1
|
||||||
return cmdname, isAsync, requests
|
return cmdname, isAsync, requests
|
||||||
|
|
||||||
def marvin_request(self, cmd, data={}, response_type=None):
|
def marvin_request(self, cmd, response_type=None, method='GET'):
|
||||||
"""
|
"""
|
||||||
Requester for marvin command objects
|
Requester for marvin command objects
|
||||||
@param cmd: marvin's command from cloudstackAPI
|
@param cmd: marvin's command from cloudstackAPI
|
||||||
@param data: any data to be sent in as POST
|
|
||||||
@param response_type: response type of the command in cmd
|
@param response_type: response type of the command in cmd
|
||||||
@param raw:
|
@param method: HTTP GET/POST, defaults to GET
|
||||||
@return:
|
@return:
|
||||||
"""
|
"""
|
||||||
cmdname, isAsync, payload = self.sanitize_command(cmd)
|
cmdname, isAsync, payload = self.sanitize_command(cmd)
|
||||||
self.logging.info("sending command: %s %s" % (cmdname, str(payload)))
|
self.logging.debug("sending %s request: %s %s" % (method, cmdname, str(payload)))
|
||||||
if self.auth:
|
|
||||||
response = self.request(
|
response = self.request(
|
||||||
cmdname, auth=True, payload=payload, data=data)
|
cmdname, self.auth, payload=payload, method=method)
|
||||||
else:
|
self.logging.debug("Request: %s Response: %s" %
|
||||||
response = self.request(
|
|
||||||
cmdname, auth=False, payload=payload, data=data)
|
|
||||||
|
|
||||||
self.logging.info("Request: %s Response: %s" %
|
|
||||||
(response.url, response.text))
|
(response.url, response.text))
|
||||||
response = jsonHelper.getResultObj(response.json(), response_type)
|
response = jsonHelper.getResultObj(response.json(), response_type)
|
||||||
|
|
||||||
|
|||||||
@ -184,9 +184,9 @@ class codeGenerator:
|
|||||||
body += "\n"
|
body += "\n"
|
||||||
|
|
||||||
for cmdName in self.cmdsName:
|
for cmdName in self.cmdsName:
|
||||||
body += self.space + 'def %s(self, command, postdata={}):\n'%cmdName
|
body += self.space + 'def %s(self, command, method="GET"):\n'%cmdName
|
||||||
body += self.space + self.space + 'response = %sResponse()\n'%cmdName
|
body += self.space + self.space + 'response = %sResponse()\n'%cmdName
|
||||||
body += self.space + self.space + 'response = self.connection.marvin_request(command, data=postdata, response_type=response)\n'
|
body += self.space + self.space + 'response = self.connection.marvin_request(command, response_type=response, method=method)\n'
|
||||||
body += self.space + self.space + 'return response\n'
|
body += self.space + self.space + 'return response\n'
|
||||||
body += '\n'
|
body += '\n'
|
||||||
|
|
||||||
|
|||||||
@ -102,7 +102,7 @@ class Account:
|
|||||||
def delete(self, apiclient):
|
def delete(self, apiclient):
|
||||||
"""Delete an account"""
|
"""Delete an account"""
|
||||||
cmd = deleteAccount.deleteAccountCmd()
|
cmd = deleteAccount.deleteAccountCmd()
|
||||||
cmd.id = self.account.id
|
cmd.id = self.id
|
||||||
apiclient.deleteAccount(cmd)
|
apiclient.deleteAccount(cmd)
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
@ -220,7 +220,7 @@ class VirtualMachine:
|
|||||||
def create(cls, apiclient, services, templateid=None, accountid=None,
|
def create(cls, apiclient, services, templateid=None, accountid=None,
|
||||||
domainid=None, zoneid=None, networkids=None, serviceofferingid=None,
|
domainid=None, zoneid=None, networkids=None, serviceofferingid=None,
|
||||||
securitygroupids=None, projectid=None, startvm=None,
|
securitygroupids=None, projectid=None, startvm=None,
|
||||||
diskofferingid=None, affinitygroupnames=None, hostid=None, mode='basic'):
|
diskofferingid=None, affinitygroupnames=None, hostid=None, mode='basic', method='GET'):
|
||||||
"""Create the instance"""
|
"""Create the instance"""
|
||||||
|
|
||||||
cmd = deployVirtualMachine.deployVirtualMachineCmd()
|
cmd = deployVirtualMachine.deployVirtualMachineCmd()
|
||||||
@ -262,8 +262,6 @@ class VirtualMachine:
|
|||||||
if securitygroupids:
|
if securitygroupids:
|
||||||
cmd.securitygroupids = [str(sg_id) for sg_id in securitygroupids]
|
cmd.securitygroupids = [str(sg_id) for sg_id in securitygroupids]
|
||||||
|
|
||||||
if "userdata" in services:
|
|
||||||
cmd.userdata = base64.b64encode(services["userdata"])
|
|
||||||
|
|
||||||
if "affinitygroupnames" in services:
|
if "affinitygroupnames" in services:
|
||||||
cmd.affinitygroupnames = services["affinitygroupnames"]
|
cmd.affinitygroupnames = services["affinitygroupnames"]
|
||||||
@ -279,7 +277,10 @@ class VirtualMachine:
|
|||||||
if hostid:
|
if hostid:
|
||||||
cmd.hostid = hostid
|
cmd.hostid = hostid
|
||||||
|
|
||||||
virtual_machine = apiclient.deployVirtualMachine(cmd)
|
if "userdata" in services:
|
||||||
|
cmd.userdata = base64.b64encode(services["userdata"])
|
||||||
|
|
||||||
|
virtual_machine = apiclient.deployVirtualMachine(cmd, method=method)
|
||||||
|
|
||||||
# VM should be in Running state after deploy
|
# VM should be in Running state after deploy
|
||||||
timeout = 10
|
timeout = 10
|
||||||
|
|||||||
Loading…
x
Reference in New Issue
Block a user