* Updated readme file with installation of suds-0.4 through easy_install.

* Removed pass functions
* Fixed pep8 errors
* Few bug fixes and other commits

Also rebased this branch to nova revision 761
This commit is contained in:
sateesh 2011-03-07 18:10:48 +05:30
commit 67e63ba87f
55 changed files with 1677 additions and 690 deletions

View File

@ -15,10 +15,12 @@
<corywright@gmail.com> <cory.wright@rackspace.com> <corywright@gmail.com> <cory.wright@rackspace.com>
<devin.carlen@gmail.com> <devcamcar@illian.local> <devin.carlen@gmail.com> <devcamcar@illian.local>
<ewan.mellor@citrix.com> <emellor@silver> <ewan.mellor@citrix.com> <emellor@silver>
<itoumsn@nttdata.co.jp> <itoumsn@shayol>
<jaypipes@gmail.com> <jpipes@serialcoder> <jaypipes@gmail.com> <jpipes@serialcoder>
<jmckenty@gmail.com> <jmckenty@joshua-mckentys-macbook-pro.local> <jmckenty@gmail.com> <jmckenty@joshua-mckentys-macbook-pro.local>
<jmckenty@gmail.com> <jmckenty@yyj-dhcp171.corp.flock.com> <jmckenty@gmail.com> <jmckenty@yyj-dhcp171.corp.flock.com>
<jmckenty@gmail.com> <joshua.mckenty@nasa.gov> <jmckenty@gmail.com> <joshua.mckenty@nasa.gov>
<josh@jk0.org> <josh.kearney@rackspace.com>
<justin@fathomdb.com> <justinsb@justinsb-desktop> <justin@fathomdb.com> <justinsb@justinsb-desktop>
<justin@fathomdb.com> <superstack@superstack.org> <justin@fathomdb.com> <superstack@superstack.org>
<masumotok@nttdata.co.jp> Masumoto<masumotok@nttdata.co.jp> <masumotok@nttdata.co.jp> Masumoto<masumotok@nttdata.co.jp>

View File

@ -31,7 +31,7 @@ John Dewey <john@dewey.ws>
Jonathan Bryce <jbryce@jbryce.com> Jonathan Bryce <jbryce@jbryce.com>
Jordan Rinke <jordan@openstack.org> Jordan Rinke <jordan@openstack.org>
Josh Durgin <joshd@hq.newdream.net> Josh Durgin <joshd@hq.newdream.net>
Josh Kearney <josh.kearney@rackspace.com> Josh Kearney <josh@jk0.org>
Joshua McKenty <jmckenty@gmail.com> Joshua McKenty <jmckenty@gmail.com>
Justin Santa Barbara <justin@fathomdb.com> Justin Santa Barbara <justin@fathomdb.com>
Kei Masumoto <masumotok@nttdata.co.jp> Kei Masumoto <masumotok@nttdata.co.jp>
@ -39,6 +39,7 @@ Ken Pepple <ken.pepple@gmail.com>
Kevin L. Mitchell <kevin.mitchell@rackspace.com> Kevin L. Mitchell <kevin.mitchell@rackspace.com>
Koji Iida <iida.koji@lab.ntt.co.jp> Koji Iida <iida.koji@lab.ntt.co.jp>
Lorin Hochstein <lorin@isi.edu> Lorin Hochstein <lorin@isi.edu>
Masanori Itoh <itoumsn@nttdata.co.jp>
Matt Dietz <matt.dietz@rackspace.com> Matt Dietz <matt.dietz@rackspace.com>
Michael Gundlach <michael.gundlach@rackspace.com> Michael Gundlach <michael.gundlach@rackspace.com>
Monsyne Dragon <mdragon@rackspace.com> Monsyne Dragon <mdragon@rackspace.com>

View File

@ -84,6 +84,7 @@ from nova import utils
from nova.api.ec2.cloud import ec2_id_to_id from nova.api.ec2.cloud import ec2_id_to_id
from nova.auth import manager from nova.auth import manager
from nova.cloudpipe import pipelib from nova.cloudpipe import pipelib
from nova.compute import instance_types
from nova.db import migration from nova.db import migration
FLAGS = flags.FLAGS FLAGS = flags.FLAGS
@ -661,6 +662,79 @@ class VolumeCommands(object):
"mountpoint": volume['mountpoint']}}) "mountpoint": volume['mountpoint']}})
class InstanceTypeCommands(object):
"""Class for managing instance types / flavors."""
def _print_instance_types(self, n, val):
deleted = ('', ', inactive')[val["deleted"] == 1]
print ("%s: Memory: %sMB, VCPUS: %s, Storage: %sGB, FlavorID: %s, "
"Swap: %sGB, RXTX Quota: %sGB, RXTX Cap: %sMB%s") % (
n, val["memory_mb"], val["vcpus"], val["local_gb"],
val["flavorid"], val["swap"], val["rxtx_quota"],
val["rxtx_cap"], deleted)
def create(self, name, memory, vcpus, local_gb, flavorid,
swap=0, rxtx_quota=0, rxtx_cap=0):
"""Creates instance types / flavors
arguments: name memory vcpus local_gb flavorid [swap] [rxtx_quota]
[rxtx_cap]
"""
try:
instance_types.create(name, memory, vcpus, local_gb,
flavorid, swap, rxtx_quota, rxtx_cap)
except exception.InvalidInputException:
print "Must supply valid parameters to create instance type"
print e
sys.exit(1)
except exception.DBError, e:
print "DB Error: %s" % e
sys.exit(2)
except:
print "Unknown error"
sys.exit(3)
else:
print "%s created" % name
def delete(self, name, purge=None):
"""Marks instance types / flavors as deleted
arguments: name"""
try:
if purge == "--purge":
instance_types.purge(name)
verb = "purged"
else:
instance_types.destroy(name)
verb = "deleted"
except exception.ApiError:
print "Valid instance type name is required"
sys.exit(1)
except exception.DBError, e:
print "DB Error: %s" % e
sys.exit(2)
except:
sys.exit(3)
else:
print "%s %s" % (name, verb)
def list(self, name=None):
"""Lists all active or specific instance types / flavors
arguments: [name]"""
try:
if name == None:
inst_types = instance_types.get_all_types()
elif name == "--all":
inst_types = instance_types.get_all_types(1)
else:
inst_types = instance_types.get_instance_type(name)
except exception.DBError, e:
_db_error(e)
if isinstance(inst_types.values()[0], dict):
for k, v in inst_types.iteritems():
self._print_instance_types(k, v)
else:
self._print_instance_types(name, inst_types)
CATEGORIES = [ CATEGORIES = [
('user', UserCommands), ('user', UserCommands),
('project', ProjectCommands), ('project', ProjectCommands),
@ -673,7 +747,9 @@ CATEGORIES = [
('service', ServiceCommands), ('service', ServiceCommands),
('log', LogCommands), ('log', LogCommands),
('db', DbCommands), ('db', DbCommands),
('volume', VolumeCommands)] ('volume', VolumeCommands),
('instance_type', InstanceTypeCommands),
('flavor', InstanceTypeCommands)]
def lazy_match(name, key_value_tuples): def lazy_match(name, key_value_tuples):

View File

@ -40,6 +40,9 @@ source/api/nova..db.sqlalchemy.migrate_repo.versions.001_austin.rst
source/api/nova..db.sqlalchemy.migrate_repo.versions.002_bexar.rst source/api/nova..db.sqlalchemy.migrate_repo.versions.002_bexar.rst
source/api/nova..db.sqlalchemy.migrate_repo.versions.003_add_label_to_networks.rst source/api/nova..db.sqlalchemy.migrate_repo.versions.003_add_label_to_networks.rst
source/api/nova..db.sqlalchemy.migrate_repo.versions.004_add_zone_tables.rst source/api/nova..db.sqlalchemy.migrate_repo.versions.004_add_zone_tables.rst
source/api/nova..db.sqlalchemy.migrate_repo.versions.005_add_instance_metadata.rst
source/api/nova..db.sqlalchemy.migrate_repo.versions.006_add_provider_data_to_volumes.rst
source/api/nova..db.sqlalchemy.migrate_repo.versions.007_add_instance_types.rst
source/api/nova..db.sqlalchemy.migration.rst source/api/nova..db.sqlalchemy.migration.rst
source/api/nova..db.sqlalchemy.models.rst source/api/nova..db.sqlalchemy.models.rst
source/api/nova..db.sqlalchemy.session.rst source/api/nova..db.sqlalchemy.session.rst
@ -98,6 +101,7 @@ source/api/nova..tests.test_compute.rst
source/api/nova..tests.test_console.rst source/api/nova..tests.test_console.rst
source/api/nova..tests.test_direct.rst source/api/nova..tests.test_direct.rst
source/api/nova..tests.test_flags.rst source/api/nova..tests.test_flags.rst
source/api/nova..tests.test_instance_types.rst
source/api/nova..tests.test_localization.rst source/api/nova..tests.test_localization.rst
source/api/nova..tests.test_log.rst source/api/nova..tests.test_log.rst
source/api/nova..tests.test_middleware.rst source/api/nova..tests.test_middleware.rst
@ -107,7 +111,9 @@ source/api/nova..tests.test_quota.rst
source/api/nova..tests.test_rpc.rst source/api/nova..tests.test_rpc.rst
source/api/nova..tests.test_scheduler.rst source/api/nova..tests.test_scheduler.rst
source/api/nova..tests.test_service.rst source/api/nova..tests.test_service.rst
source/api/nova..tests.test_test.rst
source/api/nova..tests.test_twistd.rst source/api/nova..tests.test_twistd.rst
source/api/nova..tests.test_utils.rst
source/api/nova..tests.test_virt.rst source/api/nova..tests.test_virt.rst
source/api/nova..tests.test_volume.rst source/api/nova..tests.test_volume.rst
source/api/nova..tests.test_xenapi.rst source/api/nova..tests.test_xenapi.rst
@ -176,6 +182,9 @@ source/api/nova..db.sqlalchemy.migrate_repo.versions.001_austin.rst
source/api/nova..db.sqlalchemy.migrate_repo.versions.002_bexar.rst source/api/nova..db.sqlalchemy.migrate_repo.versions.002_bexar.rst
source/api/nova..db.sqlalchemy.migrate_repo.versions.003_add_label_to_networks.rst source/api/nova..db.sqlalchemy.migrate_repo.versions.003_add_label_to_networks.rst
source/api/nova..db.sqlalchemy.migrate_repo.versions.004_add_zone_tables.rst source/api/nova..db.sqlalchemy.migrate_repo.versions.004_add_zone_tables.rst
source/api/nova..db.sqlalchemy.migrate_repo.versions.005_add_instance_metadata.rst
source/api/nova..db.sqlalchemy.migrate_repo.versions.006_add_provider_data_to_volumes.rst
source/api/nova..db.sqlalchemy.migrate_repo.versions.007_add_instance_types.rst
source/api/nova..db.sqlalchemy.migration.rst source/api/nova..db.sqlalchemy.migration.rst
source/api/nova..db.sqlalchemy.models.rst source/api/nova..db.sqlalchemy.models.rst
source/api/nova..db.sqlalchemy.session.rst source/api/nova..db.sqlalchemy.session.rst
@ -234,6 +243,7 @@ source/api/nova..tests.test_compute.rst
source/api/nova..tests.test_console.rst source/api/nova..tests.test_console.rst
source/api/nova..tests.test_direct.rst source/api/nova..tests.test_direct.rst
source/api/nova..tests.test_flags.rst source/api/nova..tests.test_flags.rst
source/api/nova..tests.test_instance_types.rst
source/api/nova..tests.test_localization.rst source/api/nova..tests.test_localization.rst
source/api/nova..tests.test_log.rst source/api/nova..tests.test_log.rst
source/api/nova..tests.test_middleware.rst source/api/nova..tests.test_middleware.rst
@ -243,142 +253,9 @@ source/api/nova..tests.test_quota.rst
source/api/nova..tests.test_rpc.rst source/api/nova..tests.test_rpc.rst
source/api/nova..tests.test_scheduler.rst source/api/nova..tests.test_scheduler.rst
source/api/nova..tests.test_service.rst source/api/nova..tests.test_service.rst
source/api/nova..tests.test_test.rst
source/api/nova..tests.test_twistd.rst source/api/nova..tests.test_twistd.rst
source/api/nova..tests.test_virt.rst source/api/nova..tests.test_utils.rst
source/api/nova..tests.test_volume.rst
source/api/nova..tests.test_xenapi.rst
source/api/nova..tests.xenapi.stubs.rst
source/api/nova..twistd.rst
source/api/nova..utils.rst
source/api/nova..version.rst
source/api/nova..virt.connection.rst
source/api/nova..virt.disk.rst
source/api/nova..virt.fake.rst
source/api/nova..virt.hyperv.rst
source/api/nova..virt.images.rst
source/api/nova..virt.libvirt_conn.rst
source/api/nova..virt.xenapi.fake.rst
source/api/nova..virt.xenapi.network_utils.rst
source/api/nova..virt.xenapi.vm_utils.rst
source/api/nova..virt.xenapi.vmops.rst
source/api/nova..virt.xenapi.volume_utils.rst
source/api/nova..virt.xenapi.volumeops.rst
source/api/nova..virt.xenapi_conn.rst
source/api/nova..volume.api.rst
source/api/nova..volume.driver.rst
source/api/nova..volume.manager.rst
source/api/nova..volume.san.rst
source/api/nova..wsgi.rst
source/api/nova..adminclient.rst
source/api/nova..api.direct.rst
source/api/nova..api.ec2.admin.rst
source/api/nova..api.ec2.apirequest.rst
source/api/nova..api.ec2.cloud.rst
source/api/nova..api.ec2.metadatarequesthandler.rst
source/api/nova..api.openstack.auth.rst
source/api/nova..api.openstack.backup_schedules.rst
source/api/nova..api.openstack.common.rst
source/api/nova..api.openstack.consoles.rst
source/api/nova..api.openstack.faults.rst
source/api/nova..api.openstack.flavors.rst
source/api/nova..api.openstack.images.rst
source/api/nova..api.openstack.servers.rst
source/api/nova..api.openstack.shared_ip_groups.rst
source/api/nova..api.openstack.zones.rst
source/api/nova..auth.dbdriver.rst
source/api/nova..auth.fakeldap.rst
source/api/nova..auth.ldapdriver.rst
source/api/nova..auth.manager.rst
source/api/nova..auth.signer.rst
source/api/nova..cloudpipe.pipelib.rst
source/api/nova..compute.api.rst
source/api/nova..compute.instance_types.rst
source/api/nova..compute.manager.rst
source/api/nova..compute.monitor.rst
source/api/nova..compute.power_state.rst
source/api/nova..console.api.rst
source/api/nova..console.fake.rst
source/api/nova..console.manager.rst
source/api/nova..console.xvp.rst
source/api/nova..context.rst
source/api/nova..crypto.rst
source/api/nova..db.api.rst
source/api/nova..db.base.rst
source/api/nova..db.migration.rst
source/api/nova..db.sqlalchemy.api.rst
source/api/nova..db.sqlalchemy.migrate_repo.manage.rst
source/api/nova..db.sqlalchemy.migrate_repo.versions.001_austin.rst
source/api/nova..db.sqlalchemy.migrate_repo.versions.002_bexar.rst
source/api/nova..db.sqlalchemy.migrate_repo.versions.003_add_label_to_networks.rst
source/api/nova..db.sqlalchemy.migrate_repo.versions.004_add_zone_tables.rst
source/api/nova..db.sqlalchemy.migration.rst
source/api/nova..db.sqlalchemy.models.rst
source/api/nova..db.sqlalchemy.session.rst
source/api/nova..exception.rst
source/api/nova..fakememcache.rst
source/api/nova..fakerabbit.rst
source/api/nova..flags.rst
source/api/nova..image.glance.rst
source/api/nova..image.local.rst
source/api/nova..image.s3.rst
source/api/nova..image.service.rst
source/api/nova..log.rst
source/api/nova..manager.rst
source/api/nova..network.api.rst
source/api/nova..network.linux_net.rst
source/api/nova..network.manager.rst
source/api/nova..objectstore.bucket.rst
source/api/nova..objectstore.handler.rst
source/api/nova..objectstore.image.rst
source/api/nova..objectstore.stored.rst
source/api/nova..quota.rst
source/api/nova..rpc.rst
source/api/nova..scheduler.chance.rst
source/api/nova..scheduler.driver.rst
source/api/nova..scheduler.manager.rst
source/api/nova..scheduler.simple.rst
source/api/nova..scheduler.zone.rst
source/api/nova..service.rst
source/api/nova..test.rst
source/api/nova..tests.api.openstack.fakes.rst
source/api/nova..tests.api.openstack.test_adminapi.rst
source/api/nova..tests.api.openstack.test_api.rst
source/api/nova..tests.api.openstack.test_auth.rst
source/api/nova..tests.api.openstack.test_common.rst
source/api/nova..tests.api.openstack.test_faults.rst
source/api/nova..tests.api.openstack.test_flavors.rst
source/api/nova..tests.api.openstack.test_images.rst
source/api/nova..tests.api.openstack.test_ratelimiting.rst
source/api/nova..tests.api.openstack.test_servers.rst
source/api/nova..tests.api.openstack.test_shared_ip_groups.rst
source/api/nova..tests.api.openstack.test_zones.rst
source/api/nova..tests.api.test_wsgi.rst
source/api/nova..tests.db.fakes.rst
source/api/nova..tests.declare_flags.rst
source/api/nova..tests.fake_flags.rst
source/api/nova..tests.glance.stubs.rst
source/api/nova..tests.hyperv_unittest.rst
source/api/nova..tests.objectstore_unittest.rst
source/api/nova..tests.real_flags.rst
source/api/nova..tests.runtime_flags.rst
source/api/nova..tests.test_access.rst
source/api/nova..tests.test_api.rst
source/api/nova..tests.test_auth.rst
source/api/nova..tests.test_cloud.rst
source/api/nova..tests.test_compute.rst
source/api/nova..tests.test_console.rst
source/api/nova..tests.test_direct.rst
source/api/nova..tests.test_flags.rst
source/api/nova..tests.test_localization.rst
source/api/nova..tests.test_log.rst
source/api/nova..tests.test_middleware.rst
source/api/nova..tests.test_misc.rst
source/api/nova..tests.test_network.rst
source/api/nova..tests.test_quota.rst
source/api/nova..tests.test_rpc.rst
source/api/nova..tests.test_scheduler.rst
source/api/nova..tests.test_service.rst
source/api/nova..tests.test_twistd.rst
source/api/nova..tests.test_virt.rst source/api/nova..tests.test_virt.rst
source/api/nova..tests.test_volume.rst source/api/nova..tests.test_volume.rst
source/api/nova..tests.test_xenapi.rst source/api/nova..tests.test_xenapi.rst

View File

@ -43,6 +43,9 @@
nova..db.sqlalchemy.migrate_repo.versions.002_bexar.rst nova..db.sqlalchemy.migrate_repo.versions.002_bexar.rst
nova..db.sqlalchemy.migrate_repo.versions.003_add_label_to_networks.rst nova..db.sqlalchemy.migrate_repo.versions.003_add_label_to_networks.rst
nova..db.sqlalchemy.migrate_repo.versions.004_add_zone_tables.rst nova..db.sqlalchemy.migrate_repo.versions.004_add_zone_tables.rst
nova..db.sqlalchemy.migrate_repo.versions.005_add_instance_metadata.rst
nova..db.sqlalchemy.migrate_repo.versions.006_add_provider_data_to_volumes.rst
nova..db.sqlalchemy.migrate_repo.versions.007_add_instance_types.rst
nova..db.sqlalchemy.migration.rst nova..db.sqlalchemy.migration.rst
nova..db.sqlalchemy.models.rst nova..db.sqlalchemy.models.rst
nova..db.sqlalchemy.session.rst nova..db.sqlalchemy.session.rst
@ -101,6 +104,7 @@
nova..tests.test_console.rst nova..tests.test_console.rst
nova..tests.test_direct.rst nova..tests.test_direct.rst
nova..tests.test_flags.rst nova..tests.test_flags.rst
nova..tests.test_instance_types.rst
nova..tests.test_localization.rst nova..tests.test_localization.rst
nova..tests.test_log.rst nova..tests.test_log.rst
nova..tests.test_middleware.rst nova..tests.test_middleware.rst
@ -110,7 +114,9 @@
nova..tests.test_rpc.rst nova..tests.test_rpc.rst
nova..tests.test_scheduler.rst nova..tests.test_scheduler.rst
nova..tests.test_service.rst nova..tests.test_service.rst
nova..tests.test_test.rst
nova..tests.test_twistd.rst nova..tests.test_twistd.rst
nova..tests.test_utils.rst
nova..tests.test_virt.rst nova..tests.test_virt.rst
nova..tests.test_volume.rst nova..tests.test_volume.rst
nova..tests.test_xenapi.rst nova..tests.test_xenapi.rst

View File

@ -0,0 +1,6 @@
The :mod:`nova..db.sqlalchemy.migrate_repo.versions.005_add_instance_metadata` Module
==============================================================================
.. automodule:: nova..db.sqlalchemy.migrate_repo.versions.005_add_instance_metadata
:members:
:undoc-members:
:show-inheritance:

View File

@ -0,0 +1,6 @@
The :mod:`nova..db.sqlalchemy.migrate_repo.versions.006_add_provider_data_to_volumes` Module
==============================================================================
.. automodule:: nova..db.sqlalchemy.migrate_repo.versions.006_add_provider_data_to_volumes
:members:
:undoc-members:
:show-inheritance:

View File

@ -0,0 +1,6 @@
The :mod:`nova..db.sqlalchemy.migrate_repo.versions.007_add_instance_types` Module
==============================================================================
.. automodule:: nova..db.sqlalchemy.migrate_repo.versions.007_add_instance_types
:members:
:undoc-members:
:show-inheritance:

View File

@ -0,0 +1,6 @@
The :mod:`nova..tests.test_instance_types` Module
==============================================================================
.. automodule:: nova..tests.test_instance_types
:members:
:undoc-members:
:show-inheritance:

View File

@ -0,0 +1,6 @@
The :mod:`nova..tests.test_test` Module
==============================================================================
.. automodule:: nova..tests.test_test
:members:
:undoc-members:
:show-inheritance:

View File

@ -0,0 +1,6 @@
The :mod:`nova..tests.test_utils` Module
==============================================================================
.. automodule:: nova..tests.test_utils
:members:
:undoc-members:
:show-inheritance:

View File

@ -179,6 +179,42 @@ Nova Floating IPs
Displays a list of all floating IP addresses. Displays a list of all floating IP addresses.
Nova Flavor
~~~~~~~~~~~
``nova-manage flavor list``
Outputs a list of all active flavors to the screen.
``nova-manage flavor list --all``
Outputs a list of all flavors (active and inactive) to the screen.
``nova-manage flavor create <name> <memory> <vCPU> <local_storage> <flavorID> <(optional) swap> <(optional) RXTX Quota> <(optional) RXTX Cap>``
creates a flavor with the following positional arguments:
* memory (expressed in megabytes)
* vcpu(s) (integer)
* local storage (expressed in gigabytes)
* flavorid (unique integer)
* swap space (expressed in megabytes, defaults to zero, optional)
* RXTX quotas (expressed in gigabytes, defaults to zero, optional)
* RXTX cap (expressed in gigabytes, defaults to zero, optional)
``nova-manage flavor delete <name>``
Delete the flavor with the name <name>. This marks the flavor as inactive and cannot be launched. However, the record stays in the database for archival and billing purposes.
``nova-manage flavor delete <name> --purge``
Purges the flavor with the name <name>. This removes this flavor from the database.
Nova Instance_type
~~~~~~~~~~~~~~~~~~
The instance_type command is provided as an alias for the flavor command. All the same subcommands and arguments from nova-manage flavor can be used.
FILES FILES
======== ========

View File

@ -64,6 +64,11 @@ Concept: Instances
An 'instance' is a word for a virtual machine that runs inside the cloud. An 'instance' is a word for a virtual machine that runs inside the cloud.
Concept: Instance Type
----------------------
An 'instance type' describes the compute, memory and storage capacity of nova computing instances. In layman terms, this is the size (in terms of vCPUs, RAM, etc.) of the virtual server that you will be launching.
Concept: System Architecture Concept: System Architecture
---------------------------- ----------------------------

View File

@ -0,0 +1,84 @@
..
Copyright 2011 Ken Pepple
Licensed 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.
Managing Instance Types and Flavors
===================================
What are Instance Types or Flavors ?
------------------------------------
Instance types describe the compute, memory and storage capacity of nova computing instances. In layman terms, this is the size (in terms of vCPUs, RAM, etc.) of the virtual server that you will be launching. In the EC2 API, these are called by names such as "m1.large" or "m1.tiny", while the OpenStack API terms these "flavors" with names like "512 MB Server".
In Nova, "flavor" and "instance type" are equivalent terms. When you create an EC2 instance type, you are also creating a OpenStack API flavor. To reduce repetition, for the rest of this document I will refer to these as instance types.
Instance types can be in either the active or inactive state:
* Active instance types are available to be used for launching instances
* Inactive instance types are not available for launching instances
In the current (Cactus) version of nova, instance types can only be created by the nova administrator through the nova-manage command. Future versions of nova (in concert with the OpenStack API or EC2 API), may expose this functionality directly to users.
Basic Management
----------------
Instance types / flavor are managed through the nova-manage binary with
the "instance_type" command and an appropriate subcommand. Note that you can also use
the "flavor" command as a synonym for "instance_types".
To see all currently active instance types, use the list subcommand::
# nova-manage instance_type list
m1.medium: Memory: 4096MB, VCPUS: 2, Storage: 40GB, FlavorID: 3, Swap: 0GB, RXTX Quota: 0GB, RXTX Cap: 0MB
m1.large: Memory: 8192MB, VCPUS: 4, Storage: 80GB, FlavorID: 4, Swap: 0GB, RXTX Quota: 0GB, RXTX Cap: 0MB
m1.tiny: Memory: 512MB, VCPUS: 1, Storage: 0GB, FlavorID: 1, Swap: 0GB, RXTX Quota: 0GB, RXTX Cap: 0MB
m1.xlarge: Memory: 16384MB, VCPUS: 8, Storage: 160GB, FlavorID: 5, Swap: 0GB, RXTX Quota: 0GB, RXTX Cap: 0MB
m1.small: Memory: 2048MB, VCPUS: 1, Storage: 20GB, FlavorID: 2, Swap: 0GB, RXTX Quota: 0GB, RXTX Cap: 0MB
By default, the list subcommand only shows active instance types. To see all instance types (inactive and active), use the list subcommand with the "--all" flag::
# nova-manage instance_type list --all
m1.medium: Memory: 4096MB, VCPUS: 2, Storage: 40GB, FlavorID: 3, Swap: 0GB, RXTX Quota: 0GB, RXTX Cap: 0MB
m1.large: Memory: 8192MB, VCPUS: 4, Storage: 80GB, FlavorID: 4, Swap: 0GB, RXTX Quota: 0GB, RXTX Cap: 0MB
m1.tiny: Memory: 512MB, VCPUS: 1, Storage: 0GB, FlavorID: 1, Swap: 0GB, RXTX Quota: 0GB, RXTX Cap: 0MB
m1.xlarge: Memory: 16384MB, VCPUS: 8, Storage: 160GB, FlavorID: 5, Swap: 0GB, RXTX Quota: 0GB, RXTX Cap: 0MB
m1.small: Memory: 2048MB, VCPUS: 1, Storage: 20GB, FlavorID: 2, Swap: 0GB, RXTX Quota: 0GB, RXTX Cap: 0MB
m1.deleted: Memory: 2048MB, VCPUS: 1, Storage: 20GB, FlavorID: 2, Swap: 0GB, RXTX Quota: 0GB, RXTX Cap: 0MB, inactive
To create an instance type, use the "create" subcommand with the following positional arguments:
* memory (expressed in megabytes)
* vcpu(s) (integer)
* local storage (expressed in gigabytes)
* flavorid (unique integer)
* swap space (expressed in megabytes, defaults to zero, optional)
* RXTX quotas (expressed in gigabytes, defaults to zero, optional)
* RXTX cap (expressed in gigabytes, defaults to zero, optional)
The following example creates an instance type named "m1.xxlarge"::
# nova-manage instance_type create m1.xxlarge 32768 16 320 0 0 0
m1.xxlarge created
To delete an instance type, use the "delete" subcommand and specify the name::
# nova-manage instance_type delete m1.xxlarge
m1.xxlarge deleted
Please note that the "delete" command only marks the instance type as
inactive in the database; it does not actually remove the instance type. This is done
to preserve the instance type definition for long running instances (which may not
terminate for months or years). If you are sure that you want to delete this instance
type from the database, pass the "--purge" flag after the name::
# nova-manage instance_type delete m1.xxlarge --purge
m1.xxlarge purged

View File

@ -55,11 +55,8 @@ Python dependencies
:: ::
sudo apt-get install python-setuptools easy_install suds==0.4
wget https://fedorahosted.org/releases/s/u/suds/python-suds-0.4.tar.gz
tar -zxvf python-suds-0.4.tar.gz
cd python-suds-0.4
sudo python setup.py install
Configuration flags required for nova-compute Configuration flags required for nova-compute
--------------------------------------------- ---------------------------------------------

View File

@ -78,20 +78,27 @@ def _bytes2int(bytes):
def _parse_network_details(machine_id): def _parse_network_details(machine_id):
"""Parse the machine.id field to get MAC, IP, Netmask and Gateway fields """Parse the machine.id field to get MAC, IP, Netmask and Gateway fields
machine.id is of the form MAC;IP;Netmask;Gateway; where ';' is machine.id is of the form MAC;IP;Netmask;Gateway;Broadcast;DNS1,DNS2
the separator. where ';' is the separator.
""" """
network_details = [] network_details = []
if machine_id[1].strip() == NO_MACHINE_ID: if machine_id[1].strip() == "1":
pass pass
else: else:
network_info_list = machine_id[0].split(';') network_info_list = machine_id[0].split(';')
assert len(network_info_list) % 4 == 0 assert len(network_info_list) % 6 == 0
for i in xrange(0, len(network_info_list) / 4): no_grps = len(network_info_list) / 6
network_details.append((network_info_list[i].strip().lower(), i = 0
network_info_list[i + 1].strip(), while i < no_grps:
network_info_list[i + 2].strip(), k = i * 6
network_info_list[i + 3].strip())) network_details.append((
network_info_list[k].strip().lower(),
network_info_list[k + 1].strip(),
network_info_list[k + 2].strip(),
network_info_list[k + 3].strip(),
network_info_list[k + 4].strip(),
network_info_list[k + 5].strip().split(',')))
i += 1
return network_details return network_details
@ -218,7 +225,7 @@ def _execute(cmd_list, process_input=None, check_exit_code=True):
return result return result
def _windows_set_ipaddress(): def _windows_set_networking():
"""Set IP address for the windows VM""" """Set IP address for the windows VM"""
program_files = os.environ.get('PROGRAMFILES') program_files = os.environ.get('PROGRAMFILES')
program_files_x86 = os.environ.get('PROGRAMFILES(X86)') program_files_x86 = os.environ.get('PROGRAMFILES(X86)')
@ -240,7 +247,8 @@ def _windows_set_ipaddress():
cmd = ['"' + vmware_tools_bin + '"', '--cmd', 'machine.id.get'] cmd = ['"' + vmware_tools_bin + '"', '--cmd', 'machine.id.get']
for network_detail in _parse_network_details(_execute(cmd, for network_detail in _parse_network_details(_execute(cmd,
check_exit_code=False)): check_exit_code=False)):
mac_address, ip_address, subnet_mask, gateway = network_detail mac_address, ip_address, subnet_mask, gateway, broadcast,\
dns_servers = network_detail
adapter_name, current_ip_address = \ adapter_name, current_ip_address = \
_get_win_adapter_name_and_ip_address(mac_address) _get_win_adapter_name_and_ip_address(mac_address)
if adapter_name and not ip_address == current_ip_address: if adapter_name and not ip_address == current_ip_address:
@ -248,11 +256,65 @@ def _windows_set_ipaddress():
'name="%s"' % adapter_name, 'source=static', ip_address, 'name="%s"' % adapter_name, 'source=static', ip_address,
subnet_mask, gateway, '1'] subnet_mask, gateway, '1']
_execute(cmd) _execute(cmd)
#Windows doesn't let you manually set the broadcast address
for dns_server in dns_servers:
if dns_server:
cmd = ['netsh', 'interface', 'ip', 'add', 'dns',
'name="%s"' % adapter_name, dns_server]
_execute(cmd)
else: else:
logging.warn(_("VMware Tools is not installed")) logging.warn(_("VMware Tools is not installed"))
def _linux_set_ipaddress(): def _filter_duplicates(all_entries):
final_list = []
for entry in all_entries:
if entry and entry not in final_list:
final_list.append(entry)
return final_list
def _set_rhel_networking(network_details=[]):
all_dns_servers = []
for network_detail in network_details:
mac_address, ip_address, subnet_mask, gateway, broadcast,\
dns_servers = network_detail
all_dns_servers.extend(dns_servers)
adapter_name, current_ip_address = \
_get_linux_adapter_name_and_ip_address(mac_address)
if adapter_name and not ip_address == current_ip_address:
interface_file_name = \
'/etc/sysconfig/network-scripts/ifcfg-%s' % adapter_name
#Remove file
os.remove(interface_file_name)
#Touch file
_execute(['touch', interface_file_name])
interface_file = open(interface_file_name, 'w')
interface_file.write('\nDEVICE=%s' % adapter_name)
interface_file.write('\nUSERCTL=yes')
interface_file.write('\nONBOOT=yes')
interface_file.write('\nBOOTPROTO=static')
interface_file.write('\nBROADCAST=%s' % broadcast)
interface_file.write('\nNETWORK=')
interface_file.write('\nGATEWAY=%s' % gateway)
interface_file.write('\nNETMASK=%s' % subnet_mask)
interface_file.write('\nIPADDR=%s' % ip_address)
interface_file.write('\nMACADDR=%s' % mac_address)
interface_file.close()
if all_dns_servers:
dns_file_name = "/etc/resolv.conf"
os.remove(dns_file_name)
_execute(['touch', dns_file_name])
dns_file = open(dns_file_name, 'w')
dns_file.write("; generated by OpenStack guest tools")
unique_entries = _filter_duplicates(all_dns_servers)
for dns_server in unique_entries:
dns_file.write("\nnameserver %s" % dns_server)
dns_file.close()
_execute(['/sbin/service', 'network', 'restart'])
def _linux_set_networking():
"""Set IP address for the Linux VM""" """Set IP address for the Linux VM"""
vmware_tools_bin = None vmware_tools_bin = None
if os.path.exists('/usr/sbin/vmtoolsd'): if os.path.exists('/usr/sbin/vmtoolsd'):
@ -265,38 +327,18 @@ def _linux_set_ipaddress():
vmware_tools_bin = '/usr/bin/vmware-guestd' vmware_tools_bin = '/usr/bin/vmware-guestd'
if vmware_tools_bin: if vmware_tools_bin:
cmd = [vmware_tools_bin, '--cmd', 'machine.id.get'] cmd = [vmware_tools_bin, '--cmd', 'machine.id.get']
for network_detail in _parse_network_details(_execute(cmd, network_details = _parse_network_details(_execute(cmd,
check_exit_code=False)): check_exit_code=False))
mac_address, ip_address, subnet_mask, gateway = network_detail #TODO: For other distros like ubuntu, suse, debian, BSD, etc.
adapter_name, current_ip_address = \ _set_rhel_networking(network_details)
_get_linux_adapter_name_and_ip_address(mac_address)
if adapter_name and not ip_address == current_ip_address:
interface_file_name = \
'/etc/sysconfig/network-scripts/ifcfg-%s' % adapter_name
#Remove file
os.remove(interface_file_name)
#Touch file
_execute(['touch', interface_file_name])
interface_file = open(interface_file_name, 'w')
interface_file.write('\nDEVICE=%s' % adapter_name)
interface_file.write('\nUSERCTL=yes')
interface_file.write('\nONBOOT=yes')
interface_file.write('\nBOOTPROTO=static')
interface_file.write('\nBROADCAST=')
interface_file.write('\nNETWORK=')
interface_file.write('\nNETMASK=%s' % subnet_mask)
interface_file.write('\nIPADDR=%s' % ip_address)
interface_file.write('\nMACADDR=%s' % mac_address)
interface_file.close()
_execute(['/sbin/service', 'network' 'restart'])
else: else:
logging.warn(_("VMware Tools is not installed")) logging.warn(_("VMware Tools is not installed"))
if __name__ == '__main__': if __name__ == '__main__':
pltfrm = sys.platform pltfrm = sys.platform
if pltfrm == PLATFORM_WIN: if pltfrm == PLATFORM_WIN:
_windows_set_ipaddress() _windows_set_networking()
elif pltfrm == PLATFORM_LINUX: elif pltfrm == PLATFORM_LINUX:
_linux_set_ipaddress() _linux_set_networking()
else: else:
raise NotImplementedError(_("Platform not implemented: '%s'") % pltfrm) raise NotImplementedError(_("Platform not implemented: '%s'") % pltfrm)

View File

@ -29,7 +29,6 @@ from nova import flags
from nova import log as logging from nova import log as logging
from nova import utils from nova import utils
from nova.auth import manager from nova.auth import manager
from nova.compute import instance_types
FLAGS = flags.FLAGS FLAGS = flags.FLAGS
@ -80,8 +79,8 @@ def host_dict(host, compute_service, instances, volume_service, volumes, now):
return rv return rv
def instance_dict(name, inst): def instance_dict(inst):
return {'name': name, return {'name': inst['name'],
'memory_mb': inst['memory_mb'], 'memory_mb': inst['memory_mb'],
'vcpus': inst['vcpus'], 'vcpus': inst['vcpus'],
'disk_gb': inst['local_gb'], 'disk_gb': inst['local_gb'],
@ -115,9 +114,9 @@ class AdminController(object):
def __str__(self): def __str__(self):
return 'AdminController' return 'AdminController'
def describe_instance_types(self, _context, **_kwargs): def describe_instance_types(self, context, **_kwargs):
return {'instanceTypeSet': [instance_dict(n, v) for n, v in """Returns all active instance types data (vcpus, memory, etc.)"""
instance_types.INSTANCE_TYPES.iteritems()]} return {'instanceTypeSet': [db.instance_type_get_all(context)]}
def describe_user(self, _context, name, **_kwargs): def describe_user(self, _context, name, **_kwargs):
"""Returns user data, including access and secret keys.""" """Returns user data, including access and secret keys."""

View File

@ -52,7 +52,23 @@ def _database_to_isoformat(datetimeobj):
def _try_convert(value): def _try_convert(value):
"""Return a non-string if possible""" """Return a non-string from a string or unicode, if possible.
============= =====================================================
When value is returns
============= =====================================================
zero-length ''
'None' None
'True' True
'False' False
'0', '-0' 0
0xN, -0xN int from hex (postitive) (N is any number)
0bN, -0bN int from binary (positive) (N is any number)
* try conversion to int, float, complex, fallback value
"""
if len(value) == 0:
return ''
if value == 'None': if value == 'None':
return None return None
if value == 'True': if value == 'True':

View File

@ -298,7 +298,7 @@ class CloudController(object):
'keyFingerprint': key_pair['fingerprint'], 'keyFingerprint': key_pair['fingerprint'],
}) })
return {'keypairsSet': result} return {'keySet': result}
def create_key_pair(self, context, key_name, **kwargs): def create_key_pair(self, context, key_name, **kwargs):
LOG.audit(_("Create key pair %s"), key_name, context=context) LOG.audit(_("Create key pair %s"), key_name, context=context)
@ -838,14 +838,14 @@ class CloudController(object):
self.compute_api.unrescue(context, instance_id=instance_id) self.compute_api.unrescue(context, instance_id=instance_id)
return True return True
def update_instance(self, context, ec2_id, **kwargs): def update_instance(self, context, instance_id, **kwargs):
updatable_fields = ['display_name', 'display_description'] updatable_fields = ['display_name', 'display_description']
changes = {} changes = {}
for field in updatable_fields: for field in updatable_fields:
if field in kwargs: if field in kwargs:
changes[field] = kwargs[field] changes[field] = kwargs[field]
if changes: if changes:
instance_id = ec2_id_to_id(ec2_id) instance_id = ec2_id_to_id(instance_id)
self.compute_api.update(context, instance_id=instance_id, **kwargs) self.compute_api.update(context, instance_id=instance_id, **kwargs)
return True return True

View File

@ -74,12 +74,15 @@ class APIRouter(wsgi.Router):
server_members = {'action': 'POST'} server_members = {'action': 'POST'}
if FLAGS.allow_admin_api: if FLAGS.allow_admin_api:
LOG.debug(_("Including admin operations in API.")) LOG.debug(_("Including admin operations in API."))
server_members['pause'] = 'POST' server_members['pause'] = 'POST'
server_members['unpause'] = 'POST' server_members['unpause'] = 'POST'
server_members["diagnostics"] = "GET" server_members["diagnostics"] = "GET"
server_members["actions"] = "GET" server_members["actions"] = "GET"
server_members['suspend'] = 'POST' server_members['suspend'] = 'POST'
server_members['resume'] = 'POST' server_members['resume'] = 'POST'
server_members['rescue'] = 'POST'
server_members['unrescue'] = 'POST'
server_members['reset_network'] = 'POST' server_members['reset_network'] = 'POST'
server_members['inject_network_info'] = 'POST' server_members['inject_network_info'] = 'POST'

View File

@ -15,6 +15,8 @@
# License for the specific language governing permissions and limitations # License for the specific language governing permissions and limitations
# under the License. # under the License.
import webob.exc
from nova import exception from nova import exception
@ -27,7 +29,8 @@ def limited(items, request, max_limit=1000):
GET variables. 'offset' is where to start in the list, GET variables. 'offset' is where to start in the list,
and 'limit' is the maximum number of items to return. If and 'limit' is the maximum number of items to return. If
'limit' is not specified, 0, or > max_limit, we default 'limit' is not specified, 0, or > max_limit, we default
to max_limit. to max_limit. Negative values for either offset or limit
will cause exc.HTTPBadRequest() exceptions to be raised.
@kwarg max_limit: The maximum number of items to return from 'items' @kwarg max_limit: The maximum number of items to return from 'items'
""" """
try: try:
@ -40,6 +43,9 @@ def limited(items, request, max_limit=1000):
except ValueError: except ValueError:
limit = max_limit limit = max_limit
if offset < 0 or limit < 0:
raise webob.exc.HTTPBadRequest()
limit = min(max_limit, limit or max_limit) limit = min(max_limit, limit or max_limit)
range_end = offset + limit range_end = offset + limit
return items[offset:range_end] return items[offset:range_end]

View File

@ -17,6 +17,8 @@
from webob import exc from webob import exc
from nova import db
from nova import context
from nova.api.openstack import faults from nova.api.openstack import faults
from nova.api.openstack import common from nova.api.openstack import common
from nova.compute import instance_types from nova.compute import instance_types
@ -39,19 +41,19 @@ class Controller(wsgi.Controller):
def detail(self, req): def detail(self, req):
"""Return all flavors in detail.""" """Return all flavors in detail."""
items = [self.show(req, id)['flavor'] for id in self._all_ids()] items = [self.show(req, id)['flavor'] for id in self._all_ids(req)]
items = common.limited(items, req)
return dict(flavors=items) return dict(flavors=items)
def show(self, req, id): def show(self, req, id):
"""Return data about the given flavor id.""" """Return data about the given flavor id."""
for name, val in instance_types.INSTANCE_TYPES.iteritems(): ctxt = req.environ['nova.context']
if val['flavorid'] == int(id): values = db.instance_type_get_by_flavor_id(ctxt, id)
item = dict(ram=val['memory_mb'], disk=val['local_gb'], return dict(flavor=values)
id=val['flavorid'], name=name)
return dict(flavor=item)
raise faults.Fault(exc.HTTPNotFound()) raise faults.Fault(exc.HTTPNotFound())
def _all_ids(self): def _all_ids(self, req):
"""Return the list of all flavorids.""" """Return the list of all flavorids."""
return [i['flavorid'] for i in instance_types.INSTANCE_TYPES.values()] ctxt = req.environ['nova.context']
inst_types = db.instance_type_get_all(ctxt)
flavor_ids = [inst_types[i]['flavorid'] for i in inst_types.keys()]
return sorted(flavor_ids)

View File

@ -335,6 +335,28 @@ class Controller(wsgi.Controller):
return faults.Fault(exc.HTTPUnprocessableEntity()) return faults.Fault(exc.HTTPUnprocessableEntity())
return exc.HTTPAccepted() return exc.HTTPAccepted()
def rescue(self, req, id):
"""Permit users to rescue the server."""
context = req.environ["nova.context"]
try:
self.compute_api.rescue(context, id)
except:
readable = traceback.format_exc()
LOG.exception(_("compute.api::rescue %s"), readable)
return faults.Fault(exc.HTTPUnprocessableEntity())
return exc.HTTPAccepted()
def unrescue(self, req, id):
"""Permit users to unrescue the server."""
context = req.environ["nova.context"]
try:
self.compute_api.unrescue(context, id)
except:
readable = traceback.format_exc()
LOG.exception(_("compute.api::unrescue %s"), readable)
return faults.Fault(exc.HTTPUnprocessableEntity())
return exc.HTTPAccepted()
def get_ajax_console(self, req, id): def get_ajax_console(self, req, id):
""" Returns a url to an instance's ajaxterm console. """ """ Returns a url to an instance's ajaxterm console. """
try: try:

View File

@ -88,9 +88,9 @@ class API(base.Base):
availability_zone=None, user_data=None, metadata=[], availability_zone=None, user_data=None, metadata=[],
onset_files=None): onset_files=None):
"""Create the number of instances requested if quota and """Create the number of instances requested if quota and
other arguments check out ok. other arguments check out ok."""
"""
type_data = instance_types.INSTANCE_TYPES[instance_type] type_data = instance_types.get_instance_type(instance_type)
num_instances = quota.allowed_instances(context, max_count, type_data) num_instances = quota.allowed_instances(context, max_count, type_data)
if num_instances < min_count: if num_instances < min_count:
pid = context.project_id pid = context.project_id
@ -319,12 +319,12 @@ class API(base.Base):
try: try:
instance = self.get(context, instance_id) instance = self.get(context, instance_id)
except exception.NotFound: except exception.NotFound:
LOG.warning(_("Instance %d was not found during terminate"), LOG.warning(_("Instance %s was not found during terminate"),
instance_id) instance_id)
raise raise
if (instance['state_description'] == 'terminating'): if (instance['state_description'] == 'terminating'):
LOG.warning(_("Instance %d is already being terminated"), LOG.warning(_("Instance %s is already being terminated"),
instance_id) instance_id)
return return

View File

@ -4,6 +4,7 @@
# Administrator of the National Aeronautics and Space Administration. # Administrator of the National Aeronautics and Space Administration.
# All Rights Reserved. # All Rights Reserved.
# Copyright (c) 2010 Citrix Systems, Inc. # Copyright (c) 2010 Citrix Systems, Inc.
# Copyright 2011 Ken Pepple
# #
# Licensed under the Apache License, Version 2.0 (the "License"); you may # Licensed under the Apache License, Version 2.0 (the "License"); you may
# not use this file except in compliance with the License. You may obtain # not use this file except in compliance with the License. You may obtain
@ -21,30 +22,120 @@
The built-in instance properties. The built-in instance properties.
""" """
from nova import flags from nova import context
from nova import db
from nova import exception from nova import exception
from nova import flags
from nova import log as logging
FLAGS = flags.FLAGS FLAGS = flags.FLAGS
INSTANCE_TYPES = { LOG = logging.getLogger('nova.instance_types')
'm1.tiny': dict(memory_mb=512, vcpus=1, local_gb=0, flavorid=1),
'm1.small': dict(memory_mb=2048, vcpus=1, local_gb=20, flavorid=2),
'm1.medium': dict(memory_mb=4096, vcpus=2, local_gb=40, flavorid=3), def create(name, memory, vcpus, local_gb, flavorid, swap=0,
'm1.large': dict(memory_mb=8192, vcpus=4, local_gb=80, flavorid=4), rxtx_quota=0, rxtx_cap=0):
'm1.xlarge': dict(memory_mb=16384, vcpus=8, local_gb=160, flavorid=5)} """Creates instance types / flavors
arguments: name memory vcpus local_gb flavorid swap rxtx_quota rxtx_cap
"""
for option in [memory, vcpus, local_gb, flavorid]:
try:
int(option)
except ValueError:
raise exception.InvalidInputException(
_("create arguments must be positive integers"))
if (int(memory) <= 0) or (int(vcpus) <= 0) or (int(local_gb) < 0):
raise exception.InvalidInputException(
_("create arguments must be positive integers"))
try:
db.instance_type_create(
context.get_admin_context(),
dict(name=name,
memory_mb=memory,
vcpus=vcpus,
local_gb=local_gb,
flavorid=flavorid,
swap=swap,
rxtx_quota=rxtx_quota,
rxtx_cap=rxtx_cap))
except exception.DBError, e:
LOG.exception(_('DB error: %s' % e))
raise exception.ApiError(_("Cannot create instance type: %s" % name))
def destroy(name):
"""Marks instance types / flavors as deleted
arguments: name"""
if name == None:
raise exception.InvalidInputException(_("No instance type specified"))
else:
try:
db.instance_type_destroy(context.get_admin_context(), name)
except exception.NotFound:
LOG.exception(_('Instance type %s not found for deletion' % name))
raise exception.ApiError(_("Unknown instance type: %s" % name))
def purge(name):
"""Removes instance types / flavors from database
arguments: name"""
if name == None:
raise exception.InvalidInputException(_("No instance type specified"))
else:
try:
db.instance_type_purge(context.get_admin_context(), name)
except exception.NotFound:
LOG.exception(_('Instance type %s not found for purge' % name))
raise exception.ApiError(_("Unknown instance type: %s" % name))
def get_all_types(inactive=0):
"""Retrieves non-deleted instance_types.
Pass true as argument if you want deleted instance types returned also."""
return db.instance_type_get_all(context.get_admin_context(), inactive)
def get_all_flavors():
"""retrieves non-deleted flavors. alias for instance_types.get_all_types().
Pass true as argument if you want deleted instance types returned also."""
return get_all_types(context.get_admin_context())
def get_instance_type(name):
"""Retrieves single instance type by name"""
if name is None:
return FLAGS.default_instance_type
try:
ctxt = context.get_admin_context()
inst_type = db.instance_type_get_by_name(ctxt, name)
return inst_type
except exception.DBError:
raise exception.ApiError(_("Unknown instance type: %s" % name))
def get_by_type(instance_type): def get_by_type(instance_type):
"""Build instance data structure and save it to the data store.""" """retrieve instance type name"""
if instance_type is None: if instance_type is None:
return FLAGS.default_instance_type return FLAGS.default_instance_type
if instance_type not in INSTANCE_TYPES:
raise exception.ApiError(_("Unknown instance type: %s") % \ try:
instance_type, "Invalid") ctxt = context.get_admin_context()
return instance_type inst_type = db.instance_type_get_by_name(ctxt, instance_type)
return inst_type['name']
except exception.DBError, e:
LOG.exception(_('DB error: %s' % e))
raise exception.ApiError(_("Unknown instance type: %s" %\
instance_type))
def get_by_flavor_id(flavor_id): def get_by_flavor_id(flavor_id):
for instance_type, details in INSTANCE_TYPES.iteritems(): """retrieve instance type's name by flavor_id"""
if details['flavorid'] == int(flavor_id): if flavor_id is None:
return instance_type return FLAGS.default_instance_type
return FLAGS.default_instance_type try:
ctxt = context.get_admin_context()
flavor = db.instance_type_get_by_flavor_id(ctxt, flavor_id)
return flavor['name']
except exception.DBError, e:
LOG.exception(_('DB error: %s' % e))
raise exception.ApiError(_("Unknown flavor: %s" % flavor_id))

View File

@ -370,12 +370,19 @@ class ComputeManager(manager.Manager):
context = context.elevated() context = context.elevated()
instance_ref = self.db.instance_get(context, instance_id) instance_ref = self.db.instance_get(context, instance_id)
LOG.audit(_('instance %s: rescuing'), instance_id, context=context) LOG.audit(_('instance %s: rescuing'), instance_id, context=context)
self.db.instance_set_state(context, self.db.instance_set_state(
instance_id, context,
power_state.NOSTATE, instance_id,
'rescuing') power_state.NOSTATE,
'rescuing')
self.network_manager.setup_compute_network(context, instance_id) self.network_manager.setup_compute_network(context, instance_id)
self.driver.rescue(instance_ref) self.driver.rescue(
instance_ref,
lambda result: self._update_state_callback(
self,
context,
instance_id,
result))
self._update_state(context, instance_id) self._update_state(context, instance_id)
@exception.wrap_exception @exception.wrap_exception
@ -385,11 +392,18 @@ class ComputeManager(manager.Manager):
context = context.elevated() context = context.elevated()
instance_ref = self.db.instance_get(context, instance_id) instance_ref = self.db.instance_get(context, instance_id)
LOG.audit(_('instance %s: unrescuing'), instance_id, context=context) LOG.audit(_('instance %s: unrescuing'), instance_id, context=context)
self.db.instance_set_state(context, self.db.instance_set_state(
instance_id, context,
power_state.NOSTATE, instance_id,
'unrescuing') power_state.NOSTATE,
self.driver.unrescue(instance_ref) 'unrescuing')
self.driver.unrescue(
instance_ref,
lambda result: self._update_state_callback(
self,
context,
instance_id,
result))
self._update_state(context, instance_id) self._update_state(context, instance_id)
@staticmethod @staticmethod

View File

@ -1007,6 +1007,41 @@ def console_get(context, console_id, instance_id=None):
return IMPL.console_get(context, console_id, instance_id) return IMPL.console_get(context, console_id, instance_id)
##################
def instance_type_create(context, values):
"""Create a new instance type"""
return IMPL.instance_type_create(context, values)
def instance_type_get_all(context, inactive=0):
"""Get all instance types"""
return IMPL.instance_type_get_all(context, inactive)
def instance_type_get_by_name(context, name):
"""Get instance type by name"""
return IMPL.instance_type_get_by_name(context, name)
def instance_type_get_by_flavor_id(context, id):
"""Get instance type by name"""
return IMPL.instance_type_get_by_flavor_id(context, id)
def instance_type_destroy(context, name):
"""Delete a instance type"""
return IMPL.instance_type_destroy(context, name)
def instance_type_purge(context, name):
"""Purges (removes) an instance type from DB
Use instance_type_destroy for most cases
"""
return IMPL.instance_type_purge(context, name)
#################### ####################

View File

@ -2073,6 +2073,98 @@ def console_get(context, console_id, instance_id=None):
return result return result
##################
@require_admin_context
def instance_type_create(_context, values):
try:
instance_type_ref = models.InstanceTypes()
instance_type_ref.update(values)
instance_type_ref.save()
except:
raise exception.DBError
return instance_type_ref
@require_context
def instance_type_get_all(context, inactive=0):
"""
Returns a dict describing all instance_types with name as key.
"""
session = get_session()
if inactive:
inst_types = session.query(models.InstanceTypes).\
order_by("name").\
all()
else:
inst_types = session.query(models.InstanceTypes).\
filter_by(deleted=inactive).\
order_by("name").\
all()
if inst_types:
inst_dict = {}
for i in inst_types:
inst_dict[i['name']] = dict(i)
return inst_dict
else:
raise exception.NotFound
@require_context
def instance_type_get_by_name(context, name):
"""Returns a dict describing specific instance_type"""
session = get_session()
inst_type = session.query(models.InstanceTypes).\
filter_by(name=name).\
first()
if not inst_type:
raise exception.NotFound(_("No instance type with name %s") % name)
else:
return dict(inst_type)
@require_context
def instance_type_get_by_flavor_id(context, id):
"""Returns a dict describing specific flavor_id"""
session = get_session()
inst_type = session.query(models.InstanceTypes).\
filter_by(flavorid=int(id)).\
first()
if not inst_type:
raise exception.NotFound(_("No flavor with name %s") % id)
else:
return dict(inst_type)
@require_admin_context
def instance_type_destroy(context, name):
""" Marks specific instance_type as deleted"""
session = get_session()
instance_type_ref = session.query(models.InstanceTypes).\
filter_by(name=name)
records = instance_type_ref.update(dict(deleted=1))
if records == 0:
raise exception.NotFound
else:
return instance_type_ref
@require_admin_context
def instance_type_purge(context, name):
""" Removes specific instance_type from DB
Usually instance_type_destroy should be used
"""
session = get_session()
instance_type_ref = session.query(models.InstanceTypes).\
filter_by(name=name)
records = instance_type_ref.delete()
if records == 0:
raise exception.NotFound
else:
return instance_type_ref
#################### ####################

View File

@ -0,0 +1,90 @@
# Copyright 2011 OpenStack LLC
# All Rights Reserved.
#
# Licensed 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 sqlalchemy import *
from migrate import *
from nova import log as logging
meta = MetaData()
# Table stub-definitions
# Just for the ForeignKey and column creation to succeed, these are not the
# actual definitions of instances or services.
#
fixed_ips = Table(
"fixed_ips",
meta,
Column(
"id",
Integer(),
primary_key=True,
nullable=False))
#
# New Tables
#
# None
#
# Tables to alter
#
# None
#
# Columns to add to existing tables
#
fixed_ips_addressV6 = Column(
"addressV6",
String(
length=255,
convert_unicode=False,
assert_unicode=None,
unicode_error=None,
_warn_on_bytestring=False))
fixed_ips_netmaskV6 = Column(
"netmaskV6",
String(
length=3,
convert_unicode=False,
assert_unicode=None,
unicode_error=None,
_warn_on_bytestring=False))
fixed_ips_gatewayV6 = Column(
"gatewayV6",
String(
length=255,
convert_unicode=False,
assert_unicode=None,
unicode_error=None,
_warn_on_bytestring=False))
def upgrade(migrate_engine):
# Upgrade operations go here. Don't create your own engine;
# bind migrate_engine to your metadata
meta.bind = migrate_engine
# Add columns to existing tables
fixed_ips.create_column(fixed_ips_addressV6)
fixed_ips.create_column(fixed_ips_netmaskV6)
fixed_ips.create_column(fixed_ips_gatewayV6)

View File

@ -0,0 +1,87 @@
# vim: tabstop=4 shiftwidth=4 softtabstop=4
# Copyright 2011 Ken Pepple
# Licensed 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 sqlalchemy import *
from migrate import *
from nova import api
from nova import db
from nova import log as logging
import datetime
meta = MetaData()
#
# New Tables
#
instance_types = Table('instance_types', meta,
Column('created_at', DateTime(timezone=False)),
Column('updated_at', DateTime(timezone=False)),
Column('deleted_at', DateTime(timezone=False)),
Column('deleted', Boolean(create_constraint=True, name=None)),
Column('name',
String(length=255, convert_unicode=False, assert_unicode=None,
unicode_error=None, _warn_on_bytestring=False),
unique=True),
Column('id', Integer(), primary_key=True, nullable=False),
Column('memory_mb', Integer(), nullable=False),
Column('vcpus', Integer(), nullable=False),
Column('local_gb', Integer(), nullable=False),
Column('flavorid', Integer(), nullable=False, unique=True),
Column('swap', Integer(), nullable=False, default=0),
Column('rxtx_quota', Integer(), nullable=False, default=0),
Column('rxtx_cap', Integer(), nullable=False, default=0))
def upgrade(migrate_engine):
# Upgrade operations go here
# Don't create your own engine; bind migrate_engine
# to your metadata
meta.bind = migrate_engine
try:
instance_types.create()
except Exception:
logging.info(repr(table))
logging.exception('Exception while creating instance_types table')
raise
# Here are the old static instance types
INSTANCE_TYPES = {
'm1.tiny': dict(memory_mb=512, vcpus=1, local_gb=0, flavorid=1),
'm1.small': dict(memory_mb=2048, vcpus=1, local_gb=20, flavorid=2),
'm1.medium': dict(memory_mb=4096, vcpus=2, local_gb=40, flavorid=3),
'm1.large': dict(memory_mb=8192, vcpus=4, local_gb=80, flavorid=4),
'm1.xlarge': dict(memory_mb=16384, vcpus=8, local_gb=160, flavorid=5)}
try:
i = instance_types.insert()
for name, values in INSTANCE_TYPES.iteritems():
# FIXME(kpepple) should we be seeding created_at / updated_at ?
# now = datetime.datatime.utcnow()
i.execute({'name': name, 'memory_mb': values["memory_mb"],
'vcpus': values["vcpus"], 'deleted': 0,
'local_gb': values["local_gb"],
'flavorid': values["flavorid"]})
except Exception:
logging.info(repr(table))
logging.exception('Exception while seeding instance_types table')
raise
def downgrade(migrate_engine):
# Operations to reverse the above upgrade go here.
for table in (instance_types):
table.drop()

View File

@ -126,11 +126,16 @@ class Certificate(BASE, NovaBase):
class Instance(BASE, NovaBase): class Instance(BASE, NovaBase):
"""Represents a guest vm.""" """Represents a guest vm."""
__tablename__ = 'instances' __tablename__ = 'instances'
onset_files = []
id = Column(Integer, primary_key=True, autoincrement=True) id = Column(Integer, primary_key=True, autoincrement=True)
@property @property
def name(self): def name(self):
return FLAGS.instance_name_template % self.id base_name = FLAGS.instance_name_template % self.id
if getattr(self, '_rescue', False):
base_name += "-rescue"
return base_name
admin_pass = Column(String(255)) admin_pass = Column(String(255))
user_id = Column(String(255)) user_id = Column(String(255))
@ -210,6 +215,20 @@ class InstanceActions(BASE, NovaBase):
error = Column(Text) error = Column(Text)
class InstanceTypes(BASE, NovaBase):
"""Represent possible instance_types or flavor of VM offered"""
__tablename__ = "instance_types"
id = Column(Integer, primary_key=True)
name = Column(String(255), unique=True)
memory_mb = Column(Integer)
vcpus = Column(Integer)
local_gb = Column(Integer)
flavorid = Column(Integer, unique=True)
swap = Column(Integer, nullable=False, default=0)
rxtx_quota = Column(Integer, nullable=False, default=0)
rxtx_cap = Column(Integer, nullable=False, default=0)
class Volume(BASE, NovaBase): class Volume(BASE, NovaBase):
"""Represents a block storage device that can be attached to a vm.""" """Represents a block storage device that can be attached to a vm."""
__tablename__ = 'volumes' __tablename__ = 'volumes'
@ -437,6 +456,9 @@ class FixedIp(BASE, NovaBase):
allocated = Column(Boolean, default=False) allocated = Column(Boolean, default=False)
leased = Column(Boolean, default=False) leased = Column(Boolean, default=False)
reserved = Column(Boolean, default=False) reserved = Column(Boolean, default=False)
addressV6 = Column(String(255))
netmaskV6 = Column(String(3))
gatewayV6 = Column(String(255))
class User(BASE, NovaBase): class User(BASE, NovaBase):
@ -571,7 +593,7 @@ def register_models():
connection is lost and needs to be reestablished. connection is lost and needs to be reestablished.
""" """
from sqlalchemy import create_engine from sqlalchemy import create_engine
models = (Service, Instance, InstanceActions, models = (Service, Instance, InstanceActions, InstanceTypes,
Volume, ExportDevice, IscsiTarget, FixedIp, FloatingIp, Volume, ExportDevice, IscsiTarget, FixedIp, FloatingIp,
Network, SecurityGroup, SecurityGroupIngressRule, Network, SecurityGroup, SecurityGroupIngressRule,
SecurityGroupInstanceAssociation, AuthToken, User, SecurityGroupInstanceAssociation, AuthToken, User,

View File

@ -531,6 +531,11 @@ class VlanManager(NetworkManager):
' than 4094')) ' than 4094'))
fixed_net = IPy.IP(cidr) fixed_net = IPy.IP(cidr)
if fixed_net.len() < num_networks * network_size:
raise ValueError(_('The network range is not big enough to fit '
'%(num_networks)s. Network size is %(network_size)s' %
locals()))
fixed_net_v6 = IPy.IP(cidr_v6) fixed_net_v6 = IPy.IP(cidr_v6)
network_size_v6 = 1 << 64 network_size_v6 = 1 << 64
significant_bits_v6 = 64 significant_bits_v6 = 64

View File

@ -27,7 +27,7 @@ from nova import utils
from nova.virt.vmwareapi_conn import VMWareAPISession from nova.virt.vmwareapi_conn import VMWareAPISession
from nova.virt.vmwareapi.network_utils import NetworkHelper from nova.virt.vmwareapi.network_utils import NetworkHelper
LOG = logging.getLogger("nova.vmwareapi_net") LOG = logging.getLogger("nova.network.vmwareapi_net")
FLAGS = flags.FLAGS FLAGS = flags.FLAGS
flags.DEFINE_string('vlan_interface', 'vmnic0', flags.DEFINE_string('vlan_interface', 'vmnic0',
@ -35,34 +35,6 @@ flags.DEFINE_string('vlan_interface', 'vmnic0',
'vlan networking') 'vlan networking')
def metadata_forward():
pass
def init_host():
pass
def bind_floating_ip(floating_ip, check_exit_code=True):
pass
def unbind_floating_ip(floating_ip):
pass
def ensure_vlan_forward(public_ip, port, private_ip):
pass
def ensure_floating_forward(floating_ip, fixed_ip):
pass
def remove_floating_forward(floating_ip, fixed_ip):
pass
def ensure_vlan_bridge(vlan_num, bridge, net_attrs=None): def ensure_vlan_bridge(vlan_num, bridge, net_attrs=None):
"""Create a vlan and bridge unless they already exist""" """Create a vlan and bridge unless they already exist"""
#open vmwareapi session #open vmwareapi session
@ -77,48 +49,49 @@ def ensure_vlan_bridge(vlan_num, bridge, net_attrs=None):
session = VMWareAPISession(host_ip, host_username, host_password, session = VMWareAPISession(host_ip, host_username, host_password,
FLAGS.vmwareapi_api_retry_count) FLAGS.vmwareapi_api_retry_count)
vlan_interface = FLAGS.vlan_interface vlan_interface = FLAGS.vlan_interface
#check whether bridge already exists #Check if the vlan_interface physical network adapter exists on the host
#retrieve network whose name_label is "bridge" if not NetworkHelper.check_if_vlan_interface_exists(session,
vlan_interface):
raise Exception(_("There is no physical network adapter with the name"
" %s on the ESX host") % vlan_interface)
#check whether bridge already exists and retrieve the the ref of the
#network whose name_label is "bridge"
network_ref = NetworkHelper.get_network_with_the_name(session, bridge) network_ref = NetworkHelper.get_network_with_the_name(session, bridge)
#Get the vSwitch associated with the Physical Adapter
vswitch_associated = NetworkHelper.get_vswitch_for_vlan_interface(
session, vlan_interface)
if vswitch_associated is None:
raise Exception(_("There is no virtual switch associated with "
"the physical network adapter with name %s") %
vlan_interface)
if network_ref == None: if network_ref == None:
#Create a port group on the vSwitch associated with the vlan_interface #Create a port group on the vSwitch associated with the vlan_interface
#corresponding physical network adapter on the ESX host #corresponding physical network adapter on the ESX host
vswitches = NetworkHelper.get_vswitches_for_vlan_interface(session, NetworkHelper.create_port_group(session, bridge, vswitch_associated,
vlan_interface) vlan_num)
if len(vswitches) == 0:
raise Exception(_("There is no virtual switch connected "
"to the physical network adapter with name %s") %
vlan_interface)
#Assuming physical network interface is associated with only one
#virtual switch
NetworkHelper.create_port_group(session, bridge, vswitches[0],
vlan_num)
else: else:
#check VLAN tag is appropriate #Get the vlan id and vswitch corresponding to the port group
is_vlan_proper, ret_vlan_id = NetworkHelper.check_if_vlan_id_is_proper( pg_vlanid, pg_vswitch = \
session, bridge, vlan_num) NetworkHelper.get_vlanid_and_vswicth_for_portgroup(session, bridge)
if not is_vlan_proper:
raise Exception(_("VLAN tag not appropriate for the port group "
"%(bridge)s. Expected VLAN tag is %(vlan_num)s, "
"but the one associated with the port group is"
" %(ret_vlan_id)s") % locals())
#Check if the vsiwtch associated is proper
if pg_vswitch != vswitch_associated:
raise Exception(_("vSwitch which contains the port group "
"%(bridge)s is not associated with the desired "
"physical adapter. Expected vSwitch is "
"%(vswitch_associated)s, but the one associated"
" is %(pg_vswitch)s") %\
{"bridge": bridge,
"vswitch_associated": vswitch_associated,
"pg_vswitch": pg_vswitch})
def ensure_vlan(vlan_num): #Check if the vlan id is proper for the port group
pass if pg_vlanid != vlan_num:
raise Exception(_("VLAN tag is not appropriate for the port "
"group %(bridge)s. Expected VLAN tag is "
def ensure_bridge(bridge, interface, net_attrs=None): "%(vlan_num)s, but the one associated with the "
pass "port group is %(pg_vlanid)s") %\
{"bridge": bridge,
"vlan_num": vlan_num,
def get_dhcp_hosts(context, network_id): "pg_vlanid": pg_vlanid})
pass
def update_dhcp(context, network_id):
pass
def update_ra(context, network_id):
pass

View File

@ -19,6 +19,7 @@
Test suites for 'common' code used throughout the OpenStack HTTP API. Test suites for 'common' code used throughout the OpenStack HTTP API.
""" """
import webob.exc
from webob import Request from webob import Request
@ -160,3 +161,23 @@ class LimiterTest(test.TestCase):
self.assertEqual(limited(items, req, max_limit=2000), items[3:]) self.assertEqual(limited(items, req, max_limit=2000), items[3:])
req = Request.blank('/?offset=3000&limit=10') req = Request.blank('/?offset=3000&limit=10')
self.assertEqual(limited(items, req, max_limit=2000), []) self.assertEqual(limited(items, req, max_limit=2000), [])
def test_limiter_negative_limit(self):
"""
Test a negative limit.
"""
def _limit_large():
limited(self.large, req, max_limit=2000)
req = Request.blank('/?limit=-3000')
self.assertRaises(webob.exc.HTTPBadRequest, _limit_large)
def test_limiter_negative_offset(self):
"""
Test a negative offset.
"""
def _limit_large():
limited(self.large, req, max_limit=2000)
req = Request.blank('/?offset=-30')
self.assertRaises(webob.exc.HTTPBadRequest, _limit_large)

View File

@ -20,6 +20,8 @@ import webob
from nova import test from nova import test
import nova.api import nova.api
from nova import context
from nova import db
from nova.api.openstack import flavors from nova.api.openstack import flavors
from nova.tests.api.openstack import fakes from nova.tests.api.openstack import fakes
@ -33,6 +35,7 @@ class FlavorsTest(test.TestCase):
fakes.stub_out_networking(self.stubs) fakes.stub_out_networking(self.stubs)
fakes.stub_out_rate_limiting(self.stubs) fakes.stub_out_rate_limiting(self.stubs)
fakes.stub_out_auth(self.stubs) fakes.stub_out_auth(self.stubs)
self.context = context.get_admin_context()
def tearDown(self): def tearDown(self):
self.stubs.UnsetAll() self.stubs.UnsetAll()
@ -41,6 +44,9 @@ class FlavorsTest(test.TestCase):
def test_get_flavor_list(self): def test_get_flavor_list(self):
req = webob.Request.blank('/v1.0/flavors') req = webob.Request.blank('/v1.0/flavors')
res = req.get_response(fakes.wsgi_app()) res = req.get_response(fakes.wsgi_app())
self.assertEqual(res.status_int, 200)
def test_get_flavor_by_id(self): def test_get_flavor_by_id(self):
pass req = webob.Request.blank('/v1.0/flavors/1')
res = req.get_response(fakes.wsgi_app())
self.assertEqual(res.status_int, 200)

View File

@ -20,13 +20,22 @@
import time import time
from nova import db from nova import db
from nova import test
from nova import utils from nova import utils
from nova.compute import instance_types
def stub_out_db_instance_api(stubs): def stub_out_db_instance_api(stubs):
""" Stubs out the db API for creating Instances """ """ Stubs out the db API for creating Instances """
INSTANCE_TYPES = {
'm1.tiny': dict(memory_mb=512, vcpus=1, local_gb=0, flavorid=1),
'm1.small': dict(memory_mb=2048, vcpus=1, local_gb=20, flavorid=2),
'm1.medium':
dict(memory_mb=4096, vcpus=2, local_gb=40, flavorid=3),
'm1.large': dict(memory_mb=8192, vcpus=4, local_gb=80, flavorid=4),
'm1.xlarge':
dict(memory_mb=16384, vcpus=8, local_gb=160, flavorid=5)}
class FakeModel(object): class FakeModel(object):
""" Stubs out for model """ """ Stubs out for model """
def __init__(self, values): def __init__(self, values):
@ -41,10 +50,16 @@ def stub_out_db_instance_api(stubs):
else: else:
raise NotImplementedError() raise NotImplementedError()
def fake_instance_type_get_all(context, inactive=0):
return INSTANCE_TYPES
def fake_instance_type_get_by_name(context, name):
return INSTANCE_TYPES[name]
def fake_instance_create(values): def fake_instance_create(values):
""" Stubs out the db.instance_create method """ """ Stubs out the db.instance_create method """
type_data = instance_types.INSTANCE_TYPES[values['instance_type']] type_data = INSTANCE_TYPES[values['instance_type']]
base_options = { base_options = {
'name': values['name'], 'name': values['name'],
@ -73,3 +88,5 @@ def stub_out_db_instance_api(stubs):
stubs.Set(db, 'instance_create', fake_instance_create) stubs.Set(db, 'instance_create', fake_instance_create)
stubs.Set(db, 'network_get_by_instance', fake_network_get_by_instance) stubs.Set(db, 'network_get_by_instance', fake_network_get_by_instance)
stubs.Set(db, 'instance_type_get_all', fake_instance_type_get_all)
stubs.Set(db, 'instance_type_get_by_name', fake_instance_type_get_by_name)

View File

@ -267,7 +267,7 @@ class CloudTestCase(test.TestCase):
self._create_key('test1') self._create_key('test1')
self._create_key('test2') self._create_key('test2')
result = self.cloud.describe_key_pairs(self.context) result = self.cloud.describe_key_pairs(self.context)
keys = result["keypairsSet"] keys = result["keySet"]
self.assertTrue(filter(lambda k: k['keyName'] == 'test1', keys)) self.assertTrue(filter(lambda k: k['keyName'] == 'test1', keys))
self.assertTrue(filter(lambda k: k['keyName'] == 'test2', keys)) self.assertTrue(filter(lambda k: k['keyName'] == 'test2', keys))

View File

@ -0,0 +1,86 @@
# vim: tabstop=4 shiftwidth=4 softtabstop=4
# Copyright 2011 Ken Pepple
# Licensed 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.
"""
Unit Tests for instance types code
"""
import time
from nova import context
from nova import db
from nova import exception
from nova import flags
from nova import log as logging
from nova import test
from nova import utils
from nova.compute import instance_types
from nova.db.sqlalchemy.session import get_session
from nova.db.sqlalchemy import models
FLAGS = flags.FLAGS
LOG = logging.getLogger('nova.tests.compute')
class InstanceTypeTestCase(test.TestCase):
"""Test cases for instance type code"""
def setUp(self):
super(InstanceTypeTestCase, self).setUp()
session = get_session()
max_flavorid = session.query(models.InstanceTypes).\
order_by("flavorid desc").\
first()
self.flavorid = max_flavorid["flavorid"] + 1
self.name = str(int(time.time()))
def test_instance_type_create_then_delete(self):
"""Ensure instance types can be created"""
starting_inst_list = instance_types.get_all_types()
instance_types.create(self.name, 256, 1, 120, self.flavorid)
new = instance_types.get_all_types()
self.assertNotEqual(len(starting_inst_list),
len(new),
'instance type was not created')
instance_types.destroy(self.name)
self.assertEqual(1,
instance_types.get_instance_type(self.name)["deleted"])
self.assertEqual(starting_inst_list, instance_types.get_all_types())
instance_types.purge(self.name)
self.assertEqual(len(starting_inst_list),
len(instance_types.get_all_types()),
'instance type not purged')
def test_get_all_instance_types(self):
"""Ensures that all instance types can be retrieved"""
session = get_session()
total_instance_types = session.query(models.InstanceTypes).\
count()
inst_types = instance_types.get_all_types()
self.assertEqual(total_instance_types, len(inst_types))
def test_invalid_create_args_should_fail(self):
"""Ensures that instance type creation fails with invalid args"""
self.assertRaises(
exception.InvalidInputException,
instance_types.create, self.name, 0, 1, 120, self.flavorid)
self.assertRaises(
exception.InvalidInputException,
instance_types.create, self.name, 256, -1, 120, self.flavorid)
self.assertRaises(
exception.InvalidInputException,
instance_types.create, self.name, 256, 1, "aa", self.flavorid)
def test_non_existant_inst_type_shouldnt_delete(self):
"""Ensures that instance type creation fails with invalid args"""
self.assertRaises(exception.ApiError,
instance_types.destroy, "sfsfsdfdfs")

View File

@ -74,19 +74,30 @@ class QuotaTestCase(test.TestCase):
vol['size'] = size vol['size'] = size
return db.volume_create(self.context, vol)['id'] return db.volume_create(self.context, vol)['id']
def _get_instance_type(self, name):
instance_types = {
'm1.tiny': dict(memory_mb=512, vcpus=1, local_gb=0, flavorid=1),
'm1.small': dict(memory_mb=2048, vcpus=1, local_gb=20, flavorid=2),
'm1.medium':
dict(memory_mb=4096, vcpus=2, local_gb=40, flavorid=3),
'm1.large': dict(memory_mb=8192, vcpus=4, local_gb=80, flavorid=4),
'm1.xlarge':
dict(memory_mb=16384, vcpus=8, local_gb=160, flavorid=5)}
return instance_types[name]
def test_quota_overrides(self): def test_quota_overrides(self):
"""Make sure overriding a projects quotas works""" """Make sure overriding a projects quotas works"""
num_instances = quota.allowed_instances(self.context, 100, num_instances = quota.allowed_instances(self.context, 100,
instance_types.INSTANCE_TYPES['m1.small']) self._get_instance_type('m1.small'))
self.assertEqual(num_instances, 2) self.assertEqual(num_instances, 2)
db.quota_create(self.context, {'project_id': self.project.id, db.quota_create(self.context, {'project_id': self.project.id,
'instances': 10}) 'instances': 10})
num_instances = quota.allowed_instances(self.context, 100, num_instances = quota.allowed_instances(self.context, 100,
instance_types.INSTANCE_TYPES['m1.small']) self._get_instance_type('m1.small'))
self.assertEqual(num_instances, 4) self.assertEqual(num_instances, 4)
db.quota_update(self.context, self.project.id, {'cores': 100}) db.quota_update(self.context, self.project.id, {'cores': 100})
num_instances = quota.allowed_instances(self.context, 100, num_instances = quota.allowed_instances(self.context, 100,
instance_types.INSTANCE_TYPES['m1.small']) self._get_instance_type('m1.small'))
self.assertEqual(num_instances, 10) self.assertEqual(num_instances, 10)
# metadata_items # metadata_items

View File

@ -26,7 +26,6 @@ from nova import flags
from nova import test from nova import test
from nova import utils from nova import utils
from nova.auth import manager from nova.auth import manager
from nova.compute import instance_types
from nova.compute import power_state from nova.compute import power_state
from nova.tests.glance import stubs as glance_stubs from nova.tests.glance import stubs as glance_stubs
from nova.tests.vmwareapi import db_fakes from nova.tests.vmwareapi import db_fakes
@ -74,7 +73,7 @@ class VMWareAPIVMTestCase(test.TestCase):
'mac_address': 'aa:bb:cc:dd:ee:ff', 'mac_address': 'aa:bb:cc:dd:ee:ff',
} }
self.instance = db.instance_create(values) self.instance = db.instance_create(values)
self.type_data = instance_types.INSTANCE_TYPES[values['instance_type']] self.type_data = db.instance_type_get_by_name(None, 'm1.large')
self.conn.spawn(self.instance) self.conn.spawn(self.instance)
self._check_vm_record() self._check_vm_record()

View File

@ -233,7 +233,7 @@ class XenAPIVMTestCase(test.TestCase):
vm = vms[0] vm = vms[0]
# Check that m1.large above turned into the right thing. # Check that m1.large above turned into the right thing.
instance_type = instance_types.INSTANCE_TYPES['m1.large'] instance_type = db.instance_type_get_by_name(conn, 'm1.large')
mem_kib = long(instance_type['memory_mb']) << 10 mem_kib = long(instance_type['memory_mb']) << 10
mem_bytes = str(mem_kib << 10) mem_bytes = str(mem_kib << 10)
vcpus = instance_type['vcpus'] vcpus = instance_type['vcpus']

View File

@ -23,12 +23,20 @@ import time
from nova import db from nova import db
from nova import utils from nova import utils
from nova.compute import instance_types
def stub_out_db_instance_api(stubs): def stub_out_db_instance_api(stubs):
""" Stubs out the db API for creating Instances """ """ Stubs out the db API for creating Instances """
INSTANCE_TYPES = {
'm1.tiny': dict(memory_mb=512, vcpus=1, local_gb=0, flavorid=1),
'm1.small': dict(memory_mb=2048, vcpus=1, local_gb=20, flavorid=2),
'm1.medium':
dict(memory_mb=4096, vcpus=2, local_gb=40, flavorid=3),
'm1.large': dict(memory_mb=8192, vcpus=4, local_gb=80, flavorid=4),
'm1.xlarge':
dict(memory_mb=16384, vcpus=8, local_gb=160, flavorid=5)}
class FakeModel(object): class FakeModel(object):
""" Stubs out for model """ """ Stubs out for model """
@ -47,7 +55,7 @@ def stub_out_db_instance_api(stubs):
def fake_instance_create(values): def fake_instance_create(values):
""" Stubs out the db.instance_create method """ """ Stubs out the db.instance_create method """
type_data = instance_types.INSTANCE_TYPES[values['instance_type']] type_data = INSTANCE_TYPES[values['instance_type']]
base_options = { base_options = {
'name': values['name'], 'name': values['name'],
@ -86,8 +94,16 @@ def stub_out_db_instance_api(stubs):
""" Stubs out the db.instance_get_fixed_address method """ """ Stubs out the db.instance_get_fixed_address method """
return '10.10.10.10' return '10.10.10.10'
def fake_instance_type_get_all(context, inactive=0):
return INSTANCE_TYPES
def fake_instance_type_get_by_name(context, name):
return INSTANCE_TYPES[name]
stubs.Set(db, 'instance_create', fake_instance_create) stubs.Set(db, 'instance_create', fake_instance_create)
stubs.Set(db, 'network_get_by_instance', fake_network_get_by_instance) stubs.Set(db, 'network_get_by_instance', fake_network_get_by_instance)
stubs.Set(db, 'instance_action_create', fake_instance_action_create) stubs.Set(db, 'instance_action_create', fake_instance_action_create)
stubs.Set(db, 'instance_get_fixed_address', stubs.Set(db, 'instance_get_fixed_address',
fake_instance_get_fixed_address) fake_instance_get_fixed_address)
stubs.Set(db, 'instance_type_get_all', fake_instance_type_get_all)
stubs.Set(db, 'instance_type_get_by_name', fake_instance_type_get_by_name)

View File

@ -27,7 +27,7 @@ def stubout_instance_snapshot(stubs):
def fake_fetch_image(cls, session, instance_id, image, user, project, def fake_fetch_image(cls, session, instance_id, image, user, project,
type): type):
# Stubout wait_for_task # Stubout wait_for_task
def fake_wait_for_task(self, id, task): def fake_wait_for_task(self, task, id):
class FakeEvent: class FakeEvent:
def send(self, value): def send(self, value):

View File

@ -55,6 +55,7 @@ from nova import db
from nova import exception from nova import exception
from nova import flags from nova import flags
from nova import log as logging from nova import log as logging
#from nova import test
from nova import utils from nova import utils
#from nova.api import context #from nova.api import context
from nova.auth import manager from nova.auth import manager
@ -362,7 +363,7 @@ class LibvirtConnection(object):
raise exception.APIError("resume not supported for libvirt") raise exception.APIError("resume not supported for libvirt")
@exception.wrap_exception @exception.wrap_exception
def rescue(self, instance): def rescue(self, instance, callback=None):
self.destroy(instance, False) self.destroy(instance, False)
xml = self.to_xml(instance, rescue=True) xml = self.to_xml(instance, rescue=True)
@ -392,7 +393,7 @@ class LibvirtConnection(object):
return timer.start(interval=0.5, now=True) return timer.start(interval=0.5, now=True)
@exception.wrap_exception @exception.wrap_exception
def unrescue(self, instance): def unrescue(self, instance, callback=None):
# NOTE(vish): Because reboot destroys and recreates an instance using # NOTE(vish): Because reboot destroys and recreates an instance using
# the normal xml file, we can just call reboot here # the normal xml file, we can just call reboot here
self.reboot(instance) self.reboot(instance)
@ -606,7 +607,7 @@ class LibvirtConnection(object):
user=user, user=user,
project=project, project=project,
size=size) size=size)
type_data = instance_types.INSTANCE_TYPES[inst['instance_type']] type_data = instance_types.get_instance_type(inst['instance_type'])
if type_data['local_gb']: if type_data['local_gb']:
self._cache_image(fn=self._create_local, self._cache_image(fn=self._create_local,
@ -667,7 +668,8 @@ class LibvirtConnection(object):
instance['id']) instance['id'])
# FIXME(vish): stick this in db # FIXME(vish): stick this in db
instance_type = instance['instance_type'] instance_type = instance['instance_type']
instance_type = instance_types.INSTANCE_TYPES[instance_type] # instance_type = test.INSTANCE_TYPES[instance_type]
instance_type = instance_types.get_instance_type(instance_type)
ip_address = db.instance_get_fixed_address(context.get_admin_context(), ip_address = db.instance_get_fixed_address(context.get_admin_context(),
instance['id']) instance['id'])
# Assume that the gateway also acts as the dhcp server. # Assume that the gateway also acts as the dhcp server.

View File

@ -28,7 +28,8 @@ from nova.virt.vmwareapi import vim
from nova.virt.vmwareapi.vim import SessionFaultyException from nova.virt.vmwareapi.vim import SessionFaultyException
_CLASSES = ['Datacenter', 'Datastore', 'ResourcePool', 'VirtualMachine', _CLASSES = ['Datacenter', 'Datastore', 'ResourcePool', 'VirtualMachine',
'Network', 'HostSystem', 'Task', 'session', 'files'] 'Network', 'HostSystem', 'HostNetworkSystem', 'Task', 'session',
'files']
_FAKE_FILE_SIZE = 1024 _FAKE_FILE_SIZE = 1024
@ -54,6 +55,7 @@ def reset():
else: else:
_db_content[c] = {} _db_content[c] = {}
create_network() create_network()
create_host_network_system()
create_host() create_host()
create_datacenter() create_datacenter()
create_datastore() create_datastore()
@ -242,13 +244,33 @@ class Datastore(ManagedObject):
self.set("summary.name", "fake-ds") self.set("summary.name", "fake-ds")
class HostNetworkSystem(ManagedObject):
""" HostNetworkSystem class """
def __init__(self):
ManagedObject.__init__(self, "HostNetworkSystem")
self.set("name", "networkSystem")
pnic_do = DataObject()
pnic_do.device = "vmnic0"
net_info_pnic = DataObject()
net_info_pnic.PhysicalNic = [pnic_do]
self.set("networkInfo.pnic", net_info_pnic)
class HostSystem(ManagedObject): class HostSystem(ManagedObject):
""" Host System class """ """ Host System class """
def __init__(self): def __init__(self):
ManagedObject.__init__(self, "HostSystem") ManagedObject.__init__(self, "HostSystem")
self.set("name", "HostSystem") self.set("name", "ha-host")
self.set("configManager.networkSystem", "NetworkSystem") if _db_content.get("HostNetworkSystem", None) is None:
create_host_network_system()
host_net_key = _db_content["HostNetworkSystem"].keys()[0]
host_net_sys = _db_content["HostNetworkSystem"][host_net_key].obj
self.set("configManager.networkSystem", host_net_sys)
vswitch_do = DataObject() vswitch_do = DataObject()
vswitch_do.pnic = ["vmnic0"] vswitch_do.pnic = ["vmnic0"]
@ -324,6 +346,11 @@ class Task(ManagedObject):
self.set("info", info) self.set("info", info)
def create_host_network_system():
host_net_system = HostNetworkSystem()
_create_object("HostNetworkSystem", host_net_system)
def create_host(): def create_host():
host_system = HostSystem() host_system = HostSystem()
_create_object('HostSystem', host_system) _create_object('HostSystem', host_system)

View File

@ -48,8 +48,8 @@ class NetworkHelper:
return None return None
@classmethod @classmethod
def get_vswitches_for_vlan_interface(cls, session, vlan_interface): def get_vswitch_for_vlan_interface(cls, session, vlan_interface):
""" Gets the list of vswitches associated with the physical """ Gets the vswitch associated with the physical
network adapter with the name supplied""" network adapter with the name supplied"""
#Get the list of vSwicthes on the Host System #Get the list of vSwicthes on the Host System
host_mor = session._call_method(vim_util, "get_objects", host_mor = session._call_method(vim_util, "get_objects",
@ -57,21 +57,31 @@ class NetworkHelper:
vswitches = session._call_method(vim_util, vswitches = session._call_method(vim_util,
"get_dynamic_property", host_mor, "get_dynamic_property", host_mor,
"HostSystem", "config.network.vswitch").HostVirtualSwitch "HostSystem", "config.network.vswitch").HostVirtualSwitch
vswicthes_conn_to_physical_nic = [] #Get the vSwitch associated with the network adapter
#For each vSwitch check if it is associated with the network adapter
for elem in vswitches: for elem in vswitches:
try: try:
for nic_elem in elem.pnic: for nic_elem in elem.pnic:
if str(nic_elem).split('-')[-1].find(vlan_interface) != -1: if str(nic_elem).split('-')[-1].find(vlan_interface) != -1:
vswicthes_conn_to_physical_nic.append(elem.name) return elem.name
except Exception: except Exception:
pass pass
return vswicthes_conn_to_physical_nic
@classmethod @classmethod
def check_if_vlan_id_is_proper(cls, session, pg_name, vlan_id): def check_if_vlan_interface_exists(cls, session, vlan_interface):
""" Check if the vlan id associated with the port group matches the """ Checks if the vlan_inteface exists on the esx host """
vlan tag supplied """ host_net_system_mor = session._call_method(vim_util, "get_objects",
"HostSystem", ["configManager.networkSystem"])[0].propSet[0].val
physical_nics = session._call_method(vim_util,
"get_dynamic_property", host_net_system_mor,
"HostNetworkSystem", "networkInfo.pnic").PhysicalNic
for pnic in physical_nics:
if vlan_interface == pnic.device:
return True
return False
@classmethod
def get_vlanid_and_vswicth_for_portgroup(cls, session, pg_name):
""" Get the vlan id and vswicth associated with the port group """
host_mor = session._call_method(vim_util, "get_objects", host_mor = session._call_method(vim_util, "get_objects",
"HostSystem")[0].obj "HostSystem")[0].obj
port_grps_on_host = session._call_method(vim_util, port_grps_on_host = session._call_method(vim_util,
@ -79,10 +89,8 @@ class NetworkHelper:
"HostSystem", "config.network.portgroup").HostPortGroup "HostSystem", "config.network.portgroup").HostPortGroup
for p_gp in port_grps_on_host: for p_gp in port_grps_on_host:
if p_gp.spec.name == pg_name: if p_gp.spec.name == pg_name:
if p_gp.spec.vlanId == vlan_id: p_grp_vswitch_name = p_gp.vswitch.split("-")[-1]
return True, vlan_id return p_gp.spec.vlanId, p_grp_vswitch_name
else:
return False, p_gp.spec.vlanId
@classmethod @classmethod
def create_port_group(cls, session, pg_name, vswitch_name, vlan_id=0): def create_port_group(cls, session, pg_name, vswitch_name, vlan_id=0):

View File

@ -20,111 +20,74 @@ The VMware API utility module
""" """
def build_recursive_traversal_spec(client_factory): def build_selcetion_spec(client_factory, name):
"""Builds the Traversal Spec""" """ Builds the selection spec """
#Traversal through "hostFolder" branch sel_spec = client_factory.create('ns0:SelectionSpec')
visit_folders_select_spec = client_factory.create('ns0:SelectionSpec') sel_spec.name = name
visit_folders_select_spec.name = "visitFolders" return sel_spec
dc_to_hf = client_factory.create('ns0:TraversalSpec')
dc_to_hf.name = "dc_to_hf"
dc_to_hf.type = "Datacenter"
dc_to_hf.path = "hostFolder"
dc_to_hf.skip = False
dc_to_hf.selectSet = [visit_folders_select_spec]
#Traversal through "vmFolder" branch
visit_folders_select_spec = client_factory.create('ns0:SelectionSpec')
visit_folders_select_spec.name = "visitFolders"
dc_to_vmf = client_factory.create('ns0:TraversalSpec')
dc_to_vmf.name = "dc_to_vmf"
dc_to_vmf.type = "Datacenter"
dc_to_vmf.path = "vmFolder"
dc_to_vmf.skip = False
dc_to_vmf.selectSet = [visit_folders_select_spec]
#Traversal to the DataStore from the DataCenter def build_traversal_spec(client_factory, name, type, path, skip, select_set):
visit_folders_select_spec = \ """ Builds the traversal spec object """
client_factory.create('ns0:SelectionSpec')
visit_folders_select_spec.name = "traverseChild"
dc_to_ds = client_factory.create('ns0:TraversalSpec')
dc_to_ds.name = "dc_to_ds"
dc_to_ds.type = "Datacenter"
dc_to_ds.path = "datastore"
dc_to_ds.skip = False
dc_to_ds.selectSet = [visit_folders_select_spec]
#Traversal through "vm" branch
visit_folders_select_spec = \
client_factory.create('ns0:SelectionSpec')
visit_folders_select_spec.name = "visitFolders"
h_to_vm = client_factory.create('ns0:TraversalSpec')
h_to_vm.name = "h_to_vm"
h_to_vm.type = "HostSystem"
h_to_vm.path = "vm"
h_to_vm.skip = False
h_to_vm.selectSet = [visit_folders_select_spec]
#Traversal through "host" branch
cr_to_h = client_factory.create('ns0:TraversalSpec')
cr_to_h.name = "cr_to_h"
cr_to_h.type = "ComputeResource"
cr_to_h.path = "host"
cr_to_h.skip = False
cr_to_h.selectSet = []
cr_to_ds = client_factory.create('ns0:TraversalSpec')
cr_to_ds.name = "cr_to_ds"
cr_to_ds.type = "ComputeResource"
cr_to_ds.path = "datastore"
cr_to_ds.skip = False
#Traversal through "resourcePool" branch
rp_to_rp_select_spec = client_factory.create('ns0:SelectionSpec')
rp_to_rp_select_spec.name = "rp_to_rp"
rp_to_vm_select_spec = client_factory.create('ns0:SelectionSpec')
rp_to_vm_select_spec.name = "rp_to_vm"
cr_to_rp = client_factory.create('ns0:TraversalSpec')
cr_to_rp.name = "cr_to_rp"
cr_to_rp.type = "ComputeResource"
cr_to_rp.path = "resourcePool"
cr_to_rp.skip = False
cr_to_rp.selectSet = [rp_to_rp_select_spec, rp_to_vm_select_spec]
#Traversal through all ResourcePools
rp_to_rp_select_spec = client_factory.create('ns0:SelectionSpec')
rp_to_rp_select_spec.name = "rp_to_rp"
rp_to_vm_select_spec = client_factory.create('ns0:SelectionSpec')
rp_to_vm_select_spec.name = "rp_to_vm"
rp_to_rp = client_factory.create('ns0:TraversalSpec')
rp_to_rp.name = "rp_to_rp"
rp_to_rp.type = "ResourcePool"
rp_to_rp.path = "resourcePool"
rp_to_rp.skip = False
rp_to_rp.selectSet = [rp_to_rp_select_spec, rp_to_vm_select_spec]
#Traversal through ResourcePools vm folders
rp_to_rp_select_spec = client_factory.create('ns0:SelectionSpec')
rp_to_rp_select_spec.name = "rp_to_rp"
rp_to_vm_select_spec = client_factory.create('ns0:SelectionSpec')
rp_to_vm_select_spec.name = "rp_to_vm"
rp_to_vm = client_factory.create('ns0:TraversalSpec')
rp_to_vm.name = "rp_to_vm"
rp_to_vm.type = "ResourcePool"
rp_to_vm.path = "vm"
rp_to_vm.skip = False
rp_to_vm.selectSet = [rp_to_rp_select_spec, rp_to_vm_select_spec]
#Include all Traversals and Recurse into them
visit_folders_select_spec = \
client_factory.create('ns0:SelectionSpec')
visit_folders_select_spec.name = "visitFolders"
traversal_spec = client_factory.create('ns0:TraversalSpec') traversal_spec = client_factory.create('ns0:TraversalSpec')
traversal_spec.name = "visitFolders" traversal_spec.name= name
traversal_spec.type = "Folder" traversal_spec.type = type
traversal_spec.path = "childEntity" traversal_spec.path = path
traversal_spec.skip = False traversal_spec.skip = skip
traversal_spec.selectSet = [visit_folders_select_spec, dc_to_hf, dc_to_vmf, traversal_spec.selectSet = select_set
cr_to_ds, cr_to_h, cr_to_rp, rp_to_rp, h_to_vm, rp_to_vm] return traversal_spec
def build_recursive_traversal_spec(client_factory):
""" Builds the Recursive Traversal Spec to traverse the object managed
object hierarchy """
visit_folders_select_spec = build_selcetion_spec(client_factory,
"visitFolders")
#For getting to hostFolder from datacnetr
dc_to_hf = build_traversal_spec(client_factory, "dc_to_hf", "Datacenter",
"hostFolder", False,
[visit_folders_select_spec])
#For getting to vmFolder from datacenter
dc_to_vmf = build_traversal_spec(client_factory, "dc_to_vmf", "Datacenter",
"vmFolder", False,
[visit_folders_select_spec])
#For getting Host System to virtual machine
h_to_vm = build_traversal_spec(client_factory, "h_to_vm", "HostSystem",
"vm", False,
[visit_folders_select_spec])
#For getting to Host System from Compute Resource
cr_to_h = build_traversal_spec(client_factory, "cr_to_h",
"ComputeResource", "host", False, [])
#For getting to datastore from Compute Resource
cr_to_ds = build_traversal_spec(client_factory, "cr_to_ds",
"ComputeResource", "datastore", False, [])
rp_to_rp_select_spec = build_selcetion_spec(client_factory, "rp_to_rp")
rp_to_vm_select_spec = build_selcetion_spec(client_factory, "rp_to_vm")
#For getting to resource pool from Compute Resource
cr_to_rp = build_traversal_spec(client_factory, "cr_to_rp",
"ComputeResource", "resourcePool", False,
[rp_to_rp_select_spec, rp_to_vm_select_spec])
#For getting to child res pool from the parent res pool
rp_to_rp = build_traversal_spec(client_factory, "rp_to_rp", "ResourcePool",
"resourcePool", False,
[rp_to_rp_select_spec, rp_to_vm_select_spec])
#For getting to Virtual Machine from the Resource Pool
rp_to_vm = build_traversal_spec(client_factory, "rp_to_vm", "ResourcePool",
"vm", False,
[rp_to_rp_select_spec, rp_to_vm_select_spec])
#Get the assorted traversal spec which takes care of the objects to
#be searched for from the root folder
traversal_spec = build_traversal_spec(client_factory, "visitFolders",
"Folder", "childEntity", False,
[visit_folders_select_spec, dc_to_hf,
dc_to_vmf, cr_to_ds, cr_to_h, cr_to_rp,
rp_to_rp, h_to_vm, rp_to_vm])
return traversal_spec return traversal_spec

View File

@ -104,75 +104,99 @@ class VMWareVMOps(object):
"but that already exists on the host") % instance.name) "but that already exists on the host") % instance.name)
client_factory = self._session._get_vim().client.factory client_factory = self._session._get_vim().client.factory
service_content = self._session._get_vim().get_service_content()
network = db.network_get_by_instance(context.get_admin_context(), network = db.network_get_by_instance(context.get_admin_context(),
instance['id']) instance['id'])
net_name = network['bridge'] net_name = network['bridge']
network_ref = \
NetworkHelper.get_network_with_the_name(self._session, net_name)
if network_ref is None:
raise Exception(_("Network with the name '%s' doesn't exist on"
" the ESX host") % net_name)
#Get the Size of the flat vmdk file that is there on the storage
#repository.
image_size, image_properties = \
vmware_images.get_vmdk_size_and_properties(instance.image_id,
instance)
vmdk_file_size_in_kb = int(image_size) / 1024
os_type = image_properties.get("vmware_ostype", "otherGuest")
adapter_type = image_properties.get("vmware_adaptertype", "lsiLogic")
# Get the datastore list and choose the first local storage def _check_if_network_bridge_exists():
data_stores = self._session._call_method(vim_util, "get_objects", network_ref = \
"Datastore", ["summary.type", "summary.name"]) NetworkHelper.get_network_with_the_name(self._session,
data_store_name = None net_name)
for elem in data_stores: if network_ref is None:
ds_name = None raise Exception(_("Network with the name '%s' doesn't exist on"
ds_type = None " the ESX host") % net_name)
for prop in elem.propSet:
if prop.name == "summary.type":
ds_type = prop.val
elif prop.name == "summary.name":
ds_name = prop.val
#Local storage identifier
if ds_type == "VMFS":
data_store_name = ds_name
break
if data_store_name is None: _check_if_network_bridge_exists()
msg = _("Couldn't get a local Datastore reference")
LOG.exception(msg) def _get_datastore_ref():
raise Exception(msg) # Get the datastore list and choose the first local storage
data_stores = self._session._call_method(vim_util, "get_objects",
"Datastore", ["summary.type", "summary.name"])
for elem in data_stores:
ds_name = None
ds_type = None
for prop in elem.propSet:
if prop.name == "summary.type":
ds_type = prop.val
elif prop.name == "summary.name":
ds_name = prop.val
#Local storage identifier
if ds_type == "VMFS":
data_store_name = ds_name
return data_store_name
if data_store_name is None:
msg = _("Couldn't get a local Datastore reference")
LOG.exception(msg)
raise Exception(msg)
data_store_name = _get_datastore_ref()
def _get_image_properties():
#Get the Size of the flat vmdk file that is there on the storage
#repository.
image_size, image_properties = \
vmware_images.get_vmdk_size_and_properties(
instance.image_id, instance)
vmdk_file_size_in_kb = int(image_size) / 1024
os_type = image_properties.get("vmware_ostype", "otherGuest")
adapter_type = image_properties.get("vmware_adaptertype",
"lsiLogic")
return vmdk_file_size_in_kb, os_type, adapter_type
vmdk_file_size_in_kb, os_type, adapter_type = _get_image_properties()
def _get_vmfolder_and_res_pool_mors():
#Get the Vm folder ref from the datacenter
dc_objs = self._session._call_method(vim_util, "get_objects",
"Datacenter", ["vmFolder"])
#There is only one default datacenter in a standalone ESX host
vm_folder_mor = dc_objs[0].propSet[0].val
#Get the resource pool. Taking the first resource pool coming our
#way. Assuming that is the default resource pool.
res_pool_mor = self._session._call_method(vim_util, "get_objects",
"ResourcePool")[0].obj
return vm_folder_mor, res_pool_mor
vm_folder_mor, res_pool_mor = _get_vmfolder_and_res_pool_mors()
#Get the create vm config spec
config_spec = vm_util.get_vm_create_spec(client_factory, instance, config_spec = vm_util.get_vm_create_spec(client_factory, instance,
data_store_name, net_name, os_type) data_store_name, net_name, os_type)
#Get the Vm folder ref from the datacenter def _execute_create_vm():
dc_objs = self._session._call_method(vim_util, "get_objects", LOG.debug(_("Creating VM with the name %s on the ESX host") %
"Datacenter", ["vmFolder"]) instance.name)
#There is only one default datacenter in a standalone ESX host #Create the VM on the ESX host
vm_folder_ref = dc_objs[0].propSet[0].val vm_create_task = self._session._call_method(
self._session._get_vim(),
"CreateVM_Task", vm_folder_mor,
config=config_spec, pool=res_pool_mor)
self._session._wait_for_task(instance.id, vm_create_task)
#Get the resource pool. Taking the first resource pool coming our way. LOG.debug(_("Created VM with the name %s on the ESX host") %
#Assuming that is the default resource pool. instance.name)
res_pool_mor = self._session._call_method(vim_util, "get_objects",
"ResourcePool")[0].obj
LOG.debug(_("Creating VM with the name %s on the ESX host") % _execute_create_vm()
instance.name)
#Create the VM on the ESX host
vm_create_task = self._session._call_method(self._session._get_vim(),
"CreateVM_Task", vm_folder_ref,
config=config_spec, pool=res_pool_mor)
self._session._wait_for_task(instance.id, vm_create_task)
LOG.debug(_("Created VM with the name %s on the ESX host") %
instance.name)
# Set the machine id for the VM for setting the IP # Set the machine id for the VM for setting the IP
self._set_machine_id(client_factory, instance) self._set_machine_id(client_factory, instance)
#Naming the VM files in correspondence with the VM instance name #Naming the VM files in correspondence with the VM instance name
# The flat vmdk file name # The flat vmdk file name
flat_uploaded_vmdk_name = "%s/%s-flat.vmdk" % (instance.name, flat_uploaded_vmdk_name = "%s/%s-flat.vmdk" % (instance.name,
instance.name) instance.name)
@ -183,79 +207,111 @@ class VMWareVMOps(object):
uploaded_vmdk_path = vm_util.build_datastore_path(data_store_name, uploaded_vmdk_path = vm_util.build_datastore_path(data_store_name,
uploaded_vmdk_name) uploaded_vmdk_name)
#Create a Virtual Disk of the size of the flat vmdk file. This is done def _create_virtual_disk():
#just created to generate the meta-data file whose specifics #Create a Virtual Disk of the size of the flat vmdk file. This is
#depend on the size of the disk, thin/thick provisioning and the #done just to generate the meta-data file whose specifics
#storage adapter type. #depend on the size of the disk, thin/thick provisioning and the
#Here we assume thick provisioning and lsiLogic for the adapter type #storage adapter type.
LOG.debug(_("Creating Virtual Disk of size %(vmdk_file_size_in_kb)s " #Here we assume thick provisioning and lsiLogic for the adapter
"KB and adapter type %(adapter_type)s on " #type
"the ESX host local store %(data_store_name)s") % locals()) LOG.debug(_("Creating Virtual Disk of size "
vmdk_create_spec = vm_util.get_vmdk_create_spec(client_factory, "%(vmdk_file_size_in_kb)s KB and adapter type "
vmdk_file_size_in_kb, adapter_type) "%(adapter_type)s on the ESX host local store"
vmdk_create_task = self._session._call_method(self._session._get_vim(), " %(data_store_name)s") %
"CreateVirtualDisk_Task", {"vmdk_file_size_in_kb": vmdk_file_size_in_kb,
self._session._get_vim().get_service_content().virtualDiskManager, "adapter_type": adapter_type,
name=uploaded_vmdk_path, "data_store_name": data_store_name})
datacenter=self._get_datacenter_name_and_ref()[0], vmdk_create_spec = vm_util.get_vmdk_create_spec(client_factory,
spec=vmdk_create_spec) vmdk_file_size_in_kb, adapter_type)
self._session._wait_for_task(instance.id, vmdk_create_task) vmdk_create_task = self._session._call_method(
LOG.debug(_("Created Virtual Disk of size %(vmdk_file_size_in_kb)s " self._session._get_vim(),
"KB on the ESX host local store " "CreateVirtualDisk_Task",
"%(data_store_name)s") % locals()) service_content.virtualDiskManager,
name=uploaded_vmdk_path,
datacenter=self._get_datacenter_name_and_ref()[0],
spec=vmdk_create_spec)
self._session._wait_for_task(instance.id, vmdk_create_task)
LOG.debug(_("Created Virtual Disk of size %(vmdk_file_size_in_kb)s"
" KB on the ESX host local store "
"%(data_store_name)s") %
{"vmdk_file_size_in_kb": vmdk_file_size_in_kb,
"data_store_name": data_store_name})
LOG.debug(_("Deleting the file %(flat_uploaded_vmdk_path)s " _create_virtual_disk()
"on the ESX host local"
"store %(data_store_name)s") % locals()) def _delete_disk_file():
#Delete the -flat.vmdk file created. .vmdk file is retained. LOG.debug(_("Deleting the file %(flat_uploaded_vmdk_path)s "
vmdk_delete_task = self._session._call_method(self._session._get_vim(), "on the ESX host local"
"DeleteDatastoreFile_Task", "store %(data_store_name)s") %
self._session._get_vim().get_service_content().fileManager, {"flat_uploaded_vmdk_path": flat_uploaded_vmdk_path,
name=flat_uploaded_vmdk_path) "data_store_name": data_store_name})
self._session._wait_for_task(instance.id, vmdk_delete_task) #Delete the -flat.vmdk file created. .vmdk file is retained.
LOG.debug(_("Deleted the file %(flat_uploaded_vmdk_path)s on the " vmdk_delete_task = self._session._call_method(
"ESX host local store %(data_store_name)s") % locals()) self._session._get_vim(),
"DeleteDatastoreFile_Task",
service_content.fileManager,
name=flat_uploaded_vmdk_path)
self._session._wait_for_task(instance.id, vmdk_delete_task)
LOG.debug(_("Deleted the file %(flat_uploaded_vmdk_path)s on the "
"ESX host local store %(data_store_name)s") %
{"flat_uploaded_vmdk_path": flat_uploaded_vmdk_path,
"data_store_name": data_store_name})
_delete_disk_file()
LOG.debug(_("Downloading image file data %(image_id)s to the ESX "
"data store %(data_store_name)s") %
({'image_id': instance.image_id,
'data_store_name': data_store_name}))
cookies = self._session._get_vim().client.options.transport.cookiejar cookies = self._session._get_vim().client.options.transport.cookiejar
# Upload the -flat.vmdk file whose meta-data file we just created above
vmware_images.fetch_image(
instance.image_id,
instance,
host=self._session._host_ip,
data_center_name=self._get_datacenter_name_and_ref()[1],
datastore_name=data_store_name,
cookies=cookies,
file_path=flat_uploaded_vmdk_name)
LOG.debug(_("Downloaded image file data %(image_id)s to the ESX "
"data store %(data_store_name)s") %
({'image_id': instance.image_id,
'data_store_name': data_store_name}))
#Attach the vmdk uploaded to the VM. VM reconfigure is done to do so. def _fetch_image_on_esx_datastore():
vmdk_attach_config_spec = vm_util.get_vmdk_attach_config_spec( LOG.debug(_("Downloading image file data %(image_id)s to the ESX "
client_factory, "data store %(data_store_name)s") %
vmdk_file_size_in_kb, uploaded_vmdk_path, ({'image_id': instance.image_id,
adapter_type) 'data_store_name': data_store_name}))
#Upload the -flat.vmdk file whose meta-data file we just created
#above
vmware_images.fetch_image(
instance.image_id,
instance,
host=self._session._host_ip,
data_center_name=self._get_datacenter_name_and_ref()[1],
datastore_name=data_store_name,
cookies=cookies,
file_path=flat_uploaded_vmdk_name)
LOG.debug(_("Downloaded image file data %(image_id)s to the ESX "
"data store %(data_store_name)s") %
({'image_id': instance.image_id,
'data_store_name': data_store_name}))
_fetch_image_on_esx_datastore()
vm_ref = self._get_vm_ref_from_the_name(instance.name) vm_ref = self._get_vm_ref_from_the_name(instance.name)
LOG.debug(_("Reconfiguring VM instance %s to attach the image "
"disk") % instance.name)
reconfig_task = self._session._call_method(self._session._get_vim(),
"ReconfigVM_Task", vm_ref,
spec=vmdk_attach_config_spec)
self._session._wait_for_task(instance.id, reconfig_task)
LOG.debug(_("Reconfigured VM instance %s to attach the image "
"disk") % instance.name)
LOG.debug(_("Powering on the VM instance %s") % instance.name) def _attach_vmdk_to_the_vm():
#Power On the VM #Attach the vmdk uploaded to the VM. VM reconfigure is done
power_on_task = self._session._call_method(self._session._get_vim(), #to do so.
"PowerOnVM_Task", vm_ref) vmdk_attach_config_spec = vm_util.get_vmdk_attach_config_spec(
self._session._wait_for_task(instance.id, power_on_task) client_factory,
LOG.debug(_("Powered on the VM instance %s") % instance.name) vmdk_file_size_in_kb, uploaded_vmdk_path,
adapter_type)
LOG.debug(_("Reconfiguring VM instance %s to attach the image "
"disk") % instance.name)
reconfig_task = self._session._call_method(
self._session._get_vim(),
"ReconfigVM_Task", vm_ref,
spec=vmdk_attach_config_spec)
self._session._wait_for_task(instance.id, reconfig_task)
LOG.debug(_("Reconfigured VM instance %s to attach the image "
"disk") % instance.name)
_attach_vmdk_to_the_vm()
def _power_on_vm():
LOG.debug(_("Powering on the VM instance %s") % instance.name)
#Power On the VM
power_on_task = self._session._call_method(
self._session._get_vim(),
"PowerOnVM_Task", vm_ref)
self._session._wait_for_task(instance.id, power_on_task)
LOG.debug(_("Powered on the VM instance %s") % instance.name)
_power_on_vm()
def snapshot(self, instance, snapshot_name): def snapshot(self, instance, snapshot_name):
""" """
@ -275,51 +331,65 @@ class VMWareVMOps(object):
if vm_ref is None: if vm_ref is None:
raise Exception(_("instance - %s not present") % instance.name) raise Exception(_("instance - %s not present") % instance.name)
#Get the vmdk file name that the VM is pointing to
hardware_devices = self._session._call_method(vim_util,
"get_dynamic_property", vm_ref,
"VirtualMachine", "config.hardware.device")
client_factory = self._session._get_vim().client.factory client_factory = self._session._get_vim().client.factory
vmdk_file_path_before_snapshot, adapter_type = \ service_content = self._session._get_vim().get_service_content()
vm_util.get_vmdk_file_path_and_adapter_type(client_factory,
hardware_devices)
os_type = self._session._call_method(vim_util, def _get_vm_and_vmdk_attribs():
"get_dynamic_property", vm_ref, #Get the vmdk file name that the VM is pointing to
"VirtualMachine", "summary.config.guestId") hardware_devices = self._session._call_method(vim_util,
#Create a snapshot of the VM "get_dynamic_property", vm_ref,
LOG.debug(_("Creating Snapshot of the VM instance %s ") % "VirtualMachine", "config.hardware.device")
instance.name) vmdk_file_path_before_snapshot, adapter_type = \
snapshot_task = self._session._call_method(self._session._get_vim(), vm_util.get_vmdk_file_path_and_adapter_type(client_factory,
"CreateSnapshot_Task", vm_ref, hardware_devices)
name="%s-snapshot" % instance.name, datastore_name = vm_util.split_datastore_path(
description="Taking Snapshot of the VM", vmdk_file_path_before_snapshot)[0]
memory=True, os_type = self._session._call_method(vim_util,
quiesce=True) "get_dynamic_property", vm_ref,
self._session._wait_for_task(instance.id, snapshot_task) "VirtualMachine", "summary.config.guestId")
LOG.debug(_("Created Snapshot of the VM instance %s ") % instance.name) return (vmdk_file_path_before_snapshot, adapter_type,
datastore_name, os_type)
datastore_name = vm_util.split_datastore_path( vmdk_file_path_before_snapshot, adapter_type, datastore_name,\
vmdk_file_path_before_snapshot)[0] os_type = _get_vm_and_vmdk_attribs()
#Copy the contents of the VM that were there just before the snapshot
#was taken def _create_vm_snapshot():
ds_ref = vim_util.get_dynamic_property(self._session._get_vim(), #Create a snapshot of the VM
LOG.debug(_("Creating Snapshot of the VM instance %s ") %
instance.name)
snapshot_task = self._session._call_method(
self._session._get_vim(),
"CreateSnapshot_Task", vm_ref,
name="%s-snapshot" % instance.name,
description="Taking Snapshot of the VM",
memory=True,
quiesce=True)
self._session._wait_for_task(instance.id, snapshot_task)
LOG.debug(_("Created Snapshot of the VM instance %s ") %
instance.name)
_create_vm_snapshot()
def _check_if_tmp_folder_exists():
#Copy the contents of the VM that were there just before the
#snapshot was taken
ds_ref = vim_util.get_dynamic_property(self._session._get_vim(),
vm_ref, vm_ref,
"VirtualMachine", "VirtualMachine",
"datastore").ManagedObjectReference[0] "datastore").ManagedObjectReference[0]
ds_browser = vim_util.get_dynamic_property(self._session._get_vim(), ds_browser = vim_util.get_dynamic_property(
self._session._get_vim(),
ds_ref, ds_ref,
"Datastore", "Datastore",
"browser") "browser")
#Check if the vmware-tmp folder exists or not. If not, create one #Check if the vmware-tmp folder exists or not. If not, create one
tmp_folder_path = vm_util.build_datastore_path(datastore_name, tmp_folder_path = vm_util.build_datastore_path(datastore_name,
"vmware-tmp") "vmware-tmp")
if not self._path_exists(ds_browser, tmp_folder_path): if not self._path_exists(ds_browser, tmp_folder_path):
self._mkdir(vm_util.build_datastore_path(datastore_name, self._mkdir(vm_util.build_datastore_path(datastore_name,
"vmware-tmp")) "vmware-tmp"))
copy_spec = vm_util.get_copy_virtual_disk_spec(client_factory, _check_if_tmp_folder_exists()
adapter_type)
#Generate a random vmdk file name to which the coalesced vmdk content #Generate a random vmdk file name to which the coalesced vmdk content
#will be copied to. A random name is chosen so that we don't have #will be copied to. A random name is chosen so that we don't have
@ -329,50 +399,64 @@ class VMWareVMOps(object):
"vmware-tmp/%s.vmdk" % random_name) "vmware-tmp/%s.vmdk" % random_name)
dc_ref = self._get_datacenter_name_and_ref()[0] dc_ref = self._get_datacenter_name_and_ref()[0]
#Copy the contents of the disk ( or disks, if there were snapshots def _copy_vmdk_content():
#done earlier) to a temporary vmdk file. #Copy the contents of the disk ( or disks, if there were snapshots
LOG.debug(_("Copying disk data before snapshot of the VM instance %s") #done earlier) to a temporary vmdk file.
% instance.name) copy_spec = vm_util.get_copy_virtual_disk_spec(client_factory,
copy_disk_task = self._session._call_method(self._session._get_vim(), adapter_type)
"CopyVirtualDisk_Task", LOG.debug(_("Copying disk data before snapshot of the VM "
self._session._get_vim().get_service_content().virtualDiskManager, " instance %s") % instance.name)
sourceName=vmdk_file_path_before_snapshot, copy_disk_task = self._session._call_method(
sourceDatacenter=dc_ref, self._session._get_vim(),
destName=dest_vmdk_file_location, "CopyVirtualDisk_Task",
destDatacenter=dc_ref, service_content.virtualDiskManager,
destSpec=copy_spec, sourceName=vmdk_file_path_before_snapshot,
force=False) sourceDatacenter=dc_ref,
self._session._wait_for_task(instance.id, copy_disk_task) destName=dest_vmdk_file_location,
LOG.debug(_("Copied disk data before snapshot of the VM instance %s") destDatacenter=dc_ref,
% instance.name) destSpec=copy_spec,
force=False)
self._session._wait_for_task(instance.id, copy_disk_task)
LOG.debug(_("Copied disk data before snapshot of the VM "
"instance %s") % instance.name)
_copy_vmdk_content()
cookies = self._session._get_vim().client.options.transport.cookiejar cookies = self._session._get_vim().client.options.transport.cookiejar
#Upload the contents of -flat.vmdk file which has the disk data.
LOG.debug(_("Uploading image %s") % snapshot_name)
vmware_images.upload_image(
snapshot_name,
instance,
os_type=os_type,
adapter_type=adapter_type,
image_version=1,
host=self._session._host_ip,
data_center_name=self._get_datacenter_name_and_ref()[1],
datastore_name=datastore_name,
cookies=cookies,
file_path="vmware-tmp/%s-flat.vmdk" % random_name)
LOG.debug(_("Uploaded image %s") % snapshot_name)
#Delete the temporary vmdk created above. def _upload_vmdk_to_image_repository():
LOG.debug(_("Deleting temporary vmdk file %s") #Upload the contents of -flat.vmdk file which has the disk data.
% dest_vmdk_file_location) LOG.debug(_("Uploading image %s") % snapshot_name)
remove_disk_task = self._session._call_method(self._session._get_vim(), vmware_images.upload_image(
"DeleteVirtualDisk_Task", snapshot_name,
self._session._get_vim().get_service_content().virtualDiskManager, instance,
name=dest_vmdk_file_location, os_type=os_type,
datacenter=dc_ref) adapter_type=adapter_type,
self._session._wait_for_task(instance.id, remove_disk_task) image_version=1,
LOG.debug(_("Deleted temporary vmdk file %s") host=self._session._host_ip,
% dest_vmdk_file_location) data_center_name=self._get_datacenter_name_and_ref()[1],
datastore_name=datastore_name,
cookies=cookies,
file_path="vmware-tmp/%s-flat.vmdk" % random_name)
LOG.debug(_("Uploaded image %s") % snapshot_name)
_upload_vmdk_to_image_repository()
def _clean_temp_data():
#Delete the temporary vmdk created above.
LOG.debug(_("Deleting temporary vmdk file %s")
% dest_vmdk_file_location)
remove_disk_task = self._session._call_method(
self._session._get_vim(),
"DeleteVirtualDisk_Task",
service_content.virtualDiskManager,
name=dest_vmdk_file_location,
datacenter=dc_ref)
self._session._wait_for_task(instance.id, remove_disk_task)
LOG.debug(_("Deleted temporary vmdk file %s")
% dest_vmdk_file_location)
_clean_temp_data()
def reboot(self, instance): def reboot(self, instance):
""" Reboot a VM instance """ """ Reboot a VM instance """

View File

@ -401,7 +401,7 @@ class SessionBase(object):
field in _db_content[cls][ref]): field in _db_content[cls][ref]):
return _db_content[cls][ref][field] return _db_content[cls][ref][field]
LOG.debuug(_('Raising NotImplemented')) LOG.debug(_('Raising NotImplemented'))
raise NotImplementedError( raise NotImplementedError(
_('xenapi.fake does not have an implementation for %s or it has ' _('xenapi.fake does not have an implementation for %s or it has '
'been called with the wrong number of arguments') % name) 'been called with the wrong number of arguments') % name)

View File

@ -86,7 +86,8 @@ class VMHelper(HelperBase):
the pv_kernel flag indicates whether the guest is HVM or PV the pv_kernel flag indicates whether the guest is HVM or PV
""" """
instance_type = instance_types.INSTANCE_TYPES[instance.instance_type] instance_type = instance_types.\
get_instance_type(instance.instance_type)
mem = str(long(instance_type['memory_mb']) * 1024 * 1024) mem = str(long(instance_type['memory_mb']) * 1024 * 1024)
vcpus = str(instance_type['vcpus']) vcpus = str(instance_type['vcpus'])
rec = { rec = {
@ -144,7 +145,8 @@ class VMHelper(HelperBase):
@classmethod @classmethod
def ensure_free_mem(cls, session, instance): def ensure_free_mem(cls, session, instance):
instance_type = instance_types.INSTANCE_TYPES[instance.instance_type] instance_type = instance_types.get_instance_type(
instance.instance_type)
mem = long(instance_type['memory_mb']) * 1024 * 1024 mem = long(instance_type['memory_mb']) * 1024 * 1024
#get free memory from host #get free memory from host
host = session.get_xenapi_host() host = session.get_xenapi_host()
@ -205,19 +207,17 @@ class VMHelper(HelperBase):
"""Destroy VBD from host database""" """Destroy VBD from host database"""
try: try:
task = session.call_xenapi('Async.VBD.destroy', vbd_ref) task = session.call_xenapi('Async.VBD.destroy', vbd_ref)
#FIXME(armando): find a solution to missing instance_id session.wait_for_task(task)
#with Josh Kearney
session.wait_for_task(0, task)
except cls.XenAPI.Failure, exc: except cls.XenAPI.Failure, exc:
LOG.exception(exc) LOG.exception(exc)
raise StorageError(_('Unable to destroy VBD %s') % vbd_ref) raise StorageError(_('Unable to destroy VBD %s') % vbd_ref)
@classmethod @classmethod
def create_vif(cls, session, vm_ref, network_ref, mac_address): def create_vif(cls, session, vm_ref, network_ref, mac_address, dev="0"):
"""Create a VIF record. Returns a Deferred that gives the new """Create a VIF record. Returns a Deferred that gives the new
VIF reference.""" VIF reference."""
vif_rec = {} vif_rec = {}
vif_rec['device'] = '0' vif_rec['device'] = dev
vif_rec['network'] = network_ref vif_rec['network'] = network_ref
vif_rec['VM'] = vm_ref vif_rec['VM'] = vm_ref
vif_rec['MAC'] = mac_address vif_rec['MAC'] = mac_address
@ -269,7 +269,7 @@ class VMHelper(HelperBase):
original_parent_uuid = get_vhd_parent_uuid(session, vm_vdi_ref) original_parent_uuid = get_vhd_parent_uuid(session, vm_vdi_ref)
task = session.call_xenapi('Async.VM.snapshot', vm_ref, label) task = session.call_xenapi('Async.VM.snapshot', vm_ref, label)
template_vm_ref = session.wait_for_task(instance_id, task) template_vm_ref = session.wait_for_task(task, instance_id)
template_vdi_rec = get_vdi_for_vm_safely(session, template_vm_ref)[1] template_vdi_rec = get_vdi_for_vm_safely(session, template_vm_ref)[1]
template_vdi_uuid = template_vdi_rec["uuid"] template_vdi_uuid = template_vdi_rec["uuid"]
@ -302,7 +302,7 @@ class VMHelper(HelperBase):
kwargs = {'params': pickle.dumps(params)} kwargs = {'params': pickle.dumps(params)}
task = session.async_call_plugin('glance', 'upload_vhd', kwargs) task = session.async_call_plugin('glance', 'upload_vhd', kwargs)
session.wait_for_task(instance_id, task) session.wait_for_task(task, instance_id)
@classmethod @classmethod
def fetch_image(cls, session, instance_id, image, user, project, def fetch_image(cls, session, instance_id, image, user, project,
@ -345,7 +345,7 @@ class VMHelper(HelperBase):
kwargs = {'params': pickle.dumps(params)} kwargs = {'params': pickle.dumps(params)}
task = session.async_call_plugin('glance', 'download_vhd', kwargs) task = session.async_call_plugin('glance', 'download_vhd', kwargs)
vdi_uuid = session.wait_for_task(instance_id, task) vdi_uuid = session.wait_for_task(task, instance_id)
scan_sr(session, instance_id, sr_ref) scan_sr(session, instance_id, sr_ref)
@ -401,7 +401,7 @@ class VMHelper(HelperBase):
#let the plugin copy the correct number of bytes #let the plugin copy the correct number of bytes
args['image-size'] = str(vdi_size) args['image-size'] = str(vdi_size)
task = session.async_call_plugin('glance', fn, args) task = session.async_call_plugin('glance', fn, args)
filename = session.wait_for_task(instance_id, task) filename = session.wait_for_task(task, instance_id)
#remove the VDI as it is not needed anymore #remove the VDI as it is not needed anymore
session.get_xenapi().VDI.destroy(vdi) session.get_xenapi().VDI.destroy(vdi)
LOG.debug(_("Kernel/Ramdisk VDI %s destroyed"), vdi) LOG.debug(_("Kernel/Ramdisk VDI %s destroyed"), vdi)
@ -493,7 +493,7 @@ class VMHelper(HelperBase):
if image_type == ImageType.DISK_RAW: if image_type == ImageType.DISK_RAW:
args['raw'] = 'true' args['raw'] = 'true'
task = session.async_call_plugin('objectstore', fn, args) task = session.async_call_plugin('objectstore', fn, args)
uuid = session.wait_for_task(instance_id, task) uuid = session.wait_for_task(task, instance_id)
return uuid return uuid
@classmethod @classmethod
@ -513,7 +513,7 @@ class VMHelper(HelperBase):
args = {} args = {}
args['vdi-ref'] = vdi_ref args['vdi-ref'] = vdi_ref
task = session.async_call_plugin('objectstore', fn, args) task = session.async_call_plugin('objectstore', fn, args)
pv_str = session.wait_for_task(instance_id, task) pv_str = session.wait_for_task(task, instance_id)
pv = None pv = None
if pv_str.lower() == 'true': if pv_str.lower() == 'true':
pv = True pv = True
@ -654,7 +654,7 @@ def get_vhd_parent_uuid(session, vdi_ref):
def scan_sr(session, instance_id, sr_ref): def scan_sr(session, instance_id, sr_ref):
LOG.debug(_("Re-scanning SR %s"), sr_ref) LOG.debug(_("Re-scanning SR %s"), sr_ref)
task = session.call_xenapi('Async.SR.scan', sr_ref) task = session.call_xenapi('Async.SR.scan', sr_ref)
session.wait_for_task(instance_id, task) session.wait_for_task(task, instance_id)
def wait_for_vhd_coalesce(session, instance_id, sr_ref, vdi_ref, def wait_for_vhd_coalesce(session, instance_id, sr_ref, vdi_ref,

View File

@ -49,6 +49,7 @@ class VMOps(object):
def __init__(self, session): def __init__(self, session):
self.XenAPI = session.get_imported_xenapi() self.XenAPI = session.get_imported_xenapi()
self._session = session self._session = session
VMHelper.XenAPI = self.XenAPI VMHelper.XenAPI = self.XenAPI
def list_instances(self): def list_instances(self):
@ -62,20 +63,20 @@ class VMOps(object):
def spawn(self, instance): def spawn(self, instance):
"""Create VM instance""" """Create VM instance"""
vm = VMHelper.lookup(self._session, instance.name) instance_name = instance.name
vm = VMHelper.lookup(self._session, instance_name)
if vm is not None: if vm is not None:
raise exception.Duplicate(_('Attempted to create' raise exception.Duplicate(_('Attempted to create'
' non-unique name %s') % instance.name) ' non-unique name %s') % instance_name)
#ensure enough free memory is available #ensure enough free memory is available
if not VMHelper.ensure_free_mem(self._session, instance): if not VMHelper.ensure_free_mem(self._session, instance):
name = instance['name'] LOG.exception(_('instance %(instance_name)s: not enough free '
LOG.exception(_('instance %(name)s: not enough free memory') 'memory') % locals())
% locals()) db.instance_set_state(context.get_admin_context(),
db.instance_set_state(context.get_admin_context(), instance['id'],
instance['id'], power_state.SHUTDOWN)
power_state.SHUTDOWN) return
return
user = AuthManager().get_user(instance.user_id) user = AuthManager().get_user(instance.user_id)
project = AuthManager().get_project(instance.project_id) project = AuthManager().get_project(instance.project_id)
@ -116,10 +117,9 @@ class VMOps(object):
self.create_vifs(instance, networks) self.create_vifs(instance, networks)
LOG.debug(_('Starting VM %s...'), vm_ref) LOG.debug(_('Starting VM %s...'), vm_ref)
self._session.call_xenapi('VM.start', vm_ref, False, False) self._start(instance, vm_ref)
instance_name = instance.name
LOG.info(_('Spawning VM %(instance_name)s created %(vm_ref)s.') LOG.info(_('Spawning VM %(instance_name)s created %(vm_ref)s.')
% locals()) % locals())
def _inject_onset_files(): def _inject_onset_files():
onset_files = instance.onset_files onset_files = instance.onset_files
@ -143,18 +143,18 @@ class VMOps(object):
def _wait_for_boot(): def _wait_for_boot():
try: try:
state = self.get_info(instance['name'])['state'] state = self.get_info(instance_name)['state']
db.instance_set_state(context.get_admin_context(), db.instance_set_state(context.get_admin_context(),
instance['id'], state) instance['id'], state)
if state == power_state.RUNNING: if state == power_state.RUNNING:
LOG.debug(_('Instance %s: booted'), instance['name']) LOG.debug(_('Instance %s: booted'), instance_name)
timer.stop() timer.stop()
_inject_onset_files() _inject_onset_files()
return True return True
except Exception, exc: except Exception, exc:
LOG.warn(exc) LOG.warn(exc)
LOG.exception(_('instance %s: failed to boot'), LOG.exception(_('instance %s: failed to boot'),
instance['name']) instance_name)
db.instance_set_state(context.get_admin_context(), db.instance_set_state(context.get_admin_context(),
instance['id'], instance['id'],
power_state.SHUTDOWN) power_state.SHUTDOWN)
@ -202,6 +202,20 @@ class VMOps(object):
_('Instance not present %s') % instance_name) _('Instance not present %s') % instance_name)
return vm return vm
def _acquire_bootlock(self, vm):
"""Prevent an instance from booting"""
self._session.call_xenapi(
"VM.set_blocked_operations",
vm,
{"start": ""})
def _release_bootlock(self, vm):
"""Allow an instance to boot"""
self._session.call_xenapi(
"VM.remove_from_blocked_operations",
vm,
"start")
def snapshot(self, instance, image_id): def snapshot(self, instance, image_id):
""" Create snapshot from a running VM instance """ Create snapshot from a running VM instance
@ -254,7 +268,7 @@ class VMOps(object):
"""Reboot VM instance""" """Reboot VM instance"""
vm = self._get_vm_opaque_ref(instance) vm = self._get_vm_opaque_ref(instance)
task = self._session.call_xenapi('Async.VM.clean_reboot', vm) task = self._session.call_xenapi('Async.VM.clean_reboot', vm)
self._session.wait_for_task(instance.id, task) self._session.wait_for_task(task, instance.id)
def set_admin_password(self, instance, new_pass): def set_admin_password(self, instance, new_pass):
"""Set the root/admin password on the VM instance. This is done via """Set the root/admin password on the VM instance. This is done via
@ -294,6 +308,11 @@ class VMOps(object):
raise RuntimeError(resp_dict['message']) raise RuntimeError(resp_dict['message'])
return resp_dict['message'] return resp_dict['message']
def _start(self, instance, vm):
"""Start an instance"""
task = self._session.call_xenapi("Async.VM.start", vm, False, False)
self._session.wait_for_task(task, instance.id)
def inject_file(self, instance, b64_path, b64_contents): def inject_file(self, instance, b64_path, b64_contents):
"""Write a file to the VM instance. The path to which it is to be """Write a file to the VM instance. The path to which it is to be
written and the contents of the file need to be supplied; both should written and the contents of the file need to be supplied; both should
@ -320,8 +339,8 @@ class VMOps(object):
raise RuntimeError(resp_dict['message']) raise RuntimeError(resp_dict['message'])
return resp_dict['message'] return resp_dict['message']
def _shutdown(self, instance, vm): def _shutdown(self, instance, vm, hard=True):
"""Shutdown an instance """ """Shutdown an instance"""
state = self.get_info(instance['name'])['state'] state = self.get_info(instance['name'])['state']
if state == power_state.SHUTDOWN: if state == power_state.SHUTDOWN:
LOG.warn(_("VM %(vm)s already halted, skipping shutdown...") % LOG.warn(_("VM %(vm)s already halted, skipping shutdown...") %
@ -332,8 +351,13 @@ class VMOps(object):
LOG.debug(_("Shutting down VM for Instance %(instance_id)s") LOG.debug(_("Shutting down VM for Instance %(instance_id)s")
% locals()) % locals())
try: try:
task = self._session.call_xenapi('Async.VM.hard_shutdown', vm) task = None
self._session.wait_for_task(instance.id, task) if hard:
task = self._session.call_xenapi("Async.VM.hard_shutdown", vm)
else:
task = self._session.call_xenapi("Async.VM.clean_shutdown", vm)
self._session.wait_for_task(task, instance.id)
except self.XenAPI.Failure, exc: except self.XenAPI.Failure, exc:
LOG.exception(exc) LOG.exception(exc)
@ -350,7 +374,7 @@ class VMOps(object):
for vdi in vdis: for vdi in vdis:
try: try:
task = self._session.call_xenapi('Async.VDI.destroy', vdi) task = self._session.call_xenapi('Async.VDI.destroy', vdi)
self._session.wait_for_task(instance.id, task) self._session.wait_for_task(task, instance.id)
except self.XenAPI.Failure, exc: except self.XenAPI.Failure, exc:
LOG.exception(exc) LOG.exception(exc)
@ -389,7 +413,7 @@ class VMOps(object):
args = {'kernel-file': kernel, 'ramdisk-file': ramdisk} args = {'kernel-file': kernel, 'ramdisk-file': ramdisk}
task = self._session.async_call_plugin( task = self._session.async_call_plugin(
'glance', 'remove_kernel_ramdisk', args) 'glance', 'remove_kernel_ramdisk', args)
self._session.wait_for_task(instance.id, task) self._session.wait_for_task(task, instance.id)
LOG.debug(_("kernel/ramdisk files removed")) LOG.debug(_("kernel/ramdisk files removed"))
@ -398,7 +422,7 @@ class VMOps(object):
instance_id = instance.id instance_id = instance.id
try: try:
task = self._session.call_xenapi('Async.VM.destroy', vm) task = self._session.call_xenapi('Async.VM.destroy', vm)
self._session.wait_for_task(instance_id, task) self._session.wait_for_task(task, instance_id)
except self.XenAPI.Failure, exc: except self.XenAPI.Failure, exc:
LOG.exception(exc) LOG.exception(exc)
@ -441,7 +465,7 @@ class VMOps(object):
def _wait_with_callback(self, instance_id, task, callback): def _wait_with_callback(self, instance_id, task, callback):
ret = None ret = None
try: try:
ret = self._session.wait_for_task(instance_id, task) ret = self._session.wait_for_task(task, instance_id)
except self.XenAPI.Failure, exc: except self.XenAPI.Failure, exc:
LOG.exception(exc) LOG.exception(exc)
callback(ret) callback(ret)
@ -470,6 +494,78 @@ class VMOps(object):
task = self._session.call_xenapi('Async.VM.resume', vm, False, True) task = self._session.call_xenapi('Async.VM.resume', vm, False, True)
self._wait_with_callback(instance.id, task, callback) self._wait_with_callback(instance.id, task, callback)
def rescue(self, instance, callback):
"""Rescue the specified instance
- shutdown the instance VM
- set 'bootlock' to prevent the instance from starting in rescue
- spawn a rescue VM (the vm name-label will be instance-N-rescue)
"""
rescue_vm = VMHelper.lookup(self._session, instance.name + "-rescue")
if rescue_vm:
raise RuntimeError(_(
"Instance is already in Rescue Mode: %s" % instance.name))
vm = self._get_vm_opaque_ref(instance)
self._shutdown(instance, vm)
self._acquire_bootlock(vm)
instance._rescue = True
self.spawn(instance)
rescue_vm = self._get_vm_opaque_ref(instance)
vbd = self._session.get_xenapi().VM.get_VBDs(vm)[0]
vdi_ref = self._session.get_xenapi().VBD.get_record(vbd)["VDI"]
vbd_ref = VMHelper.create_vbd(
self._session,
rescue_vm,
vdi_ref,
1,
False)
self._session.call_xenapi("Async.VBD.plug", vbd_ref)
def unrescue(self, instance, callback):
"""Unrescue the specified instance
- unplug the instance VM's disk from the rescue VM
- teardown the rescue VM
- release the bootlock to allow the instance VM to start
"""
rescue_vm = VMHelper.lookup(self._session, instance.name + "-rescue")
if not rescue_vm:
raise exception.NotFound(_(
"Instance is not in Rescue Mode: %s" % instance.name))
original_vm = self._get_vm_opaque_ref(instance)
vbds = self._session.get_xenapi().VM.get_VBDs(rescue_vm)
instance._rescue = False
for vbd_ref in vbds:
vbd = self._session.get_xenapi().VBD.get_record(vbd_ref)
if vbd["userdevice"] == "1":
VMHelper.unplug_vbd(self._session, vbd_ref)
VMHelper.destroy_vbd(self._session, vbd_ref)
task1 = self._session.call_xenapi("Async.VM.hard_shutdown", rescue_vm)
self._session.wait_for_task(task1, instance.id)
vdis = VMHelper.lookup_vm_vdis(self._session, rescue_vm)
for vdi in vdis:
try:
task = self._session.call_xenapi('Async.VDI.destroy', vdi)
self._session.wait_for_task(task, instance.id)
except self.XenAPI.Failure:
continue
task2 = self._session.call_xenapi('Async.VM.destroy', rescue_vm)
self._session.wait_for_task(task2, instance.id)
self._release_bootlock(original_vm)
self._start(instance, original_vm)
def get_info(self, instance): def get_info(self, instance):
"""Return data about VM instance""" """Return data about VM instance"""
vm = self._get_vm_opaque_ref(instance) vm = self._get_vm_opaque_ref(instance)
@ -514,18 +610,30 @@ class VMOps(object):
network_IPs = [ip for ip in IPs if ip.network_id == network.id] network_IPs = [ip for ip in IPs if ip.network_id == network.id]
def ip_dict(ip): def ip_dict(ip):
return {'netmask': network['netmask'], return {
'enabled': '1', "ip": ip.address,
'ip': ip.address} "netmask": network["netmask"],
"enabled": "1"}
def ip6_dict(ip6):
return {
"ip": ip6.addressV6,
"netmask": ip6.netmaskV6,
"gateway": ip6.gatewayV6,
"enabled": "1"}
mac_id = instance.mac_address.replace(':', '') mac_id = instance.mac_address.replace(':', '')
location = 'vm-data/networking/%s' % mac_id location = 'vm-data/networking/%s' % mac_id
mapping = {'label': network['label'], mapping = {
'gateway': network['gateway'], 'label': network['label'],
'mac': instance.mac_address, 'gateway': network['gateway'],
'dns': [network['dns']], 'mac': instance.mac_address,
'ips': [ip_dict(ip) for ip in network_IPs]} 'dns': [network['dns']],
'ips': [ip_dict(ip) for ip in network_IPs],
'ip6s': [ip6_dict(ip) for ip in network_IPs]}
self.write_to_param_xenstore(vm_opaque_ref, {location: mapping}) self.write_to_param_xenstore(vm_opaque_ref, {location: mapping})
try: try:
self.write_to_xenstore(vm_opaque_ref, location, self.write_to_xenstore(vm_opaque_ref, location,
mapping['location']) mapping['location'])
@ -556,8 +664,17 @@ class VMOps(object):
NetworkHelper.find_network_with_bridge(self._session, bridge) NetworkHelper.find_network_with_bridge(self._session, bridge)
if network_ref: if network_ref:
VMHelper.create_vif(self._session, vm_opaque_ref, try:
network_ref, instance.mac_address) device = "1" if instance._rescue else "0"
except AttributeError:
device = "0"
VMHelper.create_vif(
self._session,
vm_opaque_ref,
network_ref,
instance.mac_address,
device)
def reset_network(self, instance): def reset_network(self, instance):
""" """
@ -627,7 +744,7 @@ class VMOps(object):
args.update(addl_args) args.update(addl_args)
try: try:
task = self._session.async_call_plugin(plugin, method, args) task = self._session.async_call_plugin(plugin, method, args)
ret = self._session.wait_for_task(instance_id, task) ret = self._session.wait_for_task(task, instance_id)
except self.XenAPI.Failure, e: except self.XenAPI.Failure, e:
ret = None ret = None
err_trace = e.details[-1] err_trace = e.details[-1]

View File

@ -83,7 +83,7 @@ class VolumeOps(object):
try: try:
task = self._session.call_xenapi('Async.VBD.plug', task = self._session.call_xenapi('Async.VBD.plug',
vbd_ref) vbd_ref)
self._session.wait_for_task(vol_rec['deviceNumber'], task) self._session.wait_for_task(task, vol_rec['deviceNumber'])
except self.XenAPI.Failure, exc: except self.XenAPI.Failure, exc:
LOG.exception(exc) LOG.exception(exc)
VolumeHelper.destroy_iscsi_storage(self._session, VolumeHelper.destroy_iscsi_storage(self._session,

View File

@ -196,6 +196,14 @@ class XenAPIConnection(object):
"""resume the specified instance""" """resume the specified instance"""
self._vmops.resume(instance, callback) self._vmops.resume(instance, callback)
def rescue(self, instance, callback):
"""Rescue the specified instance"""
self._vmops.rescue(instance, callback)
def unrescue(self, instance, callback):
"""Unrescue the specified instance"""
self._vmops.unrescue(instance, callback)
def reset_network(self, instance): def reset_network(self, instance):
"""reset networking for specified instance""" """reset networking for specified instance"""
self._vmops.reset_network(instance) self._vmops.reset_network(instance)
@ -279,7 +287,7 @@ class XenAPISession(object):
self._session.xenapi.Async.host.call_plugin, self._session.xenapi.Async.host.call_plugin,
self.get_xenapi_host(), plugin, fn, args) self.get_xenapi_host(), plugin, fn, args)
def wait_for_task(self, id, task): def wait_for_task(self, task, id=None):
"""Return the result of the given task. The task is polled """Return the result of the given task. The task is polled
until it completes. Not re-entrant.""" until it completes. Not re-entrant."""
done = event.Event() done = event.Event()
@ -306,10 +314,11 @@ class XenAPISession(object):
try: try:
name = self._session.xenapi.task.get_name_label(task) name = self._session.xenapi.task.get_name_label(task)
status = self._session.xenapi.task.get_status(task) status = self._session.xenapi.task.get_status(task)
action = dict( if id:
instance_id=int(id), action = dict(
action=name[0:255], # Ensure action is never > 255 instance_id=int(id),
error=None) action=name[0:255], # Ensure action is never > 255
error=None)
if status == "pending": if status == "pending":
return return
elif status == "success": elif status == "success":
@ -323,7 +332,9 @@ class XenAPISession(object):
LOG.warn(_("Task [%(name)s] %(task)s status:" LOG.warn(_("Task [%(name)s] %(task)s status:"
" %(status)s %(error_info)s") % locals()) " %(status)s %(error_info)s") % locals())
done.send_exception(self.XenAPI.Failure(error_info)) done.send_exception(self.XenAPI.Failure(error_info))
db.instance_action_create(context.get_admin_context(), action)
if id:
db.instance_action_create(context.get_admin_context(), action)
except self.XenAPI.Failure, exc: except self.XenAPI.Failure, exc:
LOG.warn(exc) LOG.warn(exc)
done.send_exception(*sys.exc_info()) done.send_exception(*sys.exc_info())

View File

@ -207,8 +207,7 @@ def _upload_tarball(staging_path, image_id, glance_host, glance_port):
'transfer-encoding': 'chunked', 'transfer-encoding': 'chunked',
'x-image-meta-is_public': 'True', 'x-image-meta-is_public': 'True',
'x-image-meta-status': 'queued', 'x-image-meta-status': 'queued',
'x-image-meta-type': 'vhd' 'x-image-meta-type': 'vhd'}
}
for header, value in headers.iteritems(): for header, value in headers.iteritems():
conn.putheader(header, value) conn.putheader(header, value)
conn.endheaders() conn.endheaders()

View File

@ -29,3 +29,4 @@ sqlalchemy-migrate
netaddr netaddr
sphinx sphinx
glance glance
suds==0.4