Merged with trunk
This commit is contained in:
		| @@ -29,8 +29,6 @@ from nova import flags | |||||||
| from nova import rpc | from nova import rpc | ||||||
| from nova import server | from nova import server | ||||||
| from nova import utils | from nova import utils | ||||||
| from nova.auth import manager |  | ||||||
| from nova.compute import model |  | ||||||
| from nova.endpoint import admin | from nova.endpoint import admin | ||||||
| from nova.endpoint import api | from nova.endpoint import api | ||||||
| from nova.endpoint import cloud | from nova.endpoint import cloud | ||||||
| @@ -39,10 +37,10 @@ FLAGS = flags.FLAGS | |||||||
|  |  | ||||||
|  |  | ||||||
| def main(_argv): | def main(_argv): | ||||||
|  |     """Load the controllers and start the tornado I/O loop.""" | ||||||
|     controllers = { |     controllers = { | ||||||
|         'Cloud': cloud.CloudController(), |         'Cloud': cloud.CloudController(), | ||||||
|         'Admin': admin.AdminController() |         'Admin': admin.AdminController()} | ||||||
|     } |  | ||||||
|     _app = api.APIServerApplication(controllers) |     _app = api.APIServerApplication(controllers) | ||||||
|  |  | ||||||
|     conn = rpc.Connection.instance() |     conn = rpc.Connection.instance() | ||||||
|   | |||||||
							
								
								
									
										34
									
								
								bin/nova-api-new
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										34
									
								
								bin/nova-api-new
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,34 @@ | |||||||
|  | #!/usr/bin/env python | ||||||
|  | # pylint: disable-msg=C0103 | ||||||
|  | # vim: tabstop=4 shiftwidth=4 softtabstop=4 | ||||||
|  |  | ||||||
|  | # Copyright 2010 United States Government as represented by the | ||||||
|  | # Administrator of the National Aeronautics and Space Administration. | ||||||
|  | # 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. | ||||||
|  | """ | ||||||
|  | Nova API daemon. | ||||||
|  | """ | ||||||
|  |  | ||||||
|  | from nova import api | ||||||
|  | from nova import flags | ||||||
|  | from nova import utils | ||||||
|  | from nova import wsgi | ||||||
|  |  | ||||||
|  | FLAGS = flags.FLAGS | ||||||
|  | flags.DEFINE_integer('api_port', 8773, 'API port') | ||||||
|  |  | ||||||
|  | if __name__ == '__main__': | ||||||
|  |     utils.default_flagfile() | ||||||
|  |     wsgi.run_server(api.API(), FLAGS.api_port) | ||||||
| @@ -18,8 +18,6 @@ | |||||||
| #    under the License. | #    under the License. | ||||||
|  |  | ||||||
| """ | """ | ||||||
| nova-dhcpbridge |  | ||||||
|  |  | ||||||
| Handle lease database updates from DHCP servers. | Handle lease database updates from DHCP servers. | ||||||
| """ | """ | ||||||
|  |  | ||||||
| @@ -42,34 +40,42 @@ from nova.network import service | |||||||
| FLAGS = flags.FLAGS | FLAGS = flags.FLAGS | ||||||
|  |  | ||||||
|  |  | ||||||
| def add_lease(mac, ip, hostname, interface): | def add_lease(_mac, ip, _hostname, _interface): | ||||||
|  |     """Set the IP that was assigned by the DHCP server.""" | ||||||
|     if FLAGS.fake_rabbit: |     if FLAGS.fake_rabbit: | ||||||
|         service.VlanNetworkService().lease_ip(ip) |         service.VlanNetworkService().lease_ip(ip) | ||||||
|     else: |     else: | ||||||
|         rpc.cast("%s.%s" (FLAGS.network_topic, FLAGS.node_name), |         rpc.cast("%s.%s" % (FLAGS.network_topic, FLAGS.node_name), | ||||||
|                  {"method": "lease_ip", |                  {"method": "lease_ip", | ||||||
|                   "args": {"fixed_ip": ip}}) |                   "args": {"fixed_ip": ip}}) | ||||||
|  |  | ||||||
| def old_lease(mac, ip, hostname, interface): |  | ||||||
|  | def old_lease(_mac, _ip, _hostname, _interface): | ||||||
|  |     """Do nothing, just an old lease update.""" | ||||||
|     logging.debug("Adopted old lease or got a change of mac/hostname") |     logging.debug("Adopted old lease or got a change of mac/hostname") | ||||||
|  |  | ||||||
| def del_lease(mac, ip, hostname, interface): |  | ||||||
|  | def del_lease(_mac, ip, _hostname, _interface): | ||||||
|  |     """Called when a lease expires.""" | ||||||
|     if FLAGS.fake_rabbit: |     if FLAGS.fake_rabbit: | ||||||
|         service.VlanNetworkService().release_ip(ip) |         service.VlanNetworkService().release_ip(ip) | ||||||
|     else: |     else: | ||||||
|         rpc.cast("%s.%s" (FLAGS.network_topic, FLAGS.node_name), |         rpc.cast("%s.%s" % (FLAGS.network_topic, FLAGS.node_name), | ||||||
|                  {"method": "release_ip", |                  {"method": "release_ip", | ||||||
|                   "args": {"fixed_ip": ip}}) |                   "args": {"fixed_ip": ip}}) | ||||||
|  |  | ||||||
|  |  | ||||||
| def init_leases(interface): | def init_leases(interface): | ||||||
|  |     """Get the list of hosts for an interface.""" | ||||||
|     net = model.get_network_by_interface(interface) |     net = model.get_network_by_interface(interface) | ||||||
|     res = "" |     res = "" | ||||||
|     for host_name in net.hosts: |     for address in net.assigned_objs: | ||||||
|         res += "%s\n" % linux_net.hostDHCP(net, host_name, net.hosts[host_name]) |         res += "%s\n" % linux_net.host_dhcp(address) | ||||||
|     return res |     return res | ||||||
|  |  | ||||||
|  |  | ||||||
| def main(): | def main(): | ||||||
|  |     """Parse environment and arguments and call the approproate action.""" | ||||||
|     flagfile = os.environ.get('FLAGFILE', FLAGS.dhcpbridge_flagfile) |     flagfile = os.environ.get('FLAGFILE', FLAGS.dhcpbridge_flagfile) | ||||||
|     utils.default_flagfile(flagfile) |     utils.default_flagfile(flagfile) | ||||||
|     argv = FLAGS(sys.argv) |     argv = FLAGS(sys.argv) | ||||||
| @@ -86,11 +92,12 @@ def main(): | |||||||
|         mac = argv[2] |         mac = argv[2] | ||||||
|         ip = argv[3] |         ip = argv[3] | ||||||
|         hostname = argv[4] |         hostname = argv[4] | ||||||
|         logging.debug("Called %s for mac %s with ip %s and hostname %s on interface %s" % (action, mac, ip, hostname, interface)) |         logging.debug("Called %s for mac %s with ip %s and " | ||||||
|  |                       "hostname %s on interface %s", | ||||||
|  |                       action, mac, ip, hostname, interface) | ||||||
|         globals()[action + '_lease'](mac, ip, hostname, interface) |         globals()[action + '_lease'](mac, ip, hostname, interface) | ||||||
|     else: |     else: | ||||||
|         print init_leases(interface) |         print init_leases(interface) | ||||||
|     exit(0) |  | ||||||
|  |  | ||||||
| if __name__ == "__main__": | if __name__ == "__main__": | ||||||
|     sys.exit(main()) |     main() | ||||||
|   | |||||||
| @@ -37,20 +37,17 @@ FLAGS = flags.FLAGS | |||||||
|  |  | ||||||
| api_url = 'https://imagestore.canonical.com/api/dashboard' | api_url = 'https://imagestore.canonical.com/api/dashboard' | ||||||
|  |  | ||||||
| image_cache = None |  | ||||||
| def images(): |  | ||||||
|     global image_cache |  | ||||||
|     if not image_cache: |  | ||||||
|         try: |  | ||||||
|             images = json.load(urllib2.urlopen(api_url))['images'] |  | ||||||
|             image_cache = [i for i in images if i['title'].find('amd64') > -1] |  | ||||||
|         except Exception: |  | ||||||
|             print 'unable to download canonical image list' |  | ||||||
|             sys.exit(1) |  | ||||||
|     return image_cache |  | ||||||
|  |  | ||||||
| # FIXME(ja): add checksum/signature checks | def get_images(): | ||||||
|  |     """Get a list of the images from the imagestore URL.""" | ||||||
|  |     images = json.load(urllib2.urlopen(api_url))['images'] | ||||||
|  |     images = [img for img in images if img['title'].find('amd64') > -1] | ||||||
|  |     return images | ||||||
|  |  | ||||||
|  |  | ||||||
| def download(img): | def download(img): | ||||||
|  |     """Download an image to the local filesystem.""" | ||||||
|  |     # FIXME(ja): add checksum/signature checks | ||||||
|     tempdir = tempfile.mkdtemp(prefix='cis-') |     tempdir = tempfile.mkdtemp(prefix='cis-') | ||||||
|  |  | ||||||
|     kernel_id = None |     kernel_id = None | ||||||
| @@ -79,20 +76,22 @@ def download(img): | |||||||
|  |  | ||||||
|     shutil.rmtree(tempdir) |     shutil.rmtree(tempdir) | ||||||
|  |  | ||||||
|  |  | ||||||
| def main(): | def main(): | ||||||
|  |     """Main entry point.""" | ||||||
|     utils.default_flagfile() |     utils.default_flagfile() | ||||||
|     argv = FLAGS(sys.argv) |     argv = FLAGS(sys.argv) | ||||||
|  |     images = get_images() | ||||||
|  |  | ||||||
|     if len(argv) == 2: |     if len(argv) == 2: | ||||||
|         for img in images(): |         for img in images: | ||||||
|             if argv[1] == 'all' or argv[1] == img['title']: |             if argv[1] == 'all' or argv[1] == img['title']: | ||||||
|                 download(img) |                 download(img) | ||||||
|     else: |     else: | ||||||
|         print 'usage: %s (title|all)' |         print 'usage: %s (title|all)' | ||||||
|         print 'available images:' |         print 'available images:' | ||||||
|         for image in images(): |         for img in images: | ||||||
|             print image['title'] |             print img['title'] | ||||||
|  |  | ||||||
| if __name__ == '__main__': | if __name__ == '__main__': | ||||||
|     main() |     main() | ||||||
|  |  | ||||||
|   | |||||||
| @@ -22,7 +22,6 @@ | |||||||
| """ | """ | ||||||
|  |  | ||||||
| import logging | import logging | ||||||
| from twisted.internet import task |  | ||||||
| from twisted.application import service | from twisted.application import service | ||||||
|  |  | ||||||
| from nova import twistd | from nova import twistd | ||||||
| @@ -30,7 +29,11 @@ from nova.compute import monitor | |||||||
|  |  | ||||||
| logging.getLogger('boto').setLevel(logging.WARN) | logging.getLogger('boto').setLevel(logging.WARN) | ||||||
|  |  | ||||||
| def main(): |  | ||||||
|  | if __name__ == '__main__': | ||||||
|  |     twistd.serve(__file__) | ||||||
|  |  | ||||||
|  | if __name__ == '__builtin__': | ||||||
|     logging.warn('Starting instance monitor') |     logging.warn('Starting instance monitor') | ||||||
|     m = monitor.InstanceMonitor() |     m = monitor.InstanceMonitor() | ||||||
|  |  | ||||||
| @@ -38,14 +41,3 @@ def main(): | |||||||
|     # parses this file, return it so that we can get it into globals below |     # parses this file, return it so that we can get it into globals below | ||||||
|     application = service.Application('nova-instancemonitor') |     application = service.Application('nova-instancemonitor') | ||||||
|     m.setServiceParent(application) |     m.setServiceParent(application) | ||||||
|     return application |  | ||||||
|  |  | ||||||
| if __name__ == '__main__': |  | ||||||
|     twistd.serve(__file__) |  | ||||||
|  |  | ||||||
| if __name__ == '__builtin__': |  | ||||||
|     application = main() |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										135
									
								
								bin/nova-manage
									
									
									
									
									
								
							
							
						
						
									
										135
									
								
								bin/nova-manage
									
									
									
									
									
								
							| @@ -37,12 +37,15 @@ FLAGS = flags.FLAGS | |||||||
|  |  | ||||||
|  |  | ||||||
| class VpnCommands(object): | class VpnCommands(object): | ||||||
|  |     """Class for managing VPNs.""" | ||||||
|  |  | ||||||
|     def __init__(self): |     def __init__(self): | ||||||
|         self.manager = manager.AuthManager() |         self.manager = manager.AuthManager() | ||||||
|         self.instdir = model.InstanceDirectory() |         self.instdir = model.InstanceDirectory() | ||||||
|         self.pipe = pipelib.CloudPipe(cloud.CloudController()) |         self.pipe = pipelib.CloudPipe(cloud.CloudController()) | ||||||
|  |  | ||||||
|     def list(self): |     def list(self): | ||||||
|  |         """Print a listing of the VPNs for all projects.""" | ||||||
|         print "%-12s\t" % 'project', |         print "%-12s\t" % 'project', | ||||||
|         print "%-12s\t" % 'ip:port', |         print "%-12s\t" % 'ip:port', | ||||||
|         print "%s" % 'state' |         print "%s" % 'state' | ||||||
| @@ -50,11 +53,11 @@ class VpnCommands(object): | |||||||
|             print "%-12s\t" % project.name, |             print "%-12s\t" % project.name, | ||||||
|             print "%s:%s\t" % (project.vpn_ip, project.vpn_port), |             print "%s:%s\t" % (project.vpn_ip, project.vpn_port), | ||||||
|  |  | ||||||
|             vpn = self.__vpn_for(project.id) |             vpn = self._vpn_for(project.id) | ||||||
|             if vpn: |             if vpn: | ||||||
|                 out, err = utils.execute( |                 command = "ping -c1 -w1 %s > /dev/null; echo $?" | ||||||
|                 			"ping -c1 -w1 %s > /dev/null; echo $?" |                 out, _err = utils.execute(  command % vpn['private_dns_name'], | ||||||
|                 			% vpn['private_dns_name'], check_exit_code=False) |                                             check_exit_code=False) | ||||||
|                 if out.strip() == '0': |                 if out.strip() == '0': | ||||||
|                     net = 'up' |                     net = 'up' | ||||||
|                 else: |                 else: | ||||||
| @@ -68,25 +71,32 @@ class VpnCommands(object): | |||||||
|             else: |             else: | ||||||
|                 print None |                 print None | ||||||
|  |  | ||||||
|     def __vpn_for(self, project_id): |     def _vpn_for(self, project_id): | ||||||
|  |         """Get the VPN instance for a project ID.""" | ||||||
|         for instance in self.instdir.all: |         for instance in self.instdir.all: | ||||||
|             if (instance.state.has_key('image_id') |             if ('image_id' in instance.state | ||||||
|                 and instance['image_id'] == FLAGS.vpn_image_id |                 and instance['image_id'] == FLAGS.vpn_image_id | ||||||
|                 and not instance['state_description'] in ['shutting_down', 'shutdown'] |                 and not instance['state_description'] in | ||||||
|  |                     ['shutting_down', 'shutdown'] | ||||||
|                 and instance['project_id'] == project_id): |                 and instance['project_id'] == project_id): | ||||||
|                 return instance |                 return instance | ||||||
|  |  | ||||||
|     def spawn(self): |     def spawn(self): | ||||||
|  |         """Run all VPNs.""" | ||||||
|         for p in reversed(self.manager.get_projects()): |         for p in reversed(self.manager.get_projects()): | ||||||
|              if not self.__vpn_for(p.id): |             if not self._vpn_for(p.id): | ||||||
|                 print 'spawning %s' % p.id |                 print 'spawning %s' % p.id | ||||||
|                 self.pipe.launch_vpn_instance(p.id) |                 self.pipe.launch_vpn_instance(p.id) | ||||||
|                 time.sleep(10) |                 time.sleep(10) | ||||||
|  |  | ||||||
|     def run(self, project_id): |     def run(self, project_id): | ||||||
|  |         """Start the VPN for a given project.""" | ||||||
|         self.pipe.launch_vpn_instance(project_id) |         self.pipe.launch_vpn_instance(project_id) | ||||||
|  |  | ||||||
|  |  | ||||||
| class RoleCommands(object): | class RoleCommands(object): | ||||||
|  |     """Class for managing roles.""" | ||||||
|  |  | ||||||
|     def __init__(self): |     def __init__(self): | ||||||
|         self.manager = manager.AuthManager() |         self.manager = manager.AuthManager() | ||||||
|  |  | ||||||
| @@ -109,25 +119,24 @@ class RoleCommands(object): | |||||||
|         arguments: user, role [project]""" |         arguments: user, role [project]""" | ||||||
|         self.manager.remove_role(user, role, project) |         self.manager.remove_role(user, role, project) | ||||||
|  |  | ||||||
|  |  | ||||||
| class UserCommands(object): | class UserCommands(object): | ||||||
|  |     """Class for managing users.""" | ||||||
|  |  | ||||||
|     def __init__(self): |     def __init__(self): | ||||||
|         self.manager = manager.AuthManager() |         self.manager = manager.AuthManager() | ||||||
|  |  | ||||||
|     def __print_export(self, user): |  | ||||||
|         print 'export EC2_ACCESS_KEY=%s' % user.access |  | ||||||
|         print 'export EC2_SECRET_KEY=%s' % user.secret |  | ||||||
|  |  | ||||||
|     def admin(self, name, access=None, secret=None): |     def admin(self, name, access=None, secret=None): | ||||||
|         """creates a new admin and prints exports |         """creates a new admin and prints exports | ||||||
|         arguments: name [access] [secret]""" |         arguments: name [access] [secret]""" | ||||||
|         user = self.manager.create_user(name, access, secret, True) |         user = self.manager.create_user(name, access, secret, True) | ||||||
|         self.__print_export(user) |         print_export(user) | ||||||
|  |  | ||||||
|     def create(self, name, access=None, secret=None): |     def create(self, name, access=None, secret=None): | ||||||
|         """creates a new user and prints exports |         """creates a new user and prints exports | ||||||
|         arguments: name [access] [secret]""" |         arguments: name [access] [secret]""" | ||||||
|         user = self.manager.create_user(name, access, secret, False) |         user = self.manager.create_user(name, access, secret, False) | ||||||
|         self.__print_export(user) |         print_export(user) | ||||||
|  |  | ||||||
|     def delete(self, name): |     def delete(self, name): | ||||||
|         """deletes an existing user |         """deletes an existing user | ||||||
| @@ -139,7 +148,7 @@ class UserCommands(object): | |||||||
|         arguments: name""" |         arguments: name""" | ||||||
|         user = self.manager.get_user(name) |         user = self.manager.get_user(name) | ||||||
|         if user: |         if user: | ||||||
|             self.__print_export(user) |             print_export(user) | ||||||
|         else: |         else: | ||||||
|             print "User %s doesn't exist" % name |             print "User %s doesn't exist" % name | ||||||
|  |  | ||||||
| @@ -149,53 +158,58 @@ class UserCommands(object): | |||||||
|         for user in self.manager.get_users(): |         for user in self.manager.get_users(): | ||||||
|             print user.name |             print user.name | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def print_export(user): | ||||||
|  |     """Print export variables to use with API.""" | ||||||
|  |     print 'export EC2_ACCESS_KEY=%s' % user.access | ||||||
|  |     print 'export EC2_SECRET_KEY=%s' % user.secret | ||||||
|  |  | ||||||
|  |  | ||||||
| class ProjectCommands(object): | class ProjectCommands(object): | ||||||
|  |     """Class for managing projects.""" | ||||||
|  |  | ||||||
|     def __init__(self): |     def __init__(self): | ||||||
|         self.manager = manager.AuthManager() |         self.manager = manager.AuthManager() | ||||||
|  |  | ||||||
|     def add(self, project, user): |     def add(self, project, user): | ||||||
|         """adds user to project |         """Adds user to project | ||||||
|         arguments: project user""" |         arguments: project user""" | ||||||
|         self.manager.add_to_project(user, project) |         self.manager.add_to_project(user, project) | ||||||
|  |  | ||||||
|     def create(self, name, project_manager, description=None): |     def create(self, name, project_manager, description=None): | ||||||
|         """creates a new project |         """Creates a new project | ||||||
|         arguments: name project_manager [description]""" |         arguments: name project_manager [description]""" | ||||||
|         user = self.manager.create_project(name, project_manager, description) |         self.manager.create_project(name, project_manager, description) | ||||||
|  |  | ||||||
|     def delete(self, name): |     def delete(self, name): | ||||||
|         """deletes an existing project |         """Deletes an existing project | ||||||
|         arguments: name""" |         arguments: name""" | ||||||
|         self.manager.delete_project(name) |         self.manager.delete_project(name) | ||||||
|  |  | ||||||
|     def environment(self, project_id, user_id, filename='novarc'): |     def environment(self, project_id, user_id, filename='novarc'): | ||||||
|         """exports environment variables to an sourcable file |         """Exports environment variables to an sourcable file | ||||||
|         arguments: project_id user_id [filename='novarc]""" |         arguments: project_id user_id [filename='novarc]""" | ||||||
|         rc = self.manager.get_environment_rc(project_id, user_id) |         rc = self.manager.get_environment_rc(project_id, user_id) | ||||||
|         with open(filename, 'w') as f: |         with open(filename, 'w') as f: | ||||||
|             f.write(rc) |             f.write(rc) | ||||||
|  |  | ||||||
|     def list(self): |     def list(self): | ||||||
|         """lists all projects |         """Lists all projects | ||||||
|         arguments: <none>""" |         arguments: <none>""" | ||||||
|         for project in self.manager.get_projects(): |         for project in self.manager.get_projects(): | ||||||
|             print project.name |             print project.name | ||||||
|  |  | ||||||
|     def remove(self, project, user): |     def remove(self, project, user): | ||||||
|         """removes user from project |         """Removes user from project | ||||||
|         arguments: project user""" |         arguments: project user""" | ||||||
|         self.manager.remove_from_project(user, project) |         self.manager.remove_from_project(user, project) | ||||||
|  |  | ||||||
|     def zip(self, project_id, user_id, filename='nova.zip'): |     def zipfile(self, project_id, user_id, filename='nova.zip'): | ||||||
|         """exports credentials for project to a zip file |         """Exports credentials for project to a zip file | ||||||
|         arguments: project_id user_id [filename='nova.zip]""" |         arguments: project_id user_id [filename='nova.zip]""" | ||||||
|         zip = self.manager.get_credentials(project_id, user_id) |         zip_file = self.manager.get_credentials(user_id, project_id) | ||||||
|         with open(filename, 'w') as f: |         with open(filename, 'w') as f: | ||||||
|             f.write(zip) |             f.write(zip_file) | ||||||
|  |  | ||||||
|  |  | ||||||
| def usage(script_name): |  | ||||||
|     print script_name + " category action [<args>]" |  | ||||||
|  |  | ||||||
|  |  | ||||||
| categories = [ | categories = [ | ||||||
| @@ -207,62 +221,61 @@ categories = [ | |||||||
|  |  | ||||||
|  |  | ||||||
| def lazy_match(name, key_value_tuples): | def lazy_match(name, key_value_tuples): | ||||||
|     """finds all objects that have a key that case insensitively contains [name] |     """Finds all objects that have a key that case insensitively contains | ||||||
|     key_value_tuples is a list of tuples of the form (key, value) |     [name] key_value_tuples is a list of tuples of the form (key, value) | ||||||
|     returns a list of tuples of the form (key, value)""" |     returns a list of tuples of the form (key, value)""" | ||||||
|     return [(k, v) for (k, v) in key_value_tuples if k.lower().find(name.lower()) == 0] |     result = [] | ||||||
|  |     for (k, v) in key_value_tuples: | ||||||
|  |         if k.lower().find(name.lower()) == 0: | ||||||
|  |             result.append((k, v)) | ||||||
|  |     if len(result) == 0: | ||||||
|  |         print "%s does not match any options:" % name | ||||||
|  |         for k, _v in key_value_tuples: | ||||||
|  |             print "\t%s" % k | ||||||
|  |         sys.exit(2) | ||||||
|  |     if len(result) > 1: | ||||||
|  |         print "%s matched multiple options:" % name | ||||||
|  |         for k, _v in result: | ||||||
|  |             print "\t%s" % k | ||||||
|  |         sys.exit(2) | ||||||
|  |     return result | ||||||
|  |  | ||||||
|  |  | ||||||
| def methods_of(obj): | def methods_of(obj): | ||||||
|     """get all callable methods of an object that don't start with underscore |     """Get all callable methods of an object that don't start with underscore | ||||||
|     returns a list of tuples of the form (method_name, method)""" |     returns a list of tuples of the form (method_name, method)""" | ||||||
|     return [(i, getattr(obj, i)) for i in dir(obj) if callable(getattr(obj, i)) and not i.startswith('_')] |     result = [] | ||||||
|  |     for i in dir(obj): | ||||||
|  |         if callable(getattr(obj, i)) and not i.startswith('_'): | ||||||
|  |             result.append((i, getattr(obj, i))) | ||||||
|  |     return result | ||||||
|  |  | ||||||
|  |  | ||||||
| if __name__ == '__main__': | def main(): | ||||||
|  |     """Parse options and call the appropriate class/method.""" | ||||||
|     utils.default_flagfile('/etc/nova/nova-manage.conf') |     utils.default_flagfile('/etc/nova/nova-manage.conf') | ||||||
|     argv = FLAGS(sys.argv) |     argv = FLAGS(sys.argv) | ||||||
|     script_name = argv.pop(0) |     script_name = argv.pop(0) | ||||||
|     if len(argv) < 1: |     if len(argv) < 1: | ||||||
|         usage(script_name) |         print script_name + " category action [<args>]" | ||||||
|         print "Available categories:" |         print "Available categories:" | ||||||
|         for k, v in categories: |         for k, _ in categories: | ||||||
|             print "\t%s" % k |             print "\t%s" % k | ||||||
|         sys.exit(2) |         sys.exit(2) | ||||||
|     category = argv.pop(0) |     category = argv.pop(0) | ||||||
|     matches = lazy_match(category, categories) |     matches = lazy_match(category, categories) | ||||||
|     if len(matches) == 0: |  | ||||||
|         print "%s does not match any categories:" % category |  | ||||||
|         for k, v in categories: |  | ||||||
|             print "\t%s" % k |  | ||||||
|         sys.exit(2) |  | ||||||
|     if len(matches) > 1: |  | ||||||
|         print "%s matched multiple categories:" % category |  | ||||||
|         for k, v in matches: |  | ||||||
|             print "\t%s" % k |  | ||||||
|         sys.exit(2) |  | ||||||
|     # instantiate the command group object |     # instantiate the command group object | ||||||
|     category, fn = matches[0] |     category, fn = matches[0] | ||||||
|     command_object = fn() |     command_object = fn() | ||||||
|     actions = methods_of(command_object) |     actions = methods_of(command_object) | ||||||
|     if len(argv) < 1: |     if len(argv) < 1: | ||||||
|         usage(script_name) |         print script_name + " category action [<args>]" | ||||||
|         print "Available actions for %s category:" % category |         print "Available actions for %s category:" % category | ||||||
|         for k, v in actions: |         for k, _v in actions: | ||||||
|             print "\t%s" % k |             print "\t%s" % k | ||||||
|         sys.exit(2) |         sys.exit(2) | ||||||
|     action = argv.pop(0) |     action = argv.pop(0) | ||||||
|     matches = lazy_match(action, actions) |     matches = lazy_match(action, actions) | ||||||
|     if len(matches) == 0: |  | ||||||
|         print "%s does not match any actions" % action |  | ||||||
|         for k, v in actions: |  | ||||||
|             print "\t%s" % k |  | ||||||
|         sys.exit(2) |  | ||||||
|     if len(matches) > 1: |  | ||||||
|         print "%s matched multiple actions:" % action |  | ||||||
|         for k, v in matches: |  | ||||||
|             print "\t%s" % k |  | ||||||
|         sys.exit(2) |  | ||||||
|     action, fn = matches[0] |     action, fn = matches[0] | ||||||
|     # call the action with the remaining arguments |     # call the action with the remaining arguments | ||||||
|     try: |     try: | ||||||
| @@ -273,3 +286,5 @@ if __name__ == '__main__': | |||||||
|         print "%s %s: %s" % (category, action, fn.__doc__) |         print "%s %s: %s" % (category, action, fn.__doc__) | ||||||
|         sys.exit(2) |         sys.exit(2) | ||||||
|  |  | ||||||
|  | if __name__ == '__main__': | ||||||
|  |     main() | ||||||
|   | |||||||
| @@ -30,15 +30,9 @@ from nova.objectstore import handler | |||||||
| FLAGS = flags.FLAGS | FLAGS = flags.FLAGS | ||||||
|  |  | ||||||
|  |  | ||||||
| def main(): |  | ||||||
|     app = handler.get_application() |  | ||||||
|     print app |  | ||||||
|     return app |  | ||||||
|  |  | ||||||
| # NOTE(soren): Stolen from nova-compute |  | ||||||
| if __name__ == '__main__': | if __name__ == '__main__': | ||||||
|     twistd.serve(__file__) |     twistd.serve(__file__) | ||||||
|  |  | ||||||
| if __name__ == '__builtin__': | if __name__ == '__builtin__': | ||||||
|     utils.default_flagfile() |     utils.default_flagfile() | ||||||
|     application = main() |     application = handler.get_application() | ||||||
|   | |||||||
| @@ -1,58 +0,0 @@ | |||||||
| #!/usr/bin/env python |  | ||||||
| # vim: tabstop=4 shiftwidth=4 softtabstop=4 |  | ||||||
|  |  | ||||||
| # Copyright 2010 United States Government as represented by the |  | ||||||
| # Administrator of the National Aeronautics and Space Administration. |  | ||||||
| # 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. |  | ||||||
| """ |  | ||||||
|   WSGI daemon for the main API endpoint. |  | ||||||
| """ |  | ||||||
|  |  | ||||||
| import logging |  | ||||||
| from tornado import ioloop |  | ||||||
| from wsgiref import simple_server |  | ||||||
|  |  | ||||||
| from nova import flags |  | ||||||
| from nova import rpc |  | ||||||
| from nova import server |  | ||||||
| from nova import utils |  | ||||||
| from nova.auth import manager |  | ||||||
| from nova.endpoint import rackspace |  | ||||||
|  |  | ||||||
| FLAGS = flags.FLAGS |  | ||||||
| flags.DEFINE_integer('cc_port', 8773, 'cloud controller port') |  | ||||||
|  |  | ||||||
| def main(_argv): |  | ||||||
|     user_manager = manager.AuthManager() |  | ||||||
|     api_instance = rackspace.Api(user_manager) |  | ||||||
|     conn = rpc.Connection.instance() |  | ||||||
|     rpc_consumer = rpc.AdapterConsumer(connection=conn, |  | ||||||
|                                        topic=FLAGS.cloud_topic, |  | ||||||
|                                        proxy=api_instance) |  | ||||||
|  |  | ||||||
| #   TODO: fire rpc response listener (without attach to tornado) |  | ||||||
| #    io_inst = ioloop.IOLoop.instance() |  | ||||||
| #    _injected = consumer.attach_to_tornado(io_inst) |  | ||||||
|  |  | ||||||
|     http_server = simple_server.WSGIServer(('0.0.0.0', FLAGS.cc_port), simple_server.WSGIRequestHandler) |  | ||||||
|     http_server.set_app(api_instance.handler) |  | ||||||
|     logging.debug('Started HTTP server on port %i' % FLAGS.cc_port) |  | ||||||
|     while True: |  | ||||||
|         http_server.handle_request() |  | ||||||
| #    io_inst.start() |  | ||||||
|  |  | ||||||
| if __name__ == '__main__': |  | ||||||
|     utils.default_flagfile() |  | ||||||
|     server.serve('nova-rsapi', main) |  | ||||||
| @@ -20,6 +20,7 @@ Nova User API client library. | |||||||
| """ | """ | ||||||
|  |  | ||||||
| import base64 | import base64 | ||||||
|  |  | ||||||
| import boto | import boto | ||||||
| from boto.ec2.regioninfo import RegionInfo | from boto.ec2.regioninfo import RegionInfo | ||||||
|  |  | ||||||
| @@ -57,6 +58,30 @@ class UserInfo(object): | |||||||
|         elif name == 'secretkey': |         elif name == 'secretkey': | ||||||
|             self.secretkey = str(value) |             self.secretkey = str(value) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class UserRole(object): | ||||||
|  |     """ | ||||||
|  |     Information about a Nova user's role, as parsed through SAX. | ||||||
|  |     Fields include: | ||||||
|  |         role | ||||||
|  |     """ | ||||||
|  |     def __init__(self, connection=None): | ||||||
|  |         self.connection = connection | ||||||
|  |         self.role = None | ||||||
|  |      | ||||||
|  |     def __repr__(self): | ||||||
|  |         return 'UserRole:%s' % self.role | ||||||
|  |  | ||||||
|  |     def startElement(self, name, attrs, connection): | ||||||
|  |         return None | ||||||
|  |          | ||||||
|  |     def endElement(self, name, value, connection): | ||||||
|  |         if name == 'role': | ||||||
|  |             self.role = value | ||||||
|  |         else: | ||||||
|  |             setattr(self, name, str(value)) | ||||||
|  |  | ||||||
|  |  | ||||||
| class ProjectInfo(object): | class ProjectInfo(object): | ||||||
|     """ |     """ | ||||||
|     Information about a Nova project, as parsed through SAX |     Information about a Nova project, as parsed through SAX | ||||||
| @@ -92,12 +117,14 @@ class ProjectInfo(object): | |||||||
|         else: |         else: | ||||||
|             setattr(self, name, str(value)) |             setattr(self, name, str(value)) | ||||||
|  |  | ||||||
|  |  | ||||||
| class ProjectMember(object): | class ProjectMember(object): | ||||||
|     """ |     """ | ||||||
|     Information about a Nova project member, as parsed through SAX. |     Information about a Nova project member, as parsed through SAX. | ||||||
|     Fields include: |     Fields include: | ||||||
|         memberId |         memberId | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
|     def __init__(self, connection=None): |     def __init__(self, connection=None): | ||||||
|         self.connection = connection |         self.connection = connection | ||||||
|         self.memberId = None |         self.memberId = None | ||||||
| @@ -142,6 +169,7 @@ class HostInfo(object): | |||||||
|     def endElement(self, name, value, connection): |     def endElement(self, name, value, connection): | ||||||
|         setattr(self, name, value) |         setattr(self, name, value) | ||||||
|  |  | ||||||
|  |  | ||||||
| class NovaAdminClient(object): | class NovaAdminClient(object): | ||||||
|     def __init__(self, clc_ip='127.0.0.1', region='nova', access_key='admin', |     def __init__(self, clc_ip='127.0.0.1', region='nova', access_key='admin', | ||||||
|                  secret_key='admin', **kwargs): |                  secret_key='admin', **kwargs): | ||||||
| @@ -196,6 +224,24 @@ class NovaAdminClient(object): | |||||||
|         """ deletes a user """ |         """ deletes a user """ | ||||||
|         return self.apiconn.get_object('DeregisterUser', {'Name': username}, UserInfo) |         return self.apiconn.get_object('DeregisterUser', {'Name': username}, UserInfo) | ||||||
|  |  | ||||||
|  |     def get_roles(self, project_roles=True): | ||||||
|  |         """Returns a list of available roles.""" | ||||||
|  |         return self.apiconn.get_list('DescribeRoles', | ||||||
|  |                                      {'ProjectRoles': project_roles}, | ||||||
|  |                                      [('item', UserRole)]) | ||||||
|  |  | ||||||
|  |     def get_user_roles(self, user, project=None): | ||||||
|  |         """Returns a list of roles for the given user. | ||||||
|  |            Omitting project will return any global roles that the user has. | ||||||
|  |            Specifying project will return only project specific roles. | ||||||
|  |         """ | ||||||
|  |         params = {'User':user} | ||||||
|  |         if project: | ||||||
|  |             params['Project'] = project | ||||||
|  |         return self.apiconn.get_list('DescribeUserRoles', | ||||||
|  |                                      params, | ||||||
|  |                                      [('item', UserRole)]) | ||||||
|  |  | ||||||
|     def add_user_role(self, user, role, project=None): |     def add_user_role(self, user, role, project=None): | ||||||
|         """ |         """ | ||||||
|         Add a role to a user either globally or for a specific project. |         Add a role to a user either globally or for a specific project. | ||||||
|   | |||||||
| @@ -219,7 +219,6 @@ class FakeLDAP(object): | |||||||
|             raise NO_SUCH_OBJECT() |             raise NO_SUCH_OBJECT() | ||||||
|         return objects |         return objects | ||||||
|  |  | ||||||
|  |  | ||||||
|     @property |     @property | ||||||
|     def __redis_prefix(self): |     def __redis_prefix(self): | ||||||
|         return 'ldap:' |         return 'ldap:' | ||||||
|   | |||||||
| @@ -30,6 +30,7 @@ import sys | |||||||
| from nova import exception | from nova import exception | ||||||
| from nova import flags | from nova import flags | ||||||
|  |  | ||||||
|  |  | ||||||
| FLAGS = flags.FLAGS | FLAGS = flags.FLAGS | ||||||
| flags.DEFINE_string('ldap_url', 'ldap://localhost', | flags.DEFINE_string('ldap_url', 'ldap://localhost', | ||||||
|                     'Point this at your ldap server') |                     'Point this at your ldap server') | ||||||
| @@ -182,7 +183,8 @@ class LdapDriver(object): | |||||||
|             for member_uid in member_uids: |             for member_uid in member_uids: | ||||||
|                 if not self.__user_exists(member_uid): |                 if not self.__user_exists(member_uid): | ||||||
|                     raise exception.NotFound("Project can't be created " |                     raise exception.NotFound("Project can't be created " | ||||||
|                             "because user %s doesn't exist" % member_uid) |                                              "because user %s doesn't exist" | ||||||
|  |                                              % member_uid) | ||||||
|                 members.append(self.__uid_to_dn(member_uid)) |                 members.append(self.__uid_to_dn(member_uid)) | ||||||
|         # always add the manager as a member because members is required |         # always add the manager as a member because members is required | ||||||
|         if not manager_dn in members: |         if not manager_dn in members: | ||||||
| @@ -236,6 +238,26 @@ class LdapDriver(object): | |||||||
|         role_dn = self.__role_to_dn(role, project_id) |         role_dn = self.__role_to_dn(role, project_id) | ||||||
|         return self.__remove_from_group(uid, role_dn) |         return self.__remove_from_group(uid, role_dn) | ||||||
|  |  | ||||||
|  |     def get_user_roles(self, uid, project_id=None): | ||||||
|  |         """Retrieve list of roles for user (or user and project)""" | ||||||
|  |         if project_id is None: | ||||||
|  |             # NOTE(vish): This is unneccesarily slow, but since we can't | ||||||
|  |             #             guarantee that the global roles are located | ||||||
|  |             #             together in the ldap tree, we're doing this version. | ||||||
|  |             roles = [] | ||||||
|  |             for role in FLAGS.allowed_roles: | ||||||
|  |                 role_dn = self.__role_to_dn(role) | ||||||
|  |                 if self.__is_in_group(uid, role_dn): | ||||||
|  |                     roles.append(role) | ||||||
|  |             return roles | ||||||
|  |         else: | ||||||
|  |             project_dn = 'cn=%s,%s' % (project_id, FLAGS.ldap_project_subtree) | ||||||
|  |             roles = self.__find_objects(project_dn, | ||||||
|  |                                         '(&(&(objectclass=groupOfNames)' | ||||||
|  |                                         '(!(objectclass=novaProject)))' | ||||||
|  |                                         '(member=%s))' % self.__uid_to_dn(uid)) | ||||||
|  |             return [role['cn'][0] for role in roles] | ||||||
|  |  | ||||||
|     def delete_user(self, uid): |     def delete_user(self, uid): | ||||||
|         """Delete a user""" |         """Delete a user""" | ||||||
|         if not self.__user_exists(uid): |         if not self.__user_exists(uid): | ||||||
| @@ -253,24 +275,24 @@ class LdapDriver(object): | |||||||
|         self.conn.delete_s('cn=%s,uid=%s,%s' % (key_name, uid, |         self.conn.delete_s('cn=%s,uid=%s,%s' % (key_name, uid, | ||||||
|                                           FLAGS.ldap_user_subtree)) |                                           FLAGS.ldap_user_subtree)) | ||||||
|  |  | ||||||
|     def delete_project(self, name): |     def delete_project(self, project_id): | ||||||
|         """Delete a project""" |         """Delete a project""" | ||||||
|         project_dn = 'cn=%s,%s' % (name, FLAGS.ldap_project_subtree) |         project_dn = 'cn=%s,%s' % (project_id, FLAGS.ldap_project_subtree) | ||||||
|         self.__delete_roles(project_dn) |         self.__delete_roles(project_dn) | ||||||
|         self.__delete_group(project_dn) |         self.__delete_group(project_dn) | ||||||
|  |  | ||||||
|     def __user_exists(self, name): |     def __user_exists(self, uid): | ||||||
|         """Check if user exists""" |         """Check if user exists""" | ||||||
|         return self.get_user(name) != None |         return self.get_user(uid) != None | ||||||
|  |  | ||||||
|     def __key_pair_exists(self, uid, key_name): |     def __key_pair_exists(self, uid, key_name): | ||||||
|         """Check if key pair exists""" |         """Check if key pair exists""" | ||||||
|         return self.get_user(uid) != None |         return self.get_user(uid) != None | ||||||
|         return self.get_key_pair(uid, key_name) != None |         return self.get_key_pair(uid, key_name) != None | ||||||
|  |  | ||||||
|     def __project_exists(self, name): |     def __project_exists(self, project_id): | ||||||
|         """Check if project exists""" |         """Check if project exists""" | ||||||
|         return self.get_project(name) != None |         return self.get_project(project_id) != None | ||||||
|  |  | ||||||
|     def __find_object(self, dn, query=None, scope=None): |     def __find_object(self, dn, query=None, scope=None): | ||||||
|         """Find an object by dn and query""" |         """Find an object by dn and query""" | ||||||
|   | |||||||
| @@ -29,16 +29,17 @@ import uuid | |||||||
| import zipfile | import zipfile | ||||||
|  |  | ||||||
| from nova import crypto | from nova import crypto | ||||||
| from nova import datastore |  | ||||||
| from nova import exception | from nova import exception | ||||||
| from nova import flags | from nova import flags | ||||||
| from nova import objectstore # for flags |  | ||||||
| from nova import utils | from nova import utils | ||||||
| from nova.auth import ldapdriver # for flags |  | ||||||
| from nova.auth import signer | from nova.auth import signer | ||||||
| from nova.network import vpn | from nova.network import vpn | ||||||
|  |  | ||||||
|  |  | ||||||
| FLAGS = flags.FLAGS | FLAGS = flags.FLAGS | ||||||
|  | flags.DEFINE_list('allowed_roles', | ||||||
|  |                   ['cloudadmin', 'itsec', 'sysadmin', 'netadmin', 'developer'], | ||||||
|  |                   'Allowed roles for project') | ||||||
|  |  | ||||||
| # NOTE(vish): a user with one of these roles will be a superuser and | # NOTE(vish): a user with one of these roles will be a superuser and | ||||||
| #             have access to all api commands | #             have access to all api commands | ||||||
| @@ -50,7 +51,6 @@ flags.DEFINE_list('superuser_roles', ['cloudadmin'], | |||||||
| flags.DEFINE_list('global_roles', ['cloudadmin', 'itsec'], | flags.DEFINE_list('global_roles', ['cloudadmin', 'itsec'], | ||||||
|                   'Roles that apply to all projects') |                   'Roles that apply to all projects') | ||||||
|  |  | ||||||
|  |  | ||||||
| flags.DEFINE_string('credentials_template', | flags.DEFINE_string('credentials_template', | ||||||
|                     utils.abspath('auth/novarc.template'), |                     utils.abspath('auth/novarc.template'), | ||||||
|                     'Template for creating users rc file') |                     'Template for creating users rc file') | ||||||
| @@ -65,15 +65,14 @@ flags.DEFINE_string('credential_cert_file', 'cert.pem', | |||||||
|                     'Filename of certificate in credentials zip') |                     'Filename of certificate in credentials zip') | ||||||
| flags.DEFINE_string('credential_rc_file', 'novarc', | flags.DEFINE_string('credential_rc_file', 'novarc', | ||||||
|                     'Filename of rc in credentials zip') |                     'Filename of rc in credentials zip') | ||||||
|  |  | ||||||
| flags.DEFINE_string('credential_cert_subject', | flags.DEFINE_string('credential_cert_subject', | ||||||
|                     '/C=US/ST=California/L=MountainView/O=AnsoLabs/' |                     '/C=US/ST=California/L=MountainView/O=AnsoLabs/' | ||||||
|                     'OU=NovaDev/CN=%s-%s', |                     'OU=NovaDev/CN=%s-%s', | ||||||
|                     'Subject for certificate for users') |                     'Subject for certificate for users') | ||||||
|  |  | ||||||
| flags.DEFINE_string('auth_driver', 'nova.auth.ldapdriver.FakeLdapDriver', | flags.DEFINE_string('auth_driver', 'nova.auth.ldapdriver.FakeLdapDriver', | ||||||
|                     'Driver that auth manager uses') |                     'Driver that auth manager uses') | ||||||
|  |  | ||||||
|  |  | ||||||
| class AuthBase(object): | class AuthBase(object): | ||||||
|     """Base class for objects relating to auth |     """Base class for objects relating to auth | ||||||
|  |  | ||||||
| @@ -81,6 +80,7 @@ class AuthBase(object): | |||||||
|     an id member. They may optionally contain methods that delegate to |     an id member. They may optionally contain methods that delegate to | ||||||
|     AuthManager, but should not implement logic themselves. |     AuthManager, but should not implement logic themselves. | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def safe_id(cls, obj): |     def safe_id(cls, obj): | ||||||
|         """Safe get object id |         """Safe get object id | ||||||
| @@ -98,7 +98,9 @@ class AuthBase(object): | |||||||
|  |  | ||||||
| class User(AuthBase): | class User(AuthBase): | ||||||
|     """Object representing a user""" |     """Object representing a user""" | ||||||
|  |  | ||||||
|     def __init__(self, id, name, access, secret, admin): |     def __init__(self, id, name, access, secret, admin): | ||||||
|  |         AuthBase.__init__(self) | ||||||
|         self.id = id |         self.id = id | ||||||
|         self.name = name |         self.name = name | ||||||
|         self.access = access |         self.access = access | ||||||
| @@ -158,7 +160,9 @@ class KeyPair(AuthBase): | |||||||
|     Even though this object is named KeyPair, only the public key and |     Even though this object is named KeyPair, only the public key and | ||||||
|     fingerprint is stored. The user's private key is not saved. |     fingerprint is stored. The user's private key is not saved. | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
|     def __init__(self, id, name, owner_id, public_key, fingerprint): |     def __init__(self, id, name, owner_id, public_key, fingerprint): | ||||||
|  |         AuthBase.__init__(self) | ||||||
|         self.id = id |         self.id = id | ||||||
|         self.name = name |         self.name = name | ||||||
|         self.owner_id = owner_id |         self.owner_id = owner_id | ||||||
| @@ -175,7 +179,9 @@ class KeyPair(AuthBase): | |||||||
|  |  | ||||||
| class Project(AuthBase): | class Project(AuthBase): | ||||||
|     """Represents a Project returned from the datastore""" |     """Represents a Project returned from the datastore""" | ||||||
|  |  | ||||||
|     def __init__(self, id, name, project_manager_id, description, member_ids): |     def __init__(self, id, name, project_manager_id, description, member_ids): | ||||||
|  |         AuthBase.__init__(self) | ||||||
|         self.id = id |         self.id = id | ||||||
|         self.name = name |         self.name = name | ||||||
|         self.project_manager_id = project_manager_id |         self.project_manager_id = project_manager_id | ||||||
| @@ -222,7 +228,6 @@ class Project(AuthBase): | |||||||
|                                                         self.member_ids) |                                                         self.member_ids) | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class AuthManager(object): | class AuthManager(object): | ||||||
|     """Manager Singleton for dealing with Users, Projects, and Keypairs |     """Manager Singleton for dealing with Users, Projects, and Keypairs | ||||||
|  |  | ||||||
| @@ -234,7 +239,9 @@ class AuthManager(object): | |||||||
|     AuthManager also manages associated data related to Auth objects that |     AuthManager also manages associated data related to Auth objects that | ||||||
|     need to be more accessible, such as vpn ips and ports. |     need to be more accessible, such as vpn ips and ports. | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
|     _instance = None |     _instance = None | ||||||
|  |  | ||||||
|     def __new__(cls, *args, **kwargs): |     def __new__(cls, *args, **kwargs): | ||||||
|         """Returns the AuthManager singleton""" |         """Returns the AuthManager singleton""" | ||||||
|         if not cls._instance: |         if not cls._instance: | ||||||
| @@ -431,6 +438,10 @@ class AuthManager(object): | |||||||
|         @type project: Project or project_id |         @type project: Project or project_id | ||||||
|         @param project: Project in which to add local role. |         @param project: Project in which to add local role. | ||||||
|         """ |         """ | ||||||
|  |         if role not in FLAGS.allowed_roles: | ||||||
|  |             raise exception.NotFound("The %s role can not be found" % role) | ||||||
|  |         if project is not None and role in FLAGS.global_roles: | ||||||
|  |             raise exception.NotFound("The %s role is global only" % role) | ||||||
|         with self.driver() as drv: |         with self.driver() as drv: | ||||||
|             drv.add_role(User.safe_id(user), role, Project.safe_id(project)) |             drv.add_role(User.safe_id(user), role, Project.safe_id(project)) | ||||||
|  |  | ||||||
| @@ -454,6 +465,19 @@ class AuthManager(object): | |||||||
|         with self.driver() as drv: |         with self.driver() as drv: | ||||||
|             drv.remove_role(User.safe_id(user), role, Project.safe_id(project)) |             drv.remove_role(User.safe_id(user), role, Project.safe_id(project)) | ||||||
|  |  | ||||||
|  |     def get_roles(self, project_roles=True): | ||||||
|  |         """Get list of allowed roles""" | ||||||
|  |         if project_roles: | ||||||
|  |             return list(set(FLAGS.allowed_roles) - set(FLAGS.global_roles)) | ||||||
|  |         else: | ||||||
|  |             return FLAGS.allowed_roles | ||||||
|  |  | ||||||
|  |     def get_user_roles(self, user, project=None): | ||||||
|  |         """Get user global or per-project roles""" | ||||||
|  |         with self.driver() as drv: | ||||||
|  |             return drv.get_user_roles(User.safe_id(user), | ||||||
|  |                                       Project.safe_id(project)) | ||||||
|  |  | ||||||
|     def get_project(self, pid): |     def get_project(self, pid): | ||||||
|         """Get project object by id""" |         """Get project object by id""" | ||||||
|         with self.driver() as drv: |         with self.driver() as drv: | ||||||
|   | |||||||
| @@ -32,6 +32,7 @@ def allow(*roles): | |||||||
|         return wrapped_f |         return wrapped_f | ||||||
|     return wrap |     return wrap | ||||||
|  |  | ||||||
|  |  | ||||||
| def deny(*roles): | def deny(*roles): | ||||||
|     def wrap(f): |     def wrap(f): | ||||||
|         def wrapped_f(self, context, *args, **kwargs): |         def wrapped_f(self, context, *args, **kwargs): | ||||||
| @@ -44,6 +45,7 @@ def deny(*roles): | |||||||
|         return wrapped_f |         return wrapped_f | ||||||
|     return wrap |     return wrap | ||||||
|  |  | ||||||
|  |  | ||||||
| def __matches_role(context, role): | def __matches_role(context, role): | ||||||
|     if role == 'all': |     if role == 'all': | ||||||
|         return True |         return True | ||||||
|   | |||||||
| @@ -48,11 +48,15 @@ import hashlib | |||||||
| import hmac | import hmac | ||||||
| import logging | import logging | ||||||
| import urllib | import urllib | ||||||
| import boto       # NOTE(vish): for new boto |  | ||||||
| import boto.utils # NOTE(vish): for old boto | # NOTE(vish): for new boto | ||||||
|  | import boto        | ||||||
|  | # NOTE(vish): for old boto | ||||||
|  | import boto.utils  | ||||||
|  |  | ||||||
| from nova.exception import Error | from nova.exception import Error | ||||||
|  |  | ||||||
|  |  | ||||||
| class Signer(object): | class Signer(object): | ||||||
|     """ hacked up code from boto/connection.py """ |     """ hacked up code from boto/connection.py """ | ||||||
|  |  | ||||||
| @@ -77,7 +81,6 @@ class Signer(object): | |||||||
|             return self._calc_signature_2(params, verb, server_string, path) |             return self._calc_signature_2(params, verb, server_string, path) | ||||||
|         raise Error('Unknown Signature Version: %s' % self.SignatureVersion) |         raise Error('Unknown Signature Version: %s' % self.SignatureVersion) | ||||||
|  |  | ||||||
|  |  | ||||||
|     def _get_utf8_value(self, value): |     def _get_utf8_value(self, value): | ||||||
|         if not isinstance(value, str) and not isinstance(value, unicode): |         if not isinstance(value, str) and not isinstance(value, unicode): | ||||||
|             value = str(value) |             value = str(value) | ||||||
| @@ -133,5 +136,6 @@ class Signer(object): | |||||||
|         logging.debug('base64 encoded digest: %s' % b64) |         logging.debug('base64 encoded digest: %s' % b64) | ||||||
|         return b64 |         return b64 | ||||||
|  |  | ||||||
|  |  | ||||||
| if __name__ == '__main__': | if __name__ == '__main__': | ||||||
|     print Signer('foo').generate({"SignatureMethod": 'HmacSHA256', 'SignatureVersion': '2'}, "get", "server", "/foo") |     print Signer('foo').generate({"SignatureMethod": 'HmacSHA256', 'SignatureVersion': '2'}, "get", "server", "/foo") | ||||||
|   | |||||||
| @@ -124,12 +124,16 @@ class BasicModel(object): | |||||||
|             yield cls(identifier) |             yield cls(identifier) | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     @absorb_connection_error |  | ||||||
|     def associated_to(cls, foreign_type, foreign_id): |     def associated_to(cls, foreign_type, foreign_id): | ||||||
|         redis_set = cls._redis_association_name(foreign_type, foreign_id) |         for identifier in cls.associated_keys(foreign_type, foreign_id): | ||||||
|         for identifier in Redis.instance().smembers(redis_set): |  | ||||||
|             yield cls(identifier) |             yield cls(identifier) | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     @absorb_connection_error | ||||||
|  |     def associated_keys(cls, foreign_type, foreign_id): | ||||||
|  |         redis_set = cls._redis_association_name(foreign_type, foreign_id) | ||||||
|  |         return Redis.instance().smembers(redis_set) or [] | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def _redis_set_name(cls, kls_name): |     def _redis_set_name(cls, kls_name): | ||||||
|         # stupidly pluralize (for compatiblity with previous codebase) |         # stupidly pluralize (for compatiblity with previous codebase) | ||||||
| @@ -138,7 +142,7 @@ class BasicModel(object): | |||||||
|     @classmethod |     @classmethod | ||||||
|     def _redis_association_name(cls, foreign_type, foreign_id): |     def _redis_association_name(cls, foreign_type, foreign_id): | ||||||
|         return cls._redis_set_name("%s:%s:%s" % |         return cls._redis_set_name("%s:%s:%s" % | ||||||
|                                    (foreign_type, foreign_id, cls.__name__)) |                                    (foreign_type, foreign_id, cls._redis_name())) | ||||||
|  |  | ||||||
|     @property |     @property | ||||||
|     def identifier(self): |     def identifier(self): | ||||||
| @@ -170,6 +174,9 @@ class BasicModel(object): | |||||||
|     def setdefault(self, item, default): |     def setdefault(self, item, default): | ||||||
|         return self.state.setdefault(item, default) |         return self.state.setdefault(item, default) | ||||||
|  |  | ||||||
|  |     def __contains__(self, item): | ||||||
|  |         return item in self.state | ||||||
|  |  | ||||||
|     def __getitem__(self, item): |     def __getitem__(self, item): | ||||||
|         return self.state[item] |         return self.state[item] | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,32 +0,0 @@ | |||||||
| # vim: tabstop=4 shiftwidth=4 softtabstop=4 |  | ||||||
|  |  | ||||||
| # Copyright 2010 United States Government as represented by the |  | ||||||
| # Administrator of the National Aeronautics and Space Administration. |  | ||||||
| # 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. |  | ||||||
|  |  | ||||||
| """ |  | ||||||
| :mod:`nova.endpoint` -- Main NOVA Api endpoints |  | ||||||
| ===================================================== |  | ||||||
|  |  | ||||||
| .. automodule:: nova.endpoint |  | ||||||
|    :platform: Unix |  | ||||||
|    :synopsis: REST APIs for all nova functions |  | ||||||
| .. moduleauthor:: Jesse Andrews <jesse@ansolabs.com> |  | ||||||
| .. moduleauthor:: Devin Carlen <devin.carlen@gmail.com> |  | ||||||
| .. moduleauthor:: Vishvananda Ishaya <vishvananda@yahoo.com> |  | ||||||
| .. moduleauthor:: Joshua McKenty <joshua@cognition.ca> |  | ||||||
| .. moduleauthor:: Manish Singh <yosh@gimp.org> |  | ||||||
| .. moduleauthor:: Andy Smith <andy@anarkystic.com> |  | ||||||
| """ |  | ||||||
|   | |||||||
| @@ -37,6 +37,7 @@ def user_dict(user, base64_file=None): | |||||||
|     else: |     else: | ||||||
|         return {} |         return {} | ||||||
|  |  | ||||||
|  |  | ||||||
| def project_dict(project): | def project_dict(project): | ||||||
|     """Convert the project object to a result dict""" |     """Convert the project object to a result dict""" | ||||||
|     if project: |     if project: | ||||||
| @@ -47,6 +48,7 @@ def project_dict(project): | |||||||
|     else: |     else: | ||||||
|         return {} |         return {} | ||||||
|  |  | ||||||
|  |  | ||||||
| def host_dict(host): | def host_dict(host): | ||||||
|     """Convert a host model object to a result dict""" |     """Convert a host model object to a result dict""" | ||||||
|     if host: |     if host: | ||||||
| @@ -54,6 +56,7 @@ def host_dict(host): | |||||||
|     else: |     else: | ||||||
|         return {} |         return {} | ||||||
|  |  | ||||||
|  |  | ||||||
| def admin_only(target): | def admin_only(target): | ||||||
|     """Decorator for admin-only API calls""" |     """Decorator for admin-only API calls""" | ||||||
|     def wrapper(*args, **kwargs): |     def wrapper(*args, **kwargs): | ||||||
| @@ -66,6 +69,7 @@ def admin_only(target): | |||||||
|  |  | ||||||
|     return wrapper |     return wrapper | ||||||
|  |  | ||||||
|  |  | ||||||
| class AdminController(object): | class AdminController(object): | ||||||
|     """ |     """ | ||||||
|     API Controller for users, hosts, nodes, and workers. |     API Controller for users, hosts, nodes, and workers. | ||||||
| @@ -102,6 +106,21 @@ class AdminController(object): | |||||||
|  |  | ||||||
|         return True |         return True | ||||||
|  |  | ||||||
|  |     @admin_only | ||||||
|  |     def describe_roles(self, context, project_roles=True, **kwargs): | ||||||
|  |         """Returns a list of allowed roles.""" | ||||||
|  |         roles = manager.AuthManager().get_roles(project_roles) | ||||||
|  |         return { 'roles': [{'role': r} for r in roles]} | ||||||
|  |  | ||||||
|  |     @admin_only | ||||||
|  |     def describe_user_roles(self, context, user, project=None, **kwargs): | ||||||
|  |         """Returns a list of roles for the given user. | ||||||
|  |            Omitting project will return any global roles that the user has. | ||||||
|  |            Specifying project will return only project specific roles. | ||||||
|  |         """ | ||||||
|  |         roles = manager.AuthManager().get_user_roles(user, project=project) | ||||||
|  |         return { 'roles': [{'role': r} for r in roles]} | ||||||
|  |  | ||||||
|     @admin_only |     @admin_only | ||||||
|     def modify_user_role(self, context, user, role, project=None, |     def modify_user_role(self, context, user, role, project=None, | ||||||
|                          operation='add', **kwargs): |                          operation='add', **kwargs): | ||||||
|   | |||||||
| @@ -25,12 +25,13 @@ import logging | |||||||
| import multiprocessing | import multiprocessing | ||||||
| import random | import random | ||||||
| import re | import re | ||||||
| import tornado.web |  | ||||||
| from twisted.internet import defer |  | ||||||
| import urllib | import urllib | ||||||
| # TODO(termie): replace minidom with etree | # TODO(termie): replace minidom with etree | ||||||
| from xml.dom import minidom | from xml.dom import minidom | ||||||
|  |  | ||||||
|  | import tornado.web | ||||||
|  | from twisted.internet import defer | ||||||
|  |  | ||||||
| from nova import crypto | from nova import crypto | ||||||
| from nova import exception | from nova import exception | ||||||
| from nova import flags | from nova import flags | ||||||
| @@ -43,6 +44,7 @@ from nova.endpoint import cloud | |||||||
| FLAGS = flags.FLAGS | FLAGS = flags.FLAGS | ||||||
| flags.DEFINE_integer('cc_port', 8773, 'cloud controller port') | flags.DEFINE_integer('cc_port', 8773, 'cloud controller port') | ||||||
|  |  | ||||||
|  |  | ||||||
| _log = logging.getLogger("api") | _log = logging.getLogger("api") | ||||||
| _log.setLevel(logging.DEBUG) | _log.setLevel(logging.DEBUG) | ||||||
|  |  | ||||||
| @@ -227,6 +229,7 @@ class MetadataRequestHandler(tornado.web.RequestHandler): | |||||||
|         self.print_data(data) |         self.print_data(data) | ||||||
|         self.finish() |         self.finish() | ||||||
|  |  | ||||||
|  |  | ||||||
| class APIRequestHandler(tornado.web.RequestHandler): | class APIRequestHandler(tornado.web.RequestHandler): | ||||||
|     def get(self, controller_name): |     def get(self, controller_name): | ||||||
|         self.execute(controller_name) |         self.execute(controller_name) | ||||||
|   | |||||||
| @@ -26,6 +26,7 @@ import base64 | |||||||
| import logging | import logging | ||||||
| import os | import os | ||||||
| import time | import time | ||||||
|  |  | ||||||
| from twisted.internet import defer | from twisted.internet import defer | ||||||
|  |  | ||||||
| from nova import datastore | from nova import datastore | ||||||
| @@ -44,9 +45,9 @@ from nova.volume import service | |||||||
|  |  | ||||||
|  |  | ||||||
| FLAGS = flags.FLAGS | FLAGS = flags.FLAGS | ||||||
|  |  | ||||||
| flags.DEFINE_string('cloud_topic', 'cloud', 'the topic clouds listen on') | flags.DEFINE_string('cloud_topic', 'cloud', 'the topic clouds listen on') | ||||||
|  |  | ||||||
|  |  | ||||||
| def _gen_key(user_id, key_name): | def _gen_key(user_id, key_name): | ||||||
|     """ Tuck this into AuthManager """ |     """ Tuck this into AuthManager """ | ||||||
|     try: |     try: | ||||||
| @@ -85,7 +86,7 @@ class CloudController(object): | |||||||
|         """ Ensure the keychains and folders exist. """ |         """ Ensure the keychains and folders exist. """ | ||||||
|         # Create keys folder, if it doesn't exist |         # Create keys folder, if it doesn't exist | ||||||
|         if not os.path.exists(FLAGS.keys_path): |         if not os.path.exists(FLAGS.keys_path): | ||||||
|             os.makedirs(os.path.abspath(FLAGS.keys_path)) |             os.makedirs(FLAGS.keys_path) | ||||||
|         # Gen root CA, if we don't have one |         # Gen root CA, if we don't have one | ||||||
|         root_ca_path = os.path.join(FLAGS.ca_path, FLAGS.ca_file) |         root_ca_path = os.path.join(FLAGS.ca_path, FLAGS.ca_file) | ||||||
|         if not os.path.exists(root_ca_path): |         if not os.path.exists(root_ca_path): | ||||||
| @@ -102,15 +103,16 @@ class CloudController(object): | |||||||
|         result = {} |         result = {} | ||||||
|         for instance in self.instdir.all: |         for instance in self.instdir.all: | ||||||
|             if instance['project_id'] == project_id: |             if instance['project_id'] == project_id: | ||||||
|                 line = '%s slots=%d' % (instance['private_dns_name'], INSTANCE_TYPES[instance['instance_type']]['vcpus']) |                 line = '%s slots=%d' % (instance['private_dns_name'], | ||||||
|  |                     INSTANCE_TYPES[instance['instance_type']]['vcpus']) | ||||||
|                 if instance['key_name'] in result: |                 if instance['key_name'] in result: | ||||||
|                     result[instance['key_name']].append(line) |                     result[instance['key_name']].append(line) | ||||||
|                 else: |                 else: | ||||||
|                     result[instance['key_name']] = [line] |                     result[instance['key_name']] = [line] | ||||||
|         return result |         return result | ||||||
|  |  | ||||||
|     def get_metadata(self, ip): |     def get_metadata(self, ipaddress): | ||||||
|         i = self.get_instance_by_ip(ip) |         i = self.get_instance_by_ip(ipaddress) | ||||||
|         if i is None: |         if i is None: | ||||||
|             return None |             return None | ||||||
|         mpi = self._get_mpi_data(i['project_id']) |         mpi = self._get_mpi_data(i['project_id']) | ||||||
| @@ -123,6 +125,12 @@ class CloudController(object): | |||||||
|             } |             } | ||||||
|         else: |         else: | ||||||
|             keys = '' |             keys = '' | ||||||
|  |  | ||||||
|  |         address_record = network_model.FixedIp(i['private_dns_name']) | ||||||
|  |         if address_record: | ||||||
|  |             hostname = address_record['hostname'] | ||||||
|  |         else: | ||||||
|  |             hostname = 'ip-%s' % i['private_dns_name'].replace('.', '-') | ||||||
|         data = { |         data = { | ||||||
|             'user-data': base64.b64decode(i['user_data']), |             'user-data': base64.b64decode(i['user_data']), | ||||||
|             'meta-data': { |             'meta-data': { | ||||||
| @@ -135,17 +143,17 @@ class CloudController(object): | |||||||
|                     'root': '/dev/sda1', |                     'root': '/dev/sda1', | ||||||
|                     'swap': 'sda3' |                     'swap': 'sda3' | ||||||
|                 }, |                 }, | ||||||
|                 'hostname': i['private_dns_name'], # is this public sometimes? |                 'hostname': hostname, | ||||||
|                 'instance-action': 'none', |                 'instance-action': 'none', | ||||||
|                 'instance-id': i['instance_id'], |                 'instance-id': i['instance_id'], | ||||||
|                 'instance-type': i.get('instance_type', ''), |                 'instance-type': i.get('instance_type', ''), | ||||||
|                 'local-hostname': i['private_dns_name'], |                 'local-hostname': hostname, | ||||||
|                 'local-ipv4': i['private_dns_name'], # TODO: switch to IP |                 'local-ipv4': i['private_dns_name'], # TODO: switch to IP | ||||||
|                 'kernel-id': i.get('kernel_id', ''), |                 'kernel-id': i.get('kernel_id', ''), | ||||||
|                 'placement': { |                 'placement': { | ||||||
|                     'availaibility-zone': i.get('availability_zone', 'nova'), |                     'availaibility-zone': i.get('availability_zone', 'nova'), | ||||||
|                 }, |                 }, | ||||||
|                 'public-hostname': i.get('dns_name', ''), |                 'public-hostname': hostname, | ||||||
|                 'public-ipv4': i.get('dns_name', ''), # TODO: switch to IP |                 'public-ipv4': i.get('dns_name', ''), # TODO: switch to IP | ||||||
|                 'public-keys': keys, |                 'public-keys': keys, | ||||||
|                 'ramdisk-id': i.get('ramdisk_id', ''), |                 'ramdisk-id': i.get('ramdisk_id', ''), | ||||||
| @@ -207,22 +215,18 @@ class CloudController(object): | |||||||
|  |  | ||||||
|     @rbac.allow('all') |     @rbac.allow('all') | ||||||
|     def create_key_pair(self, context, key_name, **kwargs): |     def create_key_pair(self, context, key_name, **kwargs): | ||||||
|         try: |         dcall = defer.Deferred() | ||||||
|             d = defer.Deferred() |         pool = context.handler.application.settings.get('pool') | ||||||
|             p = context.handler.application.settings.get('pool') |  | ||||||
|         def _complete(kwargs): |         def _complete(kwargs): | ||||||
|             if 'exception' in kwargs: |             if 'exception' in kwargs: | ||||||
|                     d.errback(kwargs['exception']) |                 dcall.errback(kwargs['exception']) | ||||||
|                 return |                 return | ||||||
|                 d.callback({'keyName': key_name, |             dcall.callback({'keyName': key_name, | ||||||
|                 'keyFingerprint': kwargs['fingerprint'], |                 'keyFingerprint': kwargs['fingerprint'], | ||||||
|                 'keyMaterial': kwargs['private_key']}) |                 'keyMaterial': kwargs['private_key']}) | ||||||
|             p.apply_async(_gen_key, [context.user.id, key_name], |         pool.apply_async(_gen_key, [context.user.id, key_name], | ||||||
|             callback=_complete) |             callback=_complete) | ||||||
|             return d |         return dcall | ||||||
|  |  | ||||||
|         except manager.UserError as e: |  | ||||||
|             raise |  | ||||||
|  |  | ||||||
|     @rbac.allow('all') |     @rbac.allow('all') | ||||||
|     def delete_key_pair(self, context, key_name, **kwargs): |     def delete_key_pair(self, context, key_name, **kwargs): | ||||||
| @@ -302,12 +306,12 @@ class CloudController(object): | |||||||
|                                            "user_id": context.user.id, |                                            "user_id": context.user.id, | ||||||
|                                            "project_id": context.project.id}}) |                                            "project_id": context.project.id}}) | ||||||
|         # NOTE(vish): rpc returned value is in the result key in the dictionary |         # NOTE(vish): rpc returned value is in the result key in the dictionary | ||||||
|         volume = self._get_volume(context, result['result']) |         volume = self._get_volume(context, result) | ||||||
|         defer.returnValue({'volumeSet': [self.format_volume(context, volume)]}) |         defer.returnValue({'volumeSet': [self.format_volume(context, volume)]}) | ||||||
|  |  | ||||||
|     def _get_address(self, context, public_ip): |     def _get_address(self, context, public_ip): | ||||||
|         # FIXME(vish) this should move into network.py |         # FIXME(vish) this should move into network.py | ||||||
|         address = network_model.PublicAddress.lookup(public_ip) |         address = network_model.ElasticIp.lookup(public_ip) | ||||||
|         if address and (context.user.is_admin() or address['project_id'] == context.project.id): |         if address and (context.user.is_admin() or address['project_id'] == context.project.id): | ||||||
|             return address |             return address | ||||||
|         raise exception.NotFound("Address at ip %s not found" % public_ip) |         raise exception.NotFound("Address at ip %s not found" % public_ip) | ||||||
| @@ -358,7 +362,6 @@ class CloudController(object): | |||||||
|                               'status': volume['attach_status'], |                               'status': volume['attach_status'], | ||||||
|                               'volumeId': volume_id}) |                               'volumeId': volume_id}) | ||||||
|  |  | ||||||
|  |  | ||||||
|     @rbac.allow('projectmanager', 'sysadmin') |     @rbac.allow('projectmanager', 'sysadmin') | ||||||
|     def detach_volume(self, context, volume_id, **kwargs): |     def detach_volume(self, context, volume_id, **kwargs): | ||||||
|         volume = self._get_volume(context, volume_id) |         volume = self._get_volume(context, volume_id) | ||||||
| @@ -394,7 +397,15 @@ class CloudController(object): | |||||||
|  |  | ||||||
|     @rbac.allow('all') |     @rbac.allow('all') | ||||||
|     def describe_instances(self, context, **kwargs): |     def describe_instances(self, context, **kwargs): | ||||||
|         return defer.succeed(self._format_instances(context)) |         return defer.succeed(self._format_describe_instances(context)) | ||||||
|  |  | ||||||
|  |     def _format_describe_instances(self, context): | ||||||
|  |         return { 'reservationSet': self._format_instances(context) } | ||||||
|  |  | ||||||
|  |     def _format_run_instances(self, context, reservation_id): | ||||||
|  |         i = self._format_instances(context, reservation_id) | ||||||
|  |         assert len(i) == 1 | ||||||
|  |         return i[0] | ||||||
|  |  | ||||||
|     def _format_instances(self, context, reservation_id = None): |     def _format_instances(self, context, reservation_id = None): | ||||||
|         reservations = {} |         reservations = {} | ||||||
| @@ -425,7 +436,8 @@ class CloudController(object): | |||||||
|             i['key_name'] = instance.get('key_name', None) |             i['key_name'] = instance.get('key_name', None) | ||||||
|             if context.user.is_admin(): |             if context.user.is_admin(): | ||||||
|                 i['key_name'] = '%s (%s, %s)' % (i['key_name'], |                 i['key_name'] = '%s (%s, %s)' % (i['key_name'], | ||||||
|                     instance.get('project_id', None), instance.get('node_name','')) |                     instance.get('project_id', None), | ||||||
|  |                     instance.get('node_name', '')) | ||||||
|             i['product_codes_set'] = self._convert_to_set( |             i['product_codes_set'] = self._convert_to_set( | ||||||
|                 instance.get('product_codes', None), 'product_code') |                 instance.get('product_codes', None), 'product_code') | ||||||
|             i['instance_type'] = instance.get('instance_type', None) |             i['instance_type'] = instance.get('instance_type', None) | ||||||
| @@ -442,8 +454,7 @@ class CloudController(object): | |||||||
|                 reservations[res_id] = r |                 reservations[res_id] = r | ||||||
|             reservations[res_id]['instances_set'].append(i) |             reservations[res_id]['instances_set'].append(i) | ||||||
|  |  | ||||||
|         instance_response = {'reservationSet' : list(reservations.values()) } |         return list(reservations.values()) | ||||||
|         return instance_response |  | ||||||
|  |  | ||||||
|     @rbac.allow('all') |     @rbac.allow('all') | ||||||
|     def describe_addresses(self, context, **kwargs): |     def describe_addresses(self, context, **kwargs): | ||||||
| @@ -451,7 +462,7 @@ class CloudController(object): | |||||||
|  |  | ||||||
|     def format_addresses(self, context): |     def format_addresses(self, context): | ||||||
|         addresses = [] |         addresses = [] | ||||||
|         for address in network_model.PublicAddress.all(): |         for address in network_model.ElasticIp.all(): | ||||||
|             # TODO(vish): implement a by_project iterator for addresses |             # TODO(vish): implement a by_project iterator for addresses | ||||||
|             if (context.user.is_admin() or |             if (context.user.is_admin() or | ||||||
|                 address['project_id'] == context.project.id): |                 address['project_id'] == context.project.id): | ||||||
| @@ -472,11 +483,10 @@ class CloudController(object): | |||||||
|     @defer.inlineCallbacks |     @defer.inlineCallbacks | ||||||
|     def allocate_address(self, context, **kwargs): |     def allocate_address(self, context, **kwargs): | ||||||
|         network_topic = yield self._get_network_topic(context) |         network_topic = yield self._get_network_topic(context) | ||||||
|         alloc_result = yield rpc.call(network_topic, |         public_ip = yield rpc.call(network_topic, | ||||||
|                          {"method": "allocate_elastic_ip", |                          {"method": "allocate_elastic_ip", | ||||||
|                           "args": {"user_id": context.user.id, |                           "args": {"user_id": context.user.id, | ||||||
|                                    "project_id": context.project.id}}) |                                    "project_id": context.project.id}}) | ||||||
|         public_ip = alloc_result['result'] |  | ||||||
|         defer.returnValue({'addressSet': [{'publicIp': public_ip}]}) |         defer.returnValue({'addressSet': [{'publicIp': public_ip}]}) | ||||||
|  |  | ||||||
|     @rbac.allow('netadmin') |     @rbac.allow('netadmin') | ||||||
| @@ -517,11 +527,10 @@ class CloudController(object): | |||||||
|         """Retrieves the network host for a project""" |         """Retrieves the network host for a project""" | ||||||
|         host = network_service.get_host_for_project(context.project.id) |         host = network_service.get_host_for_project(context.project.id) | ||||||
|         if not host: |         if not host: | ||||||
|             result = yield rpc.call(FLAGS.network_topic, |             host = yield rpc.call(FLAGS.network_topic, | ||||||
|                                     {"method": "set_network_host", |                                     {"method": "set_network_host", | ||||||
|                                      "args": {"user_id": context.user.id, |                                      "args": {"user_id": context.user.id, | ||||||
|                                               "project_id": context.project.id}}) |                                               "project_id": context.project.id}}) | ||||||
|             host = result['result'] |  | ||||||
|         defer.returnValue('%s.%s' %(FLAGS.network_topic, host)) |         defer.returnValue('%s.%s' %(FLAGS.network_topic, host)) | ||||||
|  |  | ||||||
|     @rbac.allow('projectmanager', 'sysadmin') |     @rbac.allow('projectmanager', 'sysadmin') | ||||||
| @@ -561,17 +570,17 @@ class CloudController(object): | |||||||
|         # TODO: Get the real security group of launch in here |         # TODO: Get the real security group of launch in here | ||||||
|         security_group = "default" |         security_group = "default" | ||||||
|         for num in range(int(kwargs['max_count'])): |         for num in range(int(kwargs['max_count'])): | ||||||
|             vpn = False |             is_vpn = False | ||||||
|             if image_id  == FLAGS.vpn_image_id: |             if image_id  == FLAGS.vpn_image_id: | ||||||
|                 vpn = True |                 is_vpn = True | ||||||
|             allocate_result = yield rpc.call(network_topic, |             inst = self.instdir.new() | ||||||
|  |             allocate_data = yield rpc.call(network_topic, | ||||||
|                      {"method": "allocate_fixed_ip", |                      {"method": "allocate_fixed_ip", | ||||||
|                       "args": {"user_id": context.user.id, |                       "args": {"user_id": context.user.id, | ||||||
|                                "project_id": context.project.id, |                                "project_id": context.project.id, | ||||||
|                                "security_group": security_group, |                                "security_group": security_group, | ||||||
|                                "vpn": vpn}}) |                                "is_vpn": is_vpn, | ||||||
|             allocate_data = allocate_result['result'] |                                "hostname": inst.instance_id}}) | ||||||
|             inst = self.instdir.new() |  | ||||||
|             inst['image_id'] = image_id |             inst['image_id'] = image_id | ||||||
|             inst['kernel_id'] = kernel_id |             inst['kernel_id'] = kernel_id | ||||||
|             inst['ramdisk_id'] = ramdisk_id |             inst['ramdisk_id'] = ramdisk_id | ||||||
| @@ -585,6 +594,7 @@ class CloudController(object): | |||||||
|             inst['project_id'] = context.project.id |             inst['project_id'] = context.project.id | ||||||
|             inst['ami_launch_index'] = num |             inst['ami_launch_index'] = num | ||||||
|             inst['security_group'] = security_group |             inst['security_group'] = security_group | ||||||
|  |             inst['hostname'] = inst.instance_id | ||||||
|             for (key, value) in allocate_data.iteritems(): |             for (key, value) in allocate_data.iteritems(): | ||||||
|                 inst[key] = value |                 inst[key] = value | ||||||
|  |  | ||||||
| @@ -595,7 +605,7 @@ class CloudController(object): | |||||||
|             logging.debug("Casting to node for %s's instance with IP of %s" % |             logging.debug("Casting to node for %s's instance with IP of %s" % | ||||||
|                       (context.user.name, inst['private_dns_name'])) |                       (context.user.name, inst['private_dns_name'])) | ||||||
|         # TODO: Make Network figure out the network name from ip. |         # TODO: Make Network figure out the network name from ip. | ||||||
|         defer.returnValue(self._format_instances(context, reservation_id)) |         defer.returnValue(self._format_run_instances(context, reservation_id)) | ||||||
|  |  | ||||||
|     @rbac.allow('projectmanager', 'sysadmin') |     @rbac.allow('projectmanager', 'sysadmin') | ||||||
|     @defer.inlineCallbacks |     @defer.inlineCallbacks | ||||||
|   | |||||||
| @@ -21,10 +21,11 @@ Proxy AMI-related calls from the cloud controller, to the running | |||||||
| objectstore daemon. | objectstore daemon. | ||||||
| """ | """ | ||||||
|  |  | ||||||
| import boto.s3.connection |  | ||||||
| import json | import json | ||||||
| import urllib | import urllib | ||||||
|  |  | ||||||
|  | import boto.s3.connection | ||||||
|  |  | ||||||
| from nova import flags | from nova import flags | ||||||
| from nova import utils | from nova import utils | ||||||
| from nova.auth import manager | from nova.auth import manager | ||||||
| @@ -32,6 +33,7 @@ from nova.auth import manager | |||||||
|  |  | ||||||
| FLAGS = flags.FLAGS | FLAGS = flags.FLAGS | ||||||
|  |  | ||||||
|  |  | ||||||
| def modify(context, image_id, operation): | def modify(context, image_id, operation): | ||||||
|     conn(context).make_request( |     conn(context).make_request( | ||||||
|         method='POST', |         method='POST', | ||||||
| @@ -53,6 +55,7 @@ def register(context, image_location): | |||||||
|  |  | ||||||
|     return image_id |     return image_id | ||||||
|  |  | ||||||
|  |  | ||||||
| def list(context, filter_list=[]): | def list(context, filter_list=[]): | ||||||
|     """ return a list of all images that a user can see |     """ return a list of all images that a user can see | ||||||
|  |  | ||||||
| @@ -68,6 +71,7 @@ def list(context, filter_list=[]): | |||||||
|         return [i for i in result if i['imageId'] in filter_list] |         return [i for i in result if i['imageId'] in filter_list] | ||||||
|     return result |     return result | ||||||
|  |  | ||||||
|  |  | ||||||
| def deregister(context, image_id): | def deregister(context, image_id): | ||||||
|     """ unregister an image """ |     """ unregister an image """ | ||||||
|     conn(context).make_request( |     conn(context).make_request( | ||||||
| @@ -75,6 +79,7 @@ def deregister(context, image_id): | |||||||
|             bucket='_images', |             bucket='_images', | ||||||
|             query_args=qs({'image_id': image_id})) |             query_args=qs({'image_id': image_id})) | ||||||
|  |  | ||||||
|  |  | ||||||
| def conn(context): | def conn(context): | ||||||
|     access = manager.AuthManager().get_access_key(context.user, |     access = manager.AuthManager().get_access_key(context.user, | ||||||
|                                                   context.project) |                                                   context.project) | ||||||
|   | |||||||
| @@ -1,226 +0,0 @@ | |||||||
| # vim: tabstop=4 shiftwidth=4 softtabstop=4 |  | ||||||
|  |  | ||||||
| # Copyright 2010 United States Government as represented by the |  | ||||||
| # Administrator of the National Aeronautics and Space Administration. |  | ||||||
| # 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. |  | ||||||
|  |  | ||||||
| """ |  | ||||||
| Rackspace API |  | ||||||
| """ |  | ||||||
|  |  | ||||||
| import base64 |  | ||||||
| import json |  | ||||||
| import logging |  | ||||||
| import multiprocessing |  | ||||||
| import os |  | ||||||
| import time |  | ||||||
| import tornado.web |  | ||||||
| from twisted.internet import defer |  | ||||||
|  |  | ||||||
| from nova import datastore |  | ||||||
| from nova import exception |  | ||||||
| from nova import flags |  | ||||||
| from nova import rpc |  | ||||||
| from nova import utils |  | ||||||
| from nova.auth import manager |  | ||||||
| from nova.compute import model |  | ||||||
| from nova.compute import network |  | ||||||
| from nova.endpoint import images |  | ||||||
| from nova.endpoint import wsgi |  | ||||||
|  |  | ||||||
|  |  | ||||||
| FLAGS = flags.FLAGS |  | ||||||
| flags.DEFINE_string('cloud_topic', 'cloud', 'the topic clouds listen on') |  | ||||||
|  |  | ||||||
|  |  | ||||||
| # TODO(todd): subclass Exception so we can bubble meaningful errors |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class Api(object): |  | ||||||
|  |  | ||||||
|     def __init__(self, rpc_mechanism): |  | ||||||
|         self.controllers = { |  | ||||||
|             "v1.0":   RackspaceAuthenticationApi(), |  | ||||||
|             "servers": RackspaceCloudServerApi() |  | ||||||
|         } |  | ||||||
|         self.rpc_mechanism = rpc_mechanism |  | ||||||
|  |  | ||||||
|     def handler(self, environ, responder): |  | ||||||
|         environ['nova.context'] = self.build_context(environ) |  | ||||||
|         controller, path = wsgi.Util.route( |  | ||||||
|                              environ['PATH_INFO'], |  | ||||||
|                              self.controllers |  | ||||||
|                            ) |  | ||||||
|         if not controller: |  | ||||||
|             # TODO(todd): Exception (404) |  | ||||||
|             raise Exception("Missing Controller") |  | ||||||
|         rv = controller.process(path, environ) |  | ||||||
|         if type(rv) is tuple: |  | ||||||
|             responder(rv[0], rv[1]) |  | ||||||
|             rv = rv[2] |  | ||||||
|         else: |  | ||||||
|             responder("200 OK", []) |  | ||||||
|         return rv |  | ||||||
|  |  | ||||||
|     def build_context(self, env): |  | ||||||
|         rv = {} |  | ||||||
|         if env.has_key("HTTP_X_AUTH_TOKEN"): |  | ||||||
|             rv['user'] = manager.AuthManager().get_user_from_access_key( |  | ||||||
|                            env['HTTP_X_AUTH_TOKEN'] |  | ||||||
|                          ) |  | ||||||
|             if rv['user']: |  | ||||||
|                 rv['project'] = manager.AuthManager().get_project( |  | ||||||
|                                   rv['user'].name |  | ||||||
|                                 ) |  | ||||||
|         return rv |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class RackspaceApiEndpoint(object): |  | ||||||
|     def process(self, path, env): |  | ||||||
|         if not self.check_authentication(env): |  | ||||||
|             # TODO(todd): Exception (Unauthorized) |  | ||||||
|             raise Exception("Unable to authenticate") |  | ||||||
|  |  | ||||||
|         if len(path) == 0: |  | ||||||
|             return self.index(env) |  | ||||||
|  |  | ||||||
|         action = path.pop(0) |  | ||||||
|         if hasattr(self, action): |  | ||||||
|             method = getattr(self, action) |  | ||||||
|             return method(path, env) |  | ||||||
|         else: |  | ||||||
|             # TODO(todd): Exception (404) |  | ||||||
|             raise Exception("Missing method %s" % path[0]) |  | ||||||
|  |  | ||||||
|     def check_authentication(self, env): |  | ||||||
|         if hasattr(self, "process_without_authentication") \ |  | ||||||
|         and getattr(self, "process_without_authentication"): |  | ||||||
|             return True |  | ||||||
|         if not env['nova.context']['user']: |  | ||||||
|             return False |  | ||||||
|         return True |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class RackspaceAuthenticationApi(RackspaceApiEndpoint): |  | ||||||
|  |  | ||||||
|     def __init__(self): |  | ||||||
|         self.process_without_authentication = True |  | ||||||
|  |  | ||||||
|     # TODO(todd): make a actual session with a unique token |  | ||||||
|     # just pass the auth key back through for now |  | ||||||
|     def index(self, env): |  | ||||||
|         response = '204 No Content' |  | ||||||
|         headers = [ |  | ||||||
|             ('X-Server-Management-Url', 'http://%s' % env['HTTP_HOST']), |  | ||||||
|             ('X-Storage-Url', 'http://%s' % env['HTTP_HOST']), |  | ||||||
|             ('X-CDN-Managment-Url', 'http://%s' % env['HTTP_HOST']), |  | ||||||
|             ('X-Auth-Token', env['HTTP_X_AUTH_KEY']) |  | ||||||
|         ] |  | ||||||
|         body = "" |  | ||||||
|         return (response, headers, body) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class RackspaceCloudServerApi(RackspaceApiEndpoint): |  | ||||||
|  |  | ||||||
|     def __init__(self): |  | ||||||
|         self.instdir = model.InstanceDirectory() |  | ||||||
|         self.network = network.PublicNetworkController() |  | ||||||
|  |  | ||||||
|     def index(self, env): |  | ||||||
|         if env['REQUEST_METHOD'] == 'GET': |  | ||||||
|             return self.detail(env) |  | ||||||
|         elif env['REQUEST_METHOD'] == 'POST': |  | ||||||
|             return self.launch_server(env) |  | ||||||
|  |  | ||||||
|     def detail(self, args, env): |  | ||||||
|         value = { |  | ||||||
|             "servers": |  | ||||||
|                 [] |  | ||||||
|         } |  | ||||||
|         for inst in self.instdir.all: |  | ||||||
|             value["servers"].append(self.instance_details(inst)) |  | ||||||
|  |  | ||||||
|         return json.dumps(value) |  | ||||||
|  |  | ||||||
|     ## |  | ||||||
|     ## |  | ||||||
|  |  | ||||||
|     def launch_server(self, env): |  | ||||||
|         data = json.loads(env['wsgi.input'].read(int(env['CONTENT_LENGTH']))) |  | ||||||
|         inst = self.build_server_instance(data, env['nova.context']) |  | ||||||
|         self.schedule_launch_of_instance(inst) |  | ||||||
|         return json.dumps({"server": self.instance_details(inst)}) |  | ||||||
|  |  | ||||||
|     def instance_details(self, inst): |  | ||||||
|         return { |  | ||||||
|             "id": inst.get("instance_id", None), |  | ||||||
|             "imageId": inst.get("image_id", None), |  | ||||||
|             "flavorId": inst.get("instacne_type", None), |  | ||||||
|             "hostId": inst.get("node_name", None), |  | ||||||
|             "status": inst.get("state", "pending"), |  | ||||||
|             "addresses": { |  | ||||||
|                 "public": [self.network.get_public_ip_for_instance( |  | ||||||
|                             inst.get("instance_id", None) |  | ||||||
|                           )], |  | ||||||
|                 "private": [inst.get("private_dns_name", None)] |  | ||||||
|             }, |  | ||||||
|  |  | ||||||
|             # implemented only by Rackspace, not AWS |  | ||||||
|             "name": inst.get("name", "Not-Specified"), |  | ||||||
|  |  | ||||||
|             # not supported |  | ||||||
|             "progress": "Not-Supported", |  | ||||||
|             "metadata": { |  | ||||||
|                 "Server Label": "Not-Supported", |  | ||||||
|                 "Image Version": "Not-Supported" |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|     def build_server_instance(self, env, context): |  | ||||||
|         reservation = utils.generate_uid('r') |  | ||||||
|         ltime = time.strftime('%Y-%m-%dT%H:%M:%SZ', time.gmtime()) |  | ||||||
|         inst = self.instdir.new() |  | ||||||
|         inst['name'] = env['server']['name'] |  | ||||||
|         inst['image_id'] = env['server']['imageId'] |  | ||||||
|         inst['instance_type'] = env['server']['flavorId'] |  | ||||||
|         inst['user_id'] = context['user'].id |  | ||||||
|         inst['project_id'] = context['project'].id |  | ||||||
|         inst['reservation_id'] = reservation |  | ||||||
|         inst['launch_time'] = ltime |  | ||||||
|         inst['mac_address'] = utils.generate_mac() |  | ||||||
|         address = network.allocate_ip( |  | ||||||
|                     inst['user_id'], |  | ||||||
|                     inst['project_id'], |  | ||||||
|                     mac=inst['mac_address'] |  | ||||||
|                   ) |  | ||||||
|         inst['private_dns_name'] = str(address) |  | ||||||
|         inst['bridge_name'] = network.BridgedNetwork.get_network_for_project( |  | ||||||
|                                 inst['user_id'], |  | ||||||
|                                 inst['project_id'], |  | ||||||
|                                 'default' # security group |  | ||||||
|                               )['bridge_name'] |  | ||||||
|         # key_data, key_name, ami_launch_index |  | ||||||
|         # TODO(todd): key data or root password |  | ||||||
|         inst.save() |  | ||||||
|         return inst |  | ||||||
|  |  | ||||||
|     def schedule_launch_of_instance(self, inst): |  | ||||||
|         rpc.cast( |  | ||||||
|             FLAGS.compute_topic, |  | ||||||
|             { |  | ||||||
|                 "method": "run_instance", |  | ||||||
|                 "args": {"instance_id": inst.instance_id} |  | ||||||
|             } |  | ||||||
|         ) |  | ||||||
| @@ -16,12 +16,13 @@ | |||||||
| #    License for the specific language governing permissions and limitations | #    License for the specific language governing permissions and limitations | ||||||
| #    under the License. | #    under the License. | ||||||
|  |  | ||||||
| """ Based a bit on the carrot.backeds.queue backend... but a lot better """ | """Based a bit on the carrot.backeds.queue backend... but a lot better.""" | ||||||
|  |  | ||||||
| from carrot.backends import base |  | ||||||
| import logging | import logging | ||||||
| import Queue as queue | import Queue as queue | ||||||
|  |  | ||||||
|  | from carrot.backends import base | ||||||
|  |  | ||||||
|  |  | ||||||
| class Message(base.BaseMessage): | class Message(base.BaseMessage): | ||||||
|     pass |     pass | ||||||
|   | |||||||
							
								
								
									
										165
									
								
								nova/flags.py
									
									
									
									
									
								
							
							
						
						
									
										165
									
								
								nova/flags.py
									
									
									
									
									
								
							| @@ -21,16 +21,145 @@ Package-level global flags are defined here, the rest are defined | |||||||
| where they're used. | where they're used. | ||||||
| """ | """ | ||||||
|  |  | ||||||
|  | import getopt | ||||||
| import socket | import socket | ||||||
|  | import sys | ||||||
|  |  | ||||||
|  | import gflags | ||||||
|  |  | ||||||
|  |  | ||||||
| from gflags import * | class FlagValues(gflags.FlagValues): | ||||||
|  |     """Extension of gflags.FlagValues that allows undefined and runtime flags. | ||||||
|  |  | ||||||
|  |     Unknown flags will be ignored when parsing the command line, but the | ||||||
|  |     command line will be kept so that it can be replayed if new flags are | ||||||
|  |     defined after the initial parsing. | ||||||
|  |      | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     def __init__(self): | ||||||
|  |         gflags.FlagValues.__init__(self) | ||||||
|  |         self.__dict__['__dirty'] = [] | ||||||
|  |         self.__dict__['__was_already_parsed'] = False | ||||||
|  |         self.__dict__['__stored_argv'] = [] | ||||||
|  |  | ||||||
|  |     def __call__(self, argv): | ||||||
|  |         # We're doing some hacky stuff here so that we don't have to copy | ||||||
|  |         # out all the code of the original verbatim and then tweak a few lines. | ||||||
|  |         # We're hijacking the output of getopt so we can still return the | ||||||
|  |         # leftover args at the end | ||||||
|  |         sneaky_unparsed_args = {"value": None} | ||||||
|  |         original_argv = list(argv) | ||||||
|  |          | ||||||
|  |         if self.IsGnuGetOpt(): | ||||||
|  |             orig_getopt = getattr(getopt, 'gnu_getopt') | ||||||
|  |             orig_name = 'gnu_getopt' | ||||||
|  |         else: | ||||||
|  |             orig_getopt = getattr(getopt, 'getopt') | ||||||
|  |             orig_name = 'getopt' | ||||||
|  |  | ||||||
|  |         def _sneaky(*args, **kw): | ||||||
|  |             optlist, unparsed_args = orig_getopt(*args, **kw) | ||||||
|  |             sneaky_unparsed_args['value'] = unparsed_args | ||||||
|  |             return optlist, unparsed_args | ||||||
|  |  | ||||||
|  |         try: | ||||||
|  |             setattr(getopt, orig_name, _sneaky) | ||||||
|  |             args = gflags.FlagValues.__call__(self, argv) | ||||||
|  |         except gflags.UnrecognizedFlagError: | ||||||
|  |             # Undefined args were found, for now we don't care so just | ||||||
|  |             # act like everything went well | ||||||
|  |             # (these three lines are copied pretty much verbatim from the end | ||||||
|  |             # of the __call__ function we are wrapping) | ||||||
|  |             unparsed_args = sneaky_unparsed_args['value'] | ||||||
|  |             if unparsed_args: | ||||||
|  |                 if self.IsGnuGetOpt(): | ||||||
|  |                     args = argv[:1] + unparsed | ||||||
|  |                 else: | ||||||
|  |                     args = argv[:1] + original_argv[-len(unparsed_args):] | ||||||
|  |             else: | ||||||
|  |                 args = argv[:1] | ||||||
|  |         finally: | ||||||
|  |             setattr(getopt, orig_name, orig_getopt) | ||||||
|  |          | ||||||
|  |         # Store the arguments for later, we'll need them for new flags | ||||||
|  |         # added at runtime | ||||||
|  |         self.__dict__['__stored_argv'] = original_argv | ||||||
|  |         self.__dict__['__was_already_parsed'] = True | ||||||
|  |         self.ClearDirty() | ||||||
|  |         return args | ||||||
|  |  | ||||||
|  |     def SetDirty(self, name): | ||||||
|  |         """Mark a flag as dirty so that accessing it will case a reparse.""" | ||||||
|  |         self.__dict__['__dirty'].append(name) | ||||||
|  |      | ||||||
|  |     def IsDirty(self, name): | ||||||
|  |         return name in self.__dict__['__dirty'] | ||||||
|  |  | ||||||
|  |     def ClearDirty(self): | ||||||
|  |         self.__dict__['__is_dirty'] = [] | ||||||
|  |  | ||||||
|  |     def WasAlreadyParsed(self): | ||||||
|  |         return self.__dict__['__was_already_parsed'] | ||||||
|  |  | ||||||
|  |     def ParseNewFlags(self): | ||||||
|  |         if '__stored_argv' not in self.__dict__: | ||||||
|  |             return | ||||||
|  |         new_flags = FlagValues() | ||||||
|  |         for k in self.__dict__['__dirty']: | ||||||
|  |             new_flags[k] = gflags.FlagValues.__getitem__(self, k) | ||||||
|  |  | ||||||
|  |         new_flags(self.__dict__['__stored_argv']) | ||||||
|  |         for k in self.__dict__['__dirty']: | ||||||
|  |             setattr(self, k, getattr(new_flags, k)) | ||||||
|  |         self.ClearDirty() | ||||||
|  |          | ||||||
|  |     def __setitem__(self, name, flag): | ||||||
|  |         gflags.FlagValues.__setitem__(self, name, flag) | ||||||
|  |         if self.WasAlreadyParsed(): | ||||||
|  |             self.SetDirty(name) | ||||||
|  |      | ||||||
|  |     def __getitem__(self, name): | ||||||
|  |         if self.IsDirty(name): | ||||||
|  |             self.ParseNewFlags() | ||||||
|  |         return gflags.FlagValues.__getitem__(self, name) | ||||||
|  |  | ||||||
|  |     def __getattr__(self, name): | ||||||
|  |         if self.IsDirty(name): | ||||||
|  |             self.ParseNewFlags() | ||||||
|  |         return gflags.FlagValues.__getattr__(self, name) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | FLAGS = FlagValues() | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def _wrapper(func): | ||||||
|  |     def _wrapped(*args, **kw): | ||||||
|  |         kw.setdefault('flag_values', FLAGS) | ||||||
|  |         func(*args, **kw) | ||||||
|  |     _wrapped.func_name = func.func_name | ||||||
|  |     return _wrapped | ||||||
|  |  | ||||||
|  |  | ||||||
|  | DEFINE_string = _wrapper(gflags.DEFINE_string) | ||||||
|  | DEFINE_integer = _wrapper(gflags.DEFINE_integer) | ||||||
|  | DEFINE_bool = _wrapper(gflags.DEFINE_bool) | ||||||
|  | DEFINE_boolean = _wrapper(gflags.DEFINE_boolean) | ||||||
|  | DEFINE_float = _wrapper(gflags.DEFINE_float) | ||||||
|  | DEFINE_enum = _wrapper(gflags.DEFINE_enum) | ||||||
|  | DEFINE_list = _wrapper(gflags.DEFINE_list) | ||||||
|  | DEFINE_spaceseplist = _wrapper(gflags.DEFINE_spaceseplist) | ||||||
|  | DEFINE_multistring = _wrapper(gflags.DEFINE_multistring) | ||||||
|  | DEFINE_multi_int = _wrapper(gflags.DEFINE_multi_int) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def DECLARE(name, module_string, flag_values=FLAGS): | ||||||
|  |     if module_string not in sys.modules: | ||||||
|  |         __import__(module_string, globals(), locals()) | ||||||
|  |     if name not in flag_values: | ||||||
|  |         raise gflags.UnrecognizedFlag( | ||||||
|  |                 "%s not defined by %s" % (name, module_string)) | ||||||
|  |  | ||||||
| # This keeps pylint from barfing on the imports |  | ||||||
| FLAGS = FLAGS |  | ||||||
| DEFINE_string = DEFINE_string |  | ||||||
| DEFINE_integer = DEFINE_integer |  | ||||||
| DEFINE_bool = DEFINE_bool |  | ||||||
|  |  | ||||||
| # __GLOBAL FLAGS ONLY__ | # __GLOBAL FLAGS ONLY__ | ||||||
| # Define any app-specific flags in their own files, docs at: | # Define any app-specific flags in their own files, docs at: | ||||||
| @@ -46,28 +175,24 @@ DEFINE_string('network_topic', 'network', 'the topic network nodes listen on') | |||||||
|  |  | ||||||
| DEFINE_bool('verbose', False, 'show debug output') | DEFINE_bool('verbose', False, 'show debug output') | ||||||
| DEFINE_boolean('fake_rabbit', False, 'use a fake rabbit') | DEFINE_boolean('fake_rabbit', False, 'use a fake rabbit') | ||||||
| DEFINE_bool('fake_network', False, 'should we use fake network devices and addresses') | DEFINE_bool('fake_network', False, | ||||||
|  |             'should we use fake network devices and addresses') | ||||||
| DEFINE_string('rabbit_host', 'localhost', 'rabbit host') | DEFINE_string('rabbit_host', 'localhost', 'rabbit host') | ||||||
| DEFINE_integer('rabbit_port', 5672, 'rabbit port') | DEFINE_integer('rabbit_port', 5672, 'rabbit port') | ||||||
| DEFINE_string('rabbit_userid', 'guest', 'rabbit userid') | DEFINE_string('rabbit_userid', 'guest', 'rabbit userid') | ||||||
| DEFINE_string('rabbit_password', 'guest', 'rabbit password') | DEFINE_string('rabbit_password', 'guest', 'rabbit password') | ||||||
| DEFINE_string('rabbit_virtual_host', '/', 'rabbit virtual host') | DEFINE_string('rabbit_virtual_host', '/', 'rabbit virtual host') | ||||||
| DEFINE_string('control_exchange', 'nova', 'the main exchange to connect to') | DEFINE_string('control_exchange', 'nova', 'the main exchange to connect to') | ||||||
| DEFINE_string('ec2_url', | DEFINE_string('ec2_url', 'http://127.0.0.1:8773/services/Cloud', | ||||||
|                 'http://127.0.0.1:8773/services/Cloud', |  | ||||||
|               'Url to ec2 api server') |               'Url to ec2 api server') | ||||||
|  |  | ||||||
| DEFINE_string('default_image', | DEFINE_string('default_image', 'ami-11111', | ||||||
|                     'ami-11111', |  | ||||||
|               'default image to use, testing only') |               'default image to use, testing only') | ||||||
| DEFINE_string('default_kernel', | DEFINE_string('default_kernel', 'aki-11111', | ||||||
|                     'aki-11111', |  | ||||||
|               'default kernel to use, testing only') |               'default kernel to use, testing only') | ||||||
| DEFINE_string('default_ramdisk', | DEFINE_string('default_ramdisk', 'ari-11111', | ||||||
|                     'ari-11111', |  | ||||||
|               'default ramdisk to use, testing only') |               'default ramdisk to use, testing only') | ||||||
| DEFINE_string('default_instance_type', | DEFINE_string('default_instance_type', 'm1.small', | ||||||
|                     'm1.small', |  | ||||||
|               'default instance type to use, testing only') |               'default instance type to use, testing only') | ||||||
|  |  | ||||||
| DEFINE_string('vpn_image_id', 'ami-CLOUDPIPE', 'AMI for cloudpipe vpn server') | DEFINE_string('vpn_image_id', 'ami-CLOUDPIPE', 'AMI for cloudpipe vpn server') | ||||||
| @@ -78,10 +203,8 @@ DEFINE_string('vpn_key_suffix', | |||||||
| DEFINE_integer('auth_token_ttl', 3600, 'Seconds for auth tokens to linger') | DEFINE_integer('auth_token_ttl', 3600, 'Seconds for auth tokens to linger') | ||||||
|  |  | ||||||
| # UNUSED | # UNUSED | ||||||
| DEFINE_string('node_availability_zone', | DEFINE_string('node_availability_zone', 'nova', | ||||||
|                     'nova', |  | ||||||
|               'availability zone of this node') |               'availability zone of this node') | ||||||
| DEFINE_string('node_name', | DEFINE_string('node_name', socket.gethostname(), | ||||||
|                     socket.gethostname(), |  | ||||||
|               'name of this node') |               'name of this node') | ||||||
|  |  | ||||||
|   | |||||||
| @@ -22,6 +22,7 @@ Process pool, still buggy right now. | |||||||
| """ | """ | ||||||
|  |  | ||||||
| import StringIO | import StringIO | ||||||
|  |  | ||||||
| from twisted.internet import defer | from twisted.internet import defer | ||||||
| from twisted.internet import error | from twisted.internet import error | ||||||
| from twisted.internet import protocol | from twisted.internet import protocol | ||||||
| @@ -190,6 +191,7 @@ class ProcessPool(object): | |||||||
|         self._pool.release() |         self._pool.release() | ||||||
|         return retval |         return retval | ||||||
|  |  | ||||||
|  |  | ||||||
| class SharedPool(object): | class SharedPool(object): | ||||||
|     _instance = None |     _instance = None | ||||||
|     def __init__(self): |     def __init__(self): | ||||||
| @@ -198,5 +200,6 @@ class SharedPool(object): | |||||||
|     def __getattr__(self, key): |     def __getattr__(self, key): | ||||||
|         return getattr(self._instance, key) |         return getattr(self._instance, key) | ||||||
|  |  | ||||||
|  |  | ||||||
| def simple_execute(cmd, **kwargs): | def simple_execute(cmd, **kwargs): | ||||||
|     return SharedPool().simple_execute(cmd, **kwargs) |     return SharedPool().simple_execute(cmd, **kwargs) | ||||||
|   | |||||||
							
								
								
									
										146
									
								
								nova/rpc.py
									
									
									
									
									
								
							
							
						
						
									
										146
									
								
								nova/rpc.py
									
									
									
									
									
								
							| @@ -21,14 +21,14 @@ AMQP-based RPC. Queues have consumers and publishers. | |||||||
| No fan-out support yet. | No fan-out support yet. | ||||||
| """ | """ | ||||||
|  |  | ||||||
| from carrot import connection |  | ||||||
| from carrot import messaging |  | ||||||
| import json | import json | ||||||
| import logging | import logging | ||||||
| import sys | import sys | ||||||
| import uuid | import uuid | ||||||
|  |  | ||||||
|  | from carrot import connection as carrot_connection | ||||||
|  | from carrot import messaging | ||||||
| from twisted.internet import defer | from twisted.internet import defer | ||||||
| from twisted.internet import reactor |  | ||||||
| from twisted.internet import task | from twisted.internet import task | ||||||
|  |  | ||||||
| from nova import exception | from nova import exception | ||||||
| @@ -39,13 +39,15 @@ from nova import flags | |||||||
| FLAGS = flags.FLAGS | FLAGS = flags.FLAGS | ||||||
|  |  | ||||||
|  |  | ||||||
| _log = logging.getLogger('amqplib') | LOG = logging.getLogger('amqplib') | ||||||
| _log.setLevel(logging.WARN) | LOG.setLevel(logging.DEBUG) | ||||||
|  |  | ||||||
|  |  | ||||||
| class Connection(connection.BrokerConnection): | class Connection(carrot_connection.BrokerConnection): | ||||||
|  |     """Connection instance object""" | ||||||
|     @classmethod |     @classmethod | ||||||
|     def instance(cls): |     def instance(cls): | ||||||
|  |         """Returns the instance""" | ||||||
|         if not hasattr(cls, '_instance'): |         if not hasattr(cls, '_instance'): | ||||||
|             params = dict(hostname=FLAGS.rabbit_host, |             params = dict(hostname=FLAGS.rabbit_host, | ||||||
|                           port=FLAGS.rabbit_port, |                           port=FLAGS.rabbit_port, | ||||||
| @@ -56,18 +58,33 @@ class Connection(connection.BrokerConnection): | |||||||
|             if FLAGS.fake_rabbit: |             if FLAGS.fake_rabbit: | ||||||
|                 params['backend_cls'] = fakerabbit.Backend |                 params['backend_cls'] = fakerabbit.Backend | ||||||
|  |  | ||||||
|  |             # NOTE(vish): magic is fun! | ||||||
|  |             # pylint: disable-msg=W0142 | ||||||
|             cls._instance = cls(**params) |             cls._instance = cls(**params) | ||||||
|         return cls._instance |         return cls._instance | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def recreate(cls): |     def recreate(cls): | ||||||
|  |         """Recreates the connection instance | ||||||
|  |  | ||||||
|  |         This is necessary to recover from some network errors/disconnects""" | ||||||
|         del cls._instance |         del cls._instance | ||||||
|         return cls.instance() |         return cls.instance() | ||||||
|  |  | ||||||
|  |  | ||||||
| class Consumer(messaging.Consumer): | class Consumer(messaging.Consumer): | ||||||
|  |     """Consumer base class | ||||||
|  |  | ||||||
|  |     Contains methods for connecting the fetch method to async loops | ||||||
|  |     """ | ||||||
|  |     def __init__(self, *args, **kwargs): | ||||||
|  |         self.failed_connection = False | ||||||
|  |         super(Consumer, self).__init__(*args, **kwargs) | ||||||
|  |  | ||||||
|     # TODO(termie): it would be nice to give these some way of automatically |     # TODO(termie): it would be nice to give these some way of automatically | ||||||
|     #               cleaning up after themselves |     #               cleaning up after themselves | ||||||
|     def attach_to_tornado(self, io_inst=None): |     def attach_to_tornado(self, io_inst=None): | ||||||
|  |         """Attach a callback to tornado that fires 10 times a second""" | ||||||
|         from tornado import ioloop |         from tornado import ioloop | ||||||
|         if io_inst is None: |         if io_inst is None: | ||||||
|             io_inst = ioloop.IOLoop.instance() |             io_inst = ioloop.IOLoop.instance() | ||||||
| @@ -79,33 +96,44 @@ class Consumer(messaging.Consumer): | |||||||
|  |  | ||||||
|     attachToTornado = attach_to_tornado |     attachToTornado = attach_to_tornado | ||||||
|  |  | ||||||
|     def fetch(self, *args, **kwargs): |     def fetch(self, no_ack=None, auto_ack=None, enable_callbacks=False): | ||||||
|  |         """Wraps the parent fetch with some logic for failed connections""" | ||||||
|         # TODO(vish): the logic for failed connections and logging should be |         # TODO(vish): the logic for failed connections and logging should be | ||||||
|         #             refactored into some sort of connection manager object |         #             refactored into some sort of connection manager object | ||||||
|         try: |         try: | ||||||
|             if getattr(self, 'failed_connection', False): |             if self.failed_connection: | ||||||
|                 # attempt to reconnect |                 # NOTE(vish): conn is defined in the parent class, we can | ||||||
|  |                 #             recreate it as long as we create the backend too | ||||||
|  |                 # pylint: disable-msg=W0201 | ||||||
|                 self.conn = Connection.recreate() |                 self.conn = Connection.recreate() | ||||||
|                 self.backend = self.conn.create_backend() |                 self.backend = self.conn.create_backend() | ||||||
|             super(Consumer, self).fetch(*args, **kwargs) |             super(Consumer, self).fetch(no_ack, auto_ack, enable_callbacks) | ||||||
|             if getattr(self, 'failed_connection', False): |             if self.failed_connection: | ||||||
|                 logging.error("Reconnected to queue") |                 logging.error("Reconnected to queue") | ||||||
|                 self.failed_connection = False |                 self.failed_connection = False | ||||||
|         except Exception, ex: |         # NOTE(vish): This is catching all errors because we really don't | ||||||
|             if not getattr(self, 'failed_connection', False): |         #             exceptions to be logged 10 times a second if some | ||||||
|  |         #             persistent failure occurs. | ||||||
|  |         except Exception:  # pylint: disable-msg=W0703 | ||||||
|  |             if not self.failed_connection: | ||||||
|                 logging.exception("Failed to fetch message from queue") |                 logging.exception("Failed to fetch message from queue") | ||||||
|                 self.failed_connection = True |                 self.failed_connection = True | ||||||
|  |  | ||||||
|     def attach_to_twisted(self): |     def attach_to_twisted(self): | ||||||
|  |         """Attach a callback to twisted that fires 10 times a second""" | ||||||
|         loop = task.LoopingCall(self.fetch, enable_callbacks=True) |         loop = task.LoopingCall(self.fetch, enable_callbacks=True) | ||||||
|         loop.start(interval=0.1) |         loop.start(interval=0.1) | ||||||
|  |  | ||||||
|  |  | ||||||
| class Publisher(messaging.Publisher): | class Publisher(messaging.Publisher): | ||||||
|  |     """Publisher base class""" | ||||||
|     pass |     pass | ||||||
|  |  | ||||||
|  |  | ||||||
| class TopicConsumer(Consumer): | class TopicConsumer(Consumer): | ||||||
|  |     """Consumes messages on a specific topic""" | ||||||
|     exchange_type = "topic" |     exchange_type = "topic" | ||||||
|  |  | ||||||
|     def __init__(self, connection=None, topic="broadcast"): |     def __init__(self, connection=None, topic="broadcast"): | ||||||
|         self.queue = topic |         self.queue = topic | ||||||
|         self.routing_key = topic |         self.routing_key = topic | ||||||
| @@ -115,14 +143,24 @@ class TopicConsumer(Consumer): | |||||||
|  |  | ||||||
|  |  | ||||||
| class AdapterConsumer(TopicConsumer): | class AdapterConsumer(TopicConsumer): | ||||||
|  |     """Calls methods on a proxy object based on method and args""" | ||||||
|     def __init__(self, connection=None, topic="broadcast", proxy=None): |     def __init__(self, connection=None, topic="broadcast", proxy=None): | ||||||
|         _log.debug('Initing the Adapter Consumer for %s' % (topic)) |         LOG.debug('Initing the Adapter Consumer for %s' % (topic)) | ||||||
|         self.proxy = proxy |         self.proxy = proxy | ||||||
|         super(AdapterConsumer, self).__init__(connection=connection, topic=topic) |         super(AdapterConsumer, self).__init__(connection=connection, | ||||||
|  |                                               topic=topic) | ||||||
|  |  | ||||||
|     @exception.wrap_exception |     @exception.wrap_exception | ||||||
|     def receive(self, message_data, message): |     def receive(self, message_data, message): | ||||||
|         _log.debug('received %s' % (message_data)) |         """Magically looks for a method on the proxy object and calls it | ||||||
|  |  | ||||||
|  |         Message data should be a dictionary with two keys: | ||||||
|  |             method: string representing the method to call | ||||||
|  |             args: dictionary of arg: value | ||||||
|  |  | ||||||
|  |         Example: {'method': 'echo', 'args': {'value': 42}} | ||||||
|  |         """ | ||||||
|  |         LOG.debug('received %s' % (message_data)) | ||||||
|         msg_id = message_data.pop('_msg_id', None) |         msg_id = message_data.pop('_msg_id', None) | ||||||
|  |  | ||||||
|         method = message_data.get('method') |         method = message_data.get('method') | ||||||
| @@ -133,21 +171,25 @@ class AdapterConsumer(TopicConsumer): | |||||||
|             #             messages stay in the queue indefinitely, so for now |             #             messages stay in the queue indefinitely, so for now | ||||||
|             #             we just log the message and send an error string |             #             we just log the message and send an error string | ||||||
|             #             back to the caller |             #             back to the caller | ||||||
|             _log.warn('no method for message: %s' % (message_data)) |             LOG.warn('no method for message: %s' % (message_data)) | ||||||
|             msg_reply(msg_id, 'No method for message: %s' % message_data) |             msg_reply(msg_id, 'No method for message: %s' % message_data) | ||||||
|             return |             return | ||||||
|  |  | ||||||
|         node_func = getattr(self.proxy, str(method)) |         node_func = getattr(self.proxy, str(method)) | ||||||
|         node_args = dict((str(k), v) for k, v in args.iteritems()) |         node_args = dict((str(k), v) for k, v in args.iteritems()) | ||||||
|  |         # NOTE(vish): magic is fun! | ||||||
|  |         # pylint: disable-msg=W0142 | ||||||
|         d = defer.maybeDeferred(node_func, **node_args) |         d = defer.maybeDeferred(node_func, **node_args) | ||||||
|         if msg_id: |         if msg_id: | ||||||
|             d.addCallback(lambda rval: msg_reply(msg_id, rval)) |             d.addCallback(lambda rval: msg_reply(msg_id, rval, None)) | ||||||
|             d.addErrback(lambda e: msg_reply(msg_id, str(e))) |             d.addErrback(lambda e: msg_reply(msg_id, None, e)) | ||||||
|         return |         return | ||||||
|  |  | ||||||
|  |  | ||||||
| class TopicPublisher(Publisher): | class TopicPublisher(Publisher): | ||||||
|  |     """Publishes messages on a specific topic""" | ||||||
|     exchange_type = "topic" |     exchange_type = "topic" | ||||||
|  |  | ||||||
|     def __init__(self, connection=None, topic="broadcast"): |     def __init__(self, connection=None, topic="broadcast"): | ||||||
|         self.routing_key = topic |         self.routing_key = topic | ||||||
|         self.exchange = FLAGS.control_exchange |         self.exchange = FLAGS.control_exchange | ||||||
| @@ -156,7 +198,9 @@ class TopicPublisher(Publisher): | |||||||
|  |  | ||||||
|  |  | ||||||
| class DirectConsumer(Consumer): | class DirectConsumer(Consumer): | ||||||
|  |     """Consumes messages directly on a channel specified by msg_id""" | ||||||
|     exchange_type = "direct" |     exchange_type = "direct" | ||||||
|  |  | ||||||
|     def __init__(self, connection=None, msg_id=None): |     def __init__(self, connection=None, msg_id=None): | ||||||
|         self.queue = msg_id |         self.queue = msg_id | ||||||
|         self.routing_key = msg_id |         self.routing_key = msg_id | ||||||
| @@ -166,7 +210,9 @@ class DirectConsumer(Consumer): | |||||||
|  |  | ||||||
|  |  | ||||||
| class DirectPublisher(Publisher): | class DirectPublisher(Publisher): | ||||||
|  |     """Publishes messages directly on a channel specified by msg_id""" | ||||||
|     exchange_type = "direct" |     exchange_type = "direct" | ||||||
|  |  | ||||||
|     def __init__(self, connection=None, msg_id=None): |     def __init__(self, connection=None, msg_id=None): | ||||||
|         self.routing_key = msg_id |         self.routing_key = msg_id | ||||||
|         self.exchange = msg_id |         self.exchange = msg_id | ||||||
| @@ -174,32 +220,63 @@ class DirectPublisher(Publisher): | |||||||
|         super(DirectPublisher, self).__init__(connection=connection) |         super(DirectPublisher, self).__init__(connection=connection) | ||||||
|  |  | ||||||
|  |  | ||||||
| def msg_reply(msg_id, reply): | def msg_reply(msg_id, reply=None, failure=None): | ||||||
|  |     """Sends a reply or an error on the channel signified by msg_id | ||||||
|  |  | ||||||
|  |     failure should be a twisted failure object""" | ||||||
|  |     if failure: | ||||||
|  |         message = failure.getErrorMessage() | ||||||
|  |         traceback = failure.getTraceback() | ||||||
|  |         logging.error("Returning exception %s to caller", message) | ||||||
|  |         logging.error(traceback) | ||||||
|  |         failure = (failure.type.__name__, str(failure.value), traceback) | ||||||
|     conn = Connection.instance() |     conn = Connection.instance() | ||||||
|     publisher = DirectPublisher(connection=conn, msg_id=msg_id) |     publisher = DirectPublisher(connection=conn, msg_id=msg_id) | ||||||
|  |  | ||||||
|     try: |     try: | ||||||
|         publisher.send({'result': reply}) |         publisher.send({'result': reply, 'failure': failure}) | ||||||
|     except TypeError: |     except TypeError: | ||||||
|         publisher.send( |         publisher.send( | ||||||
|                 {'result': dict((k, repr(v)) |                 {'result': dict((k, repr(v)) | ||||||
|                                 for k, v in reply.__dict__.iteritems()) |                                 for k, v in reply.__dict__.iteritems()), | ||||||
|                  }) |                  'failure': failure}) | ||||||
|     publisher.close() |     publisher.close() | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class RemoteError(exception.Error): | ||||||
|  |     """Signifies that a remote class has raised an exception | ||||||
|  |  | ||||||
|  |     Containes a string representation of the type of the original exception, | ||||||
|  |     the value of the original exception, and the traceback.  These are | ||||||
|  |     sent to the parent as a joined string so printing the exception | ||||||
|  |     contains all of the relevent info.""" | ||||||
|  |     def __init__(self, exc_type, value, traceback): | ||||||
|  |         self.exc_type = exc_type | ||||||
|  |         self.value = value | ||||||
|  |         self.traceback = traceback | ||||||
|  |         super(RemoteError, self).__init__("%s %s\n%s" % (exc_type, | ||||||
|  |                                                          value, | ||||||
|  |                                                          traceback)) | ||||||
|  |  | ||||||
|  |  | ||||||
| def call(topic, msg): | def call(topic, msg): | ||||||
|     _log.debug("Making asynchronous call...") |     """Sends a message on a topic and wait for a response""" | ||||||
|  |     LOG.debug("Making asynchronous call...") | ||||||
|     msg_id = uuid.uuid4().hex |     msg_id = uuid.uuid4().hex | ||||||
|     msg.update({'_msg_id': msg_id}) |     msg.update({'_msg_id': msg_id}) | ||||||
|     _log.debug("MSG_ID is %s" % (msg_id)) |     LOG.debug("MSG_ID is %s" % (msg_id)) | ||||||
|  |  | ||||||
|     conn = Connection.instance() |     conn = Connection.instance() | ||||||
|     d = defer.Deferred() |     d = defer.Deferred() | ||||||
|     consumer = DirectConsumer(connection=conn, msg_id=msg_id) |     consumer = DirectConsumer(connection=conn, msg_id=msg_id) | ||||||
|  |  | ||||||
|     def deferred_receive(data, message): |     def deferred_receive(data, message): | ||||||
|  |         """Acks message and callbacks or errbacks""" | ||||||
|         message.ack() |         message.ack() | ||||||
|         d.callback(data) |         if data['failure']: | ||||||
|  |             return d.errback(RemoteError(*data['failure'])) | ||||||
|  |         else: | ||||||
|  |             return d.callback(data['result']) | ||||||
|  |  | ||||||
|     consumer.register_callback(deferred_receive) |     consumer.register_callback(deferred_receive) | ||||||
|     injected = consumer.attach_to_tornado() |     injected = consumer.attach_to_tornado() | ||||||
|  |  | ||||||
| @@ -213,7 +290,8 @@ def call(topic, msg): | |||||||
|  |  | ||||||
|  |  | ||||||
| def cast(topic, msg): | def cast(topic, msg): | ||||||
|     _log.debug("Making asynchronous cast...") |     """Sends a message on a topic without waiting for a response""" | ||||||
|  |     LOG.debug("Making asynchronous cast...") | ||||||
|     conn = Connection.instance() |     conn = Connection.instance() | ||||||
|     publisher = TopicPublisher(connection=conn, topic=topic) |     publisher = TopicPublisher(connection=conn, topic=topic) | ||||||
|     publisher.send(msg) |     publisher.send(msg) | ||||||
| @@ -221,16 +299,18 @@ def cast(topic, msg): | |||||||
|  |  | ||||||
|  |  | ||||||
| def generic_response(message_data, message): | def generic_response(message_data, message): | ||||||
|     _log.debug('response %s', message_data) |     """Logs a result and exits""" | ||||||
|  |     LOG.debug('response %s', message_data) | ||||||
|     message.ack() |     message.ack() | ||||||
|     sys.exit(0) |     sys.exit(0) | ||||||
|  |  | ||||||
|  |  | ||||||
| def send_message(topic, message, wait=True): | def send_message(topic, message, wait=True): | ||||||
|  |     """Sends a message for testing""" | ||||||
|     msg_id = uuid.uuid4().hex |     msg_id = uuid.uuid4().hex | ||||||
|     message.update({'_msg_id': msg_id}) |     message.update({'_msg_id': msg_id}) | ||||||
|     _log.debug('topic is %s', topic) |     LOG.debug('topic is %s', topic) | ||||||
|     _log.debug('message %s', message) |     LOG.debug('message %s', message) | ||||||
|  |  | ||||||
|     if wait: |     if wait: | ||||||
|         consumer = messaging.Consumer(connection=Connection.instance(), |         consumer = messaging.Consumer(connection=Connection.instance(), | ||||||
| @@ -253,6 +333,8 @@ def send_message(topic, message, wait=True): | |||||||
|         consumer.wait() |         consumer.wait() | ||||||
|  |  | ||||||
|  |  | ||||||
| # TODO: Replace with a docstring test |  | ||||||
| if __name__ == "__main__": | if __name__ == "__main__": | ||||||
|  |     # NOTE(vish): you can send messages from the command line using | ||||||
|  |     #             topic and a json sting representing a dictionary | ||||||
|  |     #             for the method | ||||||
|     send_message(sys.argv[1], json.loads(sys.argv[2])) |     send_message(sys.argv[1], json.loads(sys.argv[2])) | ||||||
|   | |||||||
| @@ -52,13 +52,8 @@ def stop(pidfile): | |||||||
|     """ |     """ | ||||||
|     # Get the pid from the pidfile |     # Get the pid from the pidfile | ||||||
|     try: |     try: | ||||||
|         pf = file(pidfile,'r') |         pid = int(open(pidfile,'r').read().strip()) | ||||||
|         pid = int(pf.read().strip()) |  | ||||||
|         pf.close() |  | ||||||
|     except IOError: |     except IOError: | ||||||
|         pid = None |  | ||||||
|  |  | ||||||
|     if not pid: |  | ||||||
|         message = "pidfile %s does not exist. Daemon not running?\n" |         message = "pidfile %s does not exist. Daemon not running?\n" | ||||||
|         sys.stderr.write(message % pidfile) |         sys.stderr.write(message % pidfile) | ||||||
|         return # not an error in a restart |         return # not an error in a restart | ||||||
| @@ -79,6 +74,7 @@ def stop(pidfile): | |||||||
|  |  | ||||||
|  |  | ||||||
| def serve(name, main): | def serve(name, main): | ||||||
|  |     """Controller for server""" | ||||||
|     argv = FLAGS(sys.argv) |     argv = FLAGS(sys.argv) | ||||||
|  |  | ||||||
|     if not FLAGS.pidfile: |     if not FLAGS.pidfile: | ||||||
| @@ -86,7 +82,7 @@ def serve(name, main): | |||||||
|  |  | ||||||
|     logging.debug("Full set of FLAGS: \n\n\n") |     logging.debug("Full set of FLAGS: \n\n\n") | ||||||
|     for flag in FLAGS: |     for flag in FLAGS: | ||||||
|         logging.debug("%s : %s" % (flag, FLAGS.get(flag, None) )) |         logging.debug("%s : %s", flag, FLAGS.get(flag, None)) | ||||||
|  |  | ||||||
|     action = 'start' |     action = 'start' | ||||||
|     if len(argv) > 1: |     if len(argv) > 1: | ||||||
| @@ -102,7 +98,11 @@ def serve(name, main): | |||||||
|     else: |     else: | ||||||
|         print 'usage: %s [options] [start|stop|restart]' % argv[0] |         print 'usage: %s [options] [start|stop|restart]' % argv[0] | ||||||
|         sys.exit(1) |         sys.exit(1) | ||||||
|  |     daemonize(argv, name, main) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def daemonize(args, name, main): | ||||||
|  |     """Does the work of daemonizing the process""" | ||||||
|     logging.getLogger('amqplib').setLevel(logging.WARN) |     logging.getLogger('amqplib').setLevel(logging.WARN) | ||||||
|     if FLAGS.daemonize: |     if FLAGS.daemonize: | ||||||
|         logger = logging.getLogger() |         logger = logging.getLogger() | ||||||
| @@ -115,7 +115,7 @@ def serve(name, main): | |||||||
|         else: |         else: | ||||||
|             if not FLAGS.logfile: |             if not FLAGS.logfile: | ||||||
|                 FLAGS.logfile = '%s.log' % name |                 FLAGS.logfile = '%s.log' % name | ||||||
|             logfile = logging.handlers.FileHandler(FLAGS.logfile) |             logfile = logging.FileHandler(FLAGS.logfile) | ||||||
|             logfile.setFormatter(formatter) |             logfile.setFormatter(formatter) | ||||||
|             logger.addHandler(logfile) |             logger.addHandler(logfile) | ||||||
|         stdin, stdout, stderr = None, None, None |         stdin, stdout, stderr = None, None, None | ||||||
| @@ -137,4 +137,4 @@ def serve(name, main): | |||||||
|             stdout=stdout, |             stdout=stdout, | ||||||
|             stderr=stderr |             stderr=stderr | ||||||
|             ): |             ): | ||||||
|         main(argv) |         main(args) | ||||||
|   | |||||||
| @@ -179,7 +179,21 @@ class AuthTestCase(test.BaseTestCase): | |||||||
|         project.add_role('test1', 'sysadmin') |         project.add_role('test1', 'sysadmin') | ||||||
|         self.assertTrue(project.has_role('test1', 'sysadmin')) |         self.assertTrue(project.has_role('test1', 'sysadmin')) | ||||||
|  |  | ||||||
|     def test_211_can_remove_project_role(self): |     def test_211_can_list_project_roles(self): | ||||||
|  |         project = self.manager.get_project('testproj') | ||||||
|  |         user = self.manager.get_user('test1') | ||||||
|  |         self.manager.add_role(user, 'netadmin', project) | ||||||
|  |         roles = self.manager.get_user_roles(user) | ||||||
|  |         self.assertTrue('sysadmin' in roles) | ||||||
|  |         self.assertFalse('netadmin' in roles) | ||||||
|  |         project_roles = self.manager.get_user_roles(user, project) | ||||||
|  |         self.assertTrue('sysadmin' in project_roles) | ||||||
|  |         self.assertTrue('netadmin' in project_roles) | ||||||
|  |         # has role should be false because global role is missing | ||||||
|  |         self.assertFalse(self.manager.has_role(user, 'netadmin', project)) | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     def test_212_can_remove_project_role(self): | ||||||
|         project = self.manager.get_project('testproj') |         project = self.manager.get_project('testproj') | ||||||
|         self.assertTrue(project.has_role('test1', 'sysadmin')) |         self.assertTrue(project.has_role('test1', 'sysadmin')) | ||||||
|         project.remove_role('test1', 'sysadmin') |         project.remove_role('test1', 'sysadmin') | ||||||
|   | |||||||
| @@ -132,7 +132,7 @@ class CloudTestCase(test.BaseTestCase): | |||||||
|                 'state': 0x01, |                 'state': 0x01, | ||||||
|                 'user_data': '' |                 'user_data': '' | ||||||
|             } |             } | ||||||
|         rv = self.cloud._format_instances(self.context) |         rv = self.cloud._format_describe_instances(self.context) | ||||||
|         self.assert_(len(rv['reservationSet']) == 0) |         self.assert_(len(rv['reservationSet']) == 0) | ||||||
|  |  | ||||||
|         # simulate launch of 5 instances |         # simulate launch of 5 instances | ||||||
|   | |||||||
| @@ -16,25 +16,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. | ||||||
| 
 | 
 | ||||||
| ''' | from nova import flags | ||||||
| Utility methods for working with WSGI servers |  | ||||||
| ''' |  | ||||||
| 
 | 
 | ||||||
| class Util(object): | FLAGS = flags.FLAGS | ||||||
| 
 |  | ||||||
|     @staticmethod |  | ||||||
|     def route(reqstr, controllers): |  | ||||||
|         if len(reqstr) == 0: |  | ||||||
|             return Util.select_root_controller(controllers), [] |  | ||||||
|         parts = [x for x in reqstr.split("/") if len(x) > 0] |  | ||||||
|         if len(parts) == 0: |  | ||||||
|             return Util.select_root_controller(controllers), [] |  | ||||||
|         return controllers[parts[0]], parts[1:] |  | ||||||
| 
 |  | ||||||
|     @staticmethod |  | ||||||
|     def select_root_controller(controllers): |  | ||||||
|         if '' in controllers: |  | ||||||
|             return controllers[''] |  | ||||||
|         else: |  | ||||||
|             return None |  | ||||||
| 
 | 
 | ||||||
|  | flags.DEFINE_integer('answer', 42, 'test flag') | ||||||
							
								
								
									
										87
									
								
								nova/tests/flags_unittest.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										87
									
								
								nova/tests/flags_unittest.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,87 @@ | |||||||
|  | # vim: tabstop=4 shiftwidth=4 softtabstop=4 | ||||||
|  |  | ||||||
|  | # Copyright 2010 United States Government as represented by the | ||||||
|  | # Administrator of the National Aeronautics and Space Administration. | ||||||
|  | # 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 nova import exception | ||||||
|  | from nova import flags | ||||||
|  | from nova import test | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class FlagsTestCase(test.TrialTestCase): | ||||||
|  |     def setUp(self): | ||||||
|  |         super(FlagsTestCase, self).setUp() | ||||||
|  |         self.FLAGS = flags.FlagValues() | ||||||
|  |         self.global_FLAGS = flags.FLAGS | ||||||
|  |  | ||||||
|  |     def test_define(self): | ||||||
|  |         self.assert_('string' not in self.FLAGS) | ||||||
|  |         self.assert_('int' not in self.FLAGS) | ||||||
|  |         self.assert_('false' not in self.FLAGS) | ||||||
|  |         self.assert_('true' not in self.FLAGS) | ||||||
|  |  | ||||||
|  |         flags.DEFINE_string('string', 'default', 'desc', flag_values=self.FLAGS) | ||||||
|  |         flags.DEFINE_integer('int', 1, 'desc', flag_values=self.FLAGS) | ||||||
|  |         flags.DEFINE_bool('false', False, 'desc', flag_values=self.FLAGS) | ||||||
|  |         flags.DEFINE_bool('true', True, 'desc', flag_values=self.FLAGS) | ||||||
|  |  | ||||||
|  |         self.assert_(self.FLAGS['string']) | ||||||
|  |         self.assert_(self.FLAGS['int']) | ||||||
|  |         self.assert_(self.FLAGS['false']) | ||||||
|  |         self.assert_(self.FLAGS['true']) | ||||||
|  |         self.assertEqual(self.FLAGS.string, 'default') | ||||||
|  |         self.assertEqual(self.FLAGS.int, 1) | ||||||
|  |         self.assertEqual(self.FLAGS.false, False) | ||||||
|  |         self.assertEqual(self.FLAGS.true, True) | ||||||
|  |  | ||||||
|  |         argv = ['flags_test', | ||||||
|  |                 '--string', 'foo', | ||||||
|  |                 '--int', '2', | ||||||
|  |                 '--false', | ||||||
|  |                 '--notrue'] | ||||||
|  |  | ||||||
|  |         self.FLAGS(argv) | ||||||
|  |         self.assertEqual(self.FLAGS.string, 'foo') | ||||||
|  |         self.assertEqual(self.FLAGS.int, 2) | ||||||
|  |         self.assertEqual(self.FLAGS.false, True) | ||||||
|  |         self.assertEqual(self.FLAGS.true, False) | ||||||
|  |  | ||||||
|  |     def test_declare(self): | ||||||
|  |         self.assert_('answer' not in self.global_FLAGS) | ||||||
|  |         flags.DECLARE('answer', 'nova.tests.declare_flags') | ||||||
|  |         self.assert_('answer' in self.global_FLAGS) | ||||||
|  |         self.assertEqual(self.global_FLAGS.answer, 42) | ||||||
|  |  | ||||||
|  |         # Make sure we don't overwrite anything | ||||||
|  |         self.global_FLAGS.answer = 256 | ||||||
|  |         self.assertEqual(self.global_FLAGS.answer, 256) | ||||||
|  |         flags.DECLARE('answer', 'nova.tests.declare_flags') | ||||||
|  |         self.assertEqual(self.global_FLAGS.answer, 256) | ||||||
|  |  | ||||||
|  |     def test_runtime_and_unknown_flags(self): | ||||||
|  |         self.assert_('runtime_answer' not in self.global_FLAGS) | ||||||
|  |  | ||||||
|  |         argv = ['flags_test', '--runtime_answer=60', 'extra_arg'] | ||||||
|  |         args = self.global_FLAGS(argv) | ||||||
|  |         self.assertEqual(len(args), 2) | ||||||
|  |         self.assertEqual(args[1], 'extra_arg') | ||||||
|  |  | ||||||
|  |         self.assert_('runtime_answer' not in self.global_FLAGS) | ||||||
|  |  | ||||||
|  |         import nova.tests.runtime_flags | ||||||
|  |  | ||||||
|  |         self.assert_('runtime_answer' in self.global_FLAGS) | ||||||
|  |         self.assertEqual(self.global_FLAGS.runtime_answer, 60) | ||||||
| @@ -15,7 +15,9 @@ | |||||||
| #    WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the | #    WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the | ||||||
| #    License for the specific language governing permissions and limitations | #    License for the specific language governing permissions and limitations | ||||||
| #    under the License. | #    under the License. | ||||||
|  | """ | ||||||
|  | Unit Tests for network code | ||||||
|  | """ | ||||||
| import IPy | import IPy | ||||||
| import os | import os | ||||||
| import logging | import logging | ||||||
| @@ -31,8 +33,10 @@ from nova.network.exception import NoMoreAddresses | |||||||
|  |  | ||||||
| FLAGS = flags.FLAGS | FLAGS = flags.FLAGS | ||||||
|  |  | ||||||
|  |  | ||||||
| class NetworkTestCase(test.TrialTestCase): | class NetworkTestCase(test.TrialTestCase): | ||||||
|     def setUp(self): |     """Test cases for network code""" | ||||||
|  |     def setUp(self):  # pylint: disable-msg=C0103 | ||||||
|         super(NetworkTestCase, self).setUp() |         super(NetworkTestCase, self).setUp() | ||||||
|         # NOTE(vish): if you change these flags, make sure to change the |         # NOTE(vish): if you change these flags, make sure to change the | ||||||
|         #             flags in the corresponding section in nova-dhcpbridge |         #             flags in the corresponding section in nova-dhcpbridge | ||||||
| @@ -43,7 +47,6 @@ class NetworkTestCase(test.TrialTestCase): | |||||||
|                    network_size=32) |                    network_size=32) | ||||||
|         logging.getLogger().setLevel(logging.DEBUG) |         logging.getLogger().setLevel(logging.DEBUG) | ||||||
|         self.manager = manager.AuthManager() |         self.manager = manager.AuthManager() | ||||||
|         self.dnsmasq = FakeDNSMasq() |  | ||||||
|         self.user = self.manager.create_user('netuser', 'netuser', 'netuser') |         self.user = self.manager.create_user('netuser', 'netuser', 'netuser') | ||||||
|         self.projects = [] |         self.projects = [] | ||||||
|         self.projects.append(self.manager.create_project('netuser', |         self.projects.append(self.manager.create_project('netuser', | ||||||
| @@ -54,47 +57,49 @@ class NetworkTestCase(test.TrialTestCase): | |||||||
|             self.projects.append(self.manager.create_project(name, |             self.projects.append(self.manager.create_project(name, | ||||||
|                                                              'netuser', |                                                              'netuser', | ||||||
|                                                              name)) |                                                              name)) | ||||||
|         self.network = model.PublicNetworkController() |             vpn.NetworkData.create(self.projects[i].id) | ||||||
|         self.service = service.VlanNetworkService() |         self.service = service.VlanNetworkService() | ||||||
|  |  | ||||||
|     def tearDown(self): |     def tearDown(self):  # pylint: disable-msg=C0103 | ||||||
|         super(NetworkTestCase, self).tearDown() |         super(NetworkTestCase, self).tearDown() | ||||||
|         for project in self.projects: |         for project in self.projects: | ||||||
|             self.manager.delete_project(project) |             self.manager.delete_project(project) | ||||||
|         self.manager.delete_user(self.user) |         self.manager.delete_user(self.user) | ||||||
|  |  | ||||||
|     def test_public_network_allocation(self): |     def test_public_network_allocation(self): | ||||||
|  |         """Makes sure that we can allocaate a public ip""" | ||||||
|         pubnet = IPy.IP(flags.FLAGS.public_range) |         pubnet = IPy.IP(flags.FLAGS.public_range) | ||||||
|         address = self.network.allocate_ip(self.user.id, self.projects[0].id, "public") |         address = self.service.allocate_elastic_ip(self.user.id, | ||||||
|  |                                            self.projects[0].id) | ||||||
|         self.assertTrue(IPy.IP(address) in pubnet) |         self.assertTrue(IPy.IP(address) in pubnet) | ||||||
|         self.assertTrue(IPy.IP(address) in self.network.network) |  | ||||||
|  |  | ||||||
|     def test_allocate_deallocate_fixed_ip(self): |     def test_allocate_deallocate_fixed_ip(self): | ||||||
|         result  = yield self.service.allocate_fixed_ip( |         """Makes sure that we can allocate and deallocate a fixed ip""" | ||||||
|  |         result = self.service.allocate_fixed_ip( | ||||||
|                 self.user.id, self.projects[0].id) |                 self.user.id, self.projects[0].id) | ||||||
|         address = result['private_dns_name'] |         address = result['private_dns_name'] | ||||||
|         mac = result['mac_address'] |         mac = result['mac_address'] | ||||||
|         logging.debug("Was allocated %s" % (address)) |  | ||||||
|         net = model.get_project_network(self.projects[0].id, "default") |         net = model.get_project_network(self.projects[0].id, "default") | ||||||
|         self.assertEqual(True, is_in_project(address, self.projects[0].id)) |         self.assertEqual(True, is_in_project(address, self.projects[0].id)) | ||||||
|         hostname = "test-host" |         hostname = "test-host" | ||||||
|         self.dnsmasq.issue_ip(mac, address, hostname, net.bridge_name) |         issue_ip(mac, address, hostname, net.bridge_name) | ||||||
|         rv = self.service.deallocate_fixed_ip(address) |         self.service.deallocate_fixed_ip(address) | ||||||
|  |  | ||||||
|         # Doesn't go away until it's dhcp released |         # Doesn't go away until it's dhcp released | ||||||
|         self.assertEqual(True, is_in_project(address, self.projects[0].id)) |         self.assertEqual(True, is_in_project(address, self.projects[0].id)) | ||||||
|  |  | ||||||
|         self.dnsmasq.release_ip(mac, address, hostname, net.bridge_name) |         release_ip(mac, address, hostname, net.bridge_name) | ||||||
|         self.assertEqual(False, is_in_project(address, self.projects[0].id)) |         self.assertEqual(False, is_in_project(address, self.projects[0].id)) | ||||||
|  |  | ||||||
|     def test_range_allocation(self): |     def test_side_effects(self): | ||||||
|         hostname = "test-host" |         """Ensures allocating and releasing has no side effects""" | ||||||
|         result = yield self.service.allocate_fixed_ip( |         hostname = "side-effect-host" | ||||||
|                     self.user.id, self.projects[0].id) |         result = self.service.allocate_fixed_ip(self.user.id, | ||||||
|  |                                                 self.projects[0].id) | ||||||
|         mac = result['mac_address'] |         mac = result['mac_address'] | ||||||
|         address = result['private_dns_name'] |         address = result['private_dns_name'] | ||||||
|         result = yield self.service.allocate_fixed_ip( |         result = self.service.allocate_fixed_ip(self.user, | ||||||
|                 self.user, self.projects[1].id) |                                                 self.projects[1].id) | ||||||
|         secondmac = result['mac_address'] |         secondmac = result['mac_address'] | ||||||
|         secondaddress = result['private_dns_name'] |         secondaddress = result['private_dns_name'] | ||||||
|  |  | ||||||
| @@ -102,66 +107,75 @@ class NetworkTestCase(test.TrialTestCase): | |||||||
|         secondnet = model.get_project_network(self.projects[1].id, "default") |         secondnet = model.get_project_network(self.projects[1].id, "default") | ||||||
|  |  | ||||||
|         self.assertEqual(True, is_in_project(address, self.projects[0].id)) |         self.assertEqual(True, is_in_project(address, self.projects[0].id)) | ||||||
|         self.assertEqual(True, is_in_project(secondaddress, self.projects[1].id)) |         self.assertEqual(True, is_in_project(secondaddress, | ||||||
|  |                                              self.projects[1].id)) | ||||||
|         self.assertEqual(False, is_in_project(address, self.projects[1].id)) |         self.assertEqual(False, is_in_project(address, self.projects[1].id)) | ||||||
|  |  | ||||||
|         # Addresses are allocated before they're issued |         # Addresses are allocated before they're issued | ||||||
|         self.dnsmasq.issue_ip(mac, address, hostname, net.bridge_name) |         issue_ip(mac, address, hostname, net.bridge_name) | ||||||
|         self.dnsmasq.issue_ip(secondmac, secondaddress, |         issue_ip(secondmac, secondaddress, hostname, secondnet.bridge_name) | ||||||
|                                 hostname, secondnet.bridge_name) |  | ||||||
|  |  | ||||||
|         rv = self.service.deallocate_fixed_ip(address) |         self.service.deallocate_fixed_ip(address) | ||||||
|         self.dnsmasq.release_ip(mac, address, hostname, net.bridge_name) |         release_ip(mac, address, hostname, net.bridge_name) | ||||||
|         self.assertEqual(False, is_in_project(address, self.projects[0].id)) |         self.assertEqual(False, is_in_project(address, self.projects[0].id)) | ||||||
|  |  | ||||||
|         # First address release shouldn't affect the second |         # First address release shouldn't affect the second | ||||||
|         self.assertEqual(True, is_in_project(secondaddress, self.projects[1].id)) |         self.assertEqual(True, is_in_project(secondaddress, | ||||||
|  |                                              self.projects[1].id)) | ||||||
|  |  | ||||||
|         rv = self.service.deallocate_fixed_ip(secondaddress) |         self.service.deallocate_fixed_ip(secondaddress) | ||||||
|         self.dnsmasq.release_ip(secondmac, secondaddress, |         release_ip(secondmac, secondaddress, hostname, secondnet.bridge_name) | ||||||
|                                 hostname, secondnet.bridge_name) |         self.assertEqual(False, is_in_project(secondaddress, | ||||||
|         self.assertEqual(False, is_in_project(secondaddress, self.projects[1].id)) |                                               self.projects[1].id)) | ||||||
|  |  | ||||||
|     def test_subnet_edge(self): |     def test_subnet_edge(self): | ||||||
|         result = yield self.service.allocate_fixed_ip(self.user.id, |         """Makes sure that private ips don't overlap""" | ||||||
|  |         result = self.service.allocate_fixed_ip(self.user.id, | ||||||
|                                                        self.projects[0].id) |                                                        self.projects[0].id) | ||||||
|         firstaddress = result['private_dns_name'] |         firstaddress = result['private_dns_name'] | ||||||
|         hostname = "toomany-hosts" |         hostname = "toomany-hosts" | ||||||
|         for i in range(1, 5): |         for i in range(1, 5): | ||||||
|             project_id = self.projects[i].id |             project_id = self.projects[i].id | ||||||
|             result = yield self.service.allocate_fixed_ip( |             result = self.service.allocate_fixed_ip( | ||||||
|                     self.user, project_id) |                     self.user, project_id) | ||||||
|             mac = result['mac_address'] |             mac = result['mac_address'] | ||||||
|             address = result['private_dns_name'] |             address = result['private_dns_name'] | ||||||
|             result = yield self.service.allocate_fixed_ip( |             result = self.service.allocate_fixed_ip( | ||||||
|                     self.user, project_id) |                     self.user, project_id) | ||||||
|             mac2 = result['mac_address'] |             mac2 = result['mac_address'] | ||||||
|             address2 = result['private_dns_name'] |             address2 = result['private_dns_name'] | ||||||
|             result = yield self.service.allocate_fixed_ip( |             result = self.service.allocate_fixed_ip( | ||||||
|                    self.user, project_id) |                    self.user, project_id) | ||||||
|             mac3 = result['mac_address'] |             mac3 = result['mac_address'] | ||||||
|             address3 = result['private_dns_name'] |             address3 = result['private_dns_name'] | ||||||
|             self.assertEqual(False, is_in_project(address, self.projects[0].id)) |  | ||||||
|             self.assertEqual(False, is_in_project(address2, self.projects[0].id)) |  | ||||||
|             self.assertEqual(False, is_in_project(address3, self.projects[0].id)) |  | ||||||
|             rv = self.service.deallocate_fixed_ip(address) |  | ||||||
|             rv = self.service.deallocate_fixed_ip(address2) |  | ||||||
|             rv = self.service.deallocate_fixed_ip(address3) |  | ||||||
|             net = model.get_project_network(project_id, "default") |             net = model.get_project_network(project_id, "default") | ||||||
|             self.dnsmasq.release_ip(mac, address, hostname, net.bridge_name) |             issue_ip(mac, address, hostname, net.bridge_name) | ||||||
|             self.dnsmasq.release_ip(mac2, address2, hostname, net.bridge_name) |             issue_ip(mac2, address2, hostname, net.bridge_name) | ||||||
|             self.dnsmasq.release_ip(mac3, address3, hostname, net.bridge_name) |             issue_ip(mac3, address3, hostname, net.bridge_name) | ||||||
|  |             self.assertEqual(False, is_in_project(address, | ||||||
|  |                                                   self.projects[0].id)) | ||||||
|  |             self.assertEqual(False, is_in_project(address2, | ||||||
|  |                                                   self.projects[0].id)) | ||||||
|  |             self.assertEqual(False, is_in_project(address3, | ||||||
|  |                                                   self.projects[0].id)) | ||||||
|  |             self.service.deallocate_fixed_ip(address) | ||||||
|  |             self.service.deallocate_fixed_ip(address2) | ||||||
|  |             self.service.deallocate_fixed_ip(address3) | ||||||
|  |             release_ip(mac, address, hostname, net.bridge_name) | ||||||
|  |             release_ip(mac2, address2, hostname, net.bridge_name) | ||||||
|  |             release_ip(mac3, address3, hostname, net.bridge_name) | ||||||
|         net = model.get_project_network(self.projects[0].id, "default") |         net = model.get_project_network(self.projects[0].id, "default") | ||||||
|         rv = self.service.deallocate_fixed_ip(firstaddress) |         self.service.deallocate_fixed_ip(firstaddress) | ||||||
|         self.dnsmasq.release_ip(mac, firstaddress, hostname, net.bridge_name) |         release_ip(mac, firstaddress, hostname, net.bridge_name) | ||||||
|  |  | ||||||
|     def test_212_vpn_ip_and_port_looks_valid(self): |     def test_vpn_ip_and_port_looks_valid(self): | ||||||
|         vpn.NetworkData.create(self.projects[0].id) |         """Ensure the vpn ip and port are reasonable""" | ||||||
|         self.assert_(self.projects[0].vpn_ip) |         self.assert_(self.projects[0].vpn_ip) | ||||||
|         self.assert_(self.projects[0].vpn_port >= FLAGS.vpn_start_port) |         self.assert_(self.projects[0].vpn_port >= FLAGS.vpn_start_port) | ||||||
|         self.assert_(self.projects[0].vpn_port <= FLAGS.vpn_end_port) |         self.assert_(self.projects[0].vpn_port <= FLAGS.vpn_end_port) | ||||||
|  |  | ||||||
|     def test_too_many_vpns(self): |     def test_too_many_vpns(self): | ||||||
|  |         """Ensure error is raised if we run out of vpn ports""" | ||||||
|         vpns = [] |         vpns = [] | ||||||
|         for i in xrange(vpn.NetworkData.num_ports_for_ip(FLAGS.vpn_ip)): |         for i in xrange(vpn.NetworkData.num_ports_for_ip(FLAGS.vpn_ip)): | ||||||
|             vpns.append(vpn.NetworkData.create("vpnuser%s" % i)) |             vpns.append(vpn.NetworkData.create("vpnuser%s" % i)) | ||||||
| @@ -169,84 +183,102 @@ class NetworkTestCase(test.TrialTestCase): | |||||||
|         for network_datum in vpns: |         for network_datum in vpns: | ||||||
|             network_datum.destroy() |             network_datum.destroy() | ||||||
|  |  | ||||||
|     def test_release_before_deallocate(self): |     def test_ips_are_reused(self): | ||||||
|         pass |         """Makes sure that ip addresses that are deallocated get reused""" | ||||||
|  |         result = self.service.allocate_fixed_ip( | ||||||
|  |                     self.user.id, self.projects[0].id) | ||||||
|  |         mac = result['mac_address'] | ||||||
|  |         address = result['private_dns_name'] | ||||||
|  |  | ||||||
|     def test_deallocate_before_issued(self): |         hostname = "reuse-host" | ||||||
|         pass |         net = model.get_project_network(self.projects[0].id, "default") | ||||||
|  |  | ||||||
|     def test_too_many_addresses(self): |         issue_ip(mac, address, hostname, net.bridge_name) | ||||||
|         """ |         self.service.deallocate_fixed_ip(address) | ||||||
|         Here, we test that a proper NoMoreAddresses exception is raised. |         release_ip(mac, address, hostname, net.bridge_name) | ||||||
|  |  | ||||||
|         However, the number of available IP addresses depends on the test |         result = self.service.allocate_fixed_ip( | ||||||
|  |                 self.user, self.projects[0].id) | ||||||
|  |         secondmac = result['mac_address'] | ||||||
|  |         secondaddress = result['private_dns_name'] | ||||||
|  |         self.assertEqual(address, secondaddress) | ||||||
|  |         issue_ip(secondmac, secondaddress, hostname, net.bridge_name) | ||||||
|  |         self.service.deallocate_fixed_ip(secondaddress) | ||||||
|  |         release_ip(secondmac, secondaddress, hostname, net.bridge_name) | ||||||
|  |  | ||||||
|  |     def test_available_ips(self): | ||||||
|  |         """Make sure the number of available ips for the network is correct | ||||||
|  |  | ||||||
|  |         The number of available IP addresses depends on the test | ||||||
|         environment's setup. |         environment's setup. | ||||||
|  |  | ||||||
|         Network size is set in test fixture's setUp method. |         Network size is set in test fixture's setUp method. | ||||||
|  |  | ||||||
|         There are FLAGS.cnt_vpn_clients addresses reserved for VPN (NUM_RESERVED_VPN_IPS) |         There are ips reserved at the bottom and top of the range. | ||||||
|  |         services (network, gateway, CloudPipe, broadcast) | ||||||
|         And there are NUM_STATIC_IPS that are always reserved by Nova for the necessary |  | ||||||
|         services (gateway, CloudPipe, etc) |  | ||||||
|  |  | ||||||
|         So we should get flags.network_size - (NUM_STATIC_IPS + |  | ||||||
|                                                NUM_PREALLOCATED_IPS + |  | ||||||
|                                                NUM_RESERVED_VPN_IPS) |  | ||||||
|         usable addresses |  | ||||||
|         """ |         """ | ||||||
|         net = model.get_project_network(self.projects[0].id, "default") |         net = model.get_project_network(self.projects[0].id, "default") | ||||||
|  |         num_preallocated_ips = len(net.assigned) | ||||||
|         # Determine expected number of available IP addresses |         net_size = flags.FLAGS.network_size | ||||||
|         num_static_ips = net.num_static_ips |         num_available_ips = net_size - (net.num_bottom_reserved_ips + | ||||||
|         num_preallocated_ips = len(net.hosts.keys()) |  | ||||||
|         num_reserved_vpn_ips = flags.FLAGS.cnt_vpn_clients |  | ||||||
|         num_available_ips = flags.FLAGS.network_size - (num_static_ips + |  | ||||||
|                                         num_preallocated_ips + |                                         num_preallocated_ips + | ||||||
|                                                         num_reserved_vpn_ips) |                                         net.num_top_reserved_ips) | ||||||
|  |         self.assertEqual(num_available_ips, len(list(net.available))) | ||||||
|  |  | ||||||
|  |     def test_too_many_addresses(self): | ||||||
|  |         """Test for a NoMoreAddresses exception when all fixed ips are used. | ||||||
|  |         """ | ||||||
|  |         net = model.get_project_network(self.projects[0].id, "default") | ||||||
|  |  | ||||||
|         hostname = "toomany-hosts" |         hostname = "toomany-hosts" | ||||||
|         macs = {} |         macs = {} | ||||||
|         addresses = {} |         addresses = {} | ||||||
|         for i in range(0, (num_available_ips - 1)): |         # Number of availaible ips is len of the available list | ||||||
|             result = yield self.service.allocate_fixed_ip(self.user.id, self.projects[0].id) |         num_available_ips = len(list(net.available)) | ||||||
|  |         for i in range(num_available_ips): | ||||||
|  |             result = self.service.allocate_fixed_ip(self.user.id, | ||||||
|  |                                                     self.projects[0].id) | ||||||
|             macs[i] = result['mac_address'] |             macs[i] = result['mac_address'] | ||||||
|             addresses[i] = result['private_dns_name'] |             addresses[i] = result['private_dns_name'] | ||||||
|             self.dnsmasq.issue_ip(macs[i], addresses[i], hostname, net.bridge_name) |             issue_ip(macs[i], addresses[i], hostname, net.bridge_name) | ||||||
|  |  | ||||||
|         self.assertFailure(self.service.allocate_fixed_ip(self.user.id, self.projects[0].id), NoMoreAddresses) |         self.assertEqual(len(list(net.available)), 0) | ||||||
|  |         self.assertRaises(NoMoreAddresses, self.service.allocate_fixed_ip, | ||||||
|  |                           self.user.id, self.projects[0].id) | ||||||
|  |  | ||||||
|  |         for i in range(len(addresses)): | ||||||
|  |             self.service.deallocate_fixed_ip(addresses[i]) | ||||||
|  |             release_ip(macs[i], addresses[i], hostname, net.bridge_name) | ||||||
|  |         self.assertEqual(len(list(net.available)), num_available_ips) | ||||||
|  |  | ||||||
|         for i in range(0, (num_available_ips - 1)): |  | ||||||
|             rv = self.service.deallocate_fixed_ip(addresses[i]) |  | ||||||
|             self.dnsmasq.release_ip(macs[i], addresses[i], hostname, net.bridge_name) |  | ||||||
|  |  | ||||||
| def is_in_project(address, project_id): | def is_in_project(address, project_id): | ||||||
|     return address in model.get_project_network(project_id).list_addresses() |     """Returns true if address is in specified project""" | ||||||
|  |     return address in model.get_project_network(project_id).assigned | ||||||
|  |  | ||||||
| def _get_project_addresses(project_id): |  | ||||||
|     project_addresses = [] |  | ||||||
|     for addr in model.get_project_network(project_id).list_addresses(): |  | ||||||
|         project_addresses.append(addr) |  | ||||||
|     return project_addresses |  | ||||||
|  |  | ||||||
| def binpath(script): | def binpath(script): | ||||||
|  |     """Returns the absolute path to a script in bin""" | ||||||
|     return os.path.abspath(os.path.join(__file__, "../../../bin", script)) |     return os.path.abspath(os.path.join(__file__, "../../../bin", script)) | ||||||
|  |  | ||||||
| class FakeDNSMasq(object): |  | ||||||
|     def issue_ip(self, mac, ip, hostname, interface): | def issue_ip(mac, private_ip, hostname, interface): | ||||||
|  |     """Run add command on dhcpbridge""" | ||||||
|     cmd = "%s add %s %s %s" % (binpath('nova-dhcpbridge'), |     cmd = "%s add %s %s %s" % (binpath('nova-dhcpbridge'), | ||||||
|                                    mac, ip, hostname) |                                mac, private_ip, hostname) | ||||||
|     env = {'DNSMASQ_INTERFACE': interface, |     env = {'DNSMASQ_INTERFACE': interface, | ||||||
|            'TESTING': '1', |            'TESTING': '1', | ||||||
|            'FLAGFILE': FLAGS.dhcpbridge_flagfile} |            'FLAGFILE': FLAGS.dhcpbridge_flagfile} | ||||||
|     (out, err) = utils.execute(cmd, addl_env=env) |     (out, err) = utils.execute(cmd, addl_env=env) | ||||||
|         logging.debug("ISSUE_IP: %s, %s " % (out, err)) |     logging.debug("ISSUE_IP: %s, %s ", out, err) | ||||||
|  |  | ||||||
|     def release_ip(self, mac, ip, hostname, interface): |  | ||||||
|  | def release_ip(mac, private_ip, hostname, interface): | ||||||
|  |     """Run del command on dhcpbridge""" | ||||||
|     cmd = "%s del %s %s %s" % (binpath('nova-dhcpbridge'), |     cmd = "%s del %s %s %s" % (binpath('nova-dhcpbridge'), | ||||||
|                                    mac, ip, hostname) |                                mac, private_ip, hostname) | ||||||
|     env = {'DNSMASQ_INTERFACE': interface, |     env = {'DNSMASQ_INTERFACE': interface, | ||||||
|            'TESTING': '1', |            'TESTING': '1', | ||||||
|            'FLAGFILE': FLAGS.dhcpbridge_flagfile} |            'FLAGFILE': FLAGS.dhcpbridge_flagfile} | ||||||
|     (out, err) = utils.execute(cmd, addl_env=env) |     (out, err) = utils.execute(cmd, addl_env=env) | ||||||
|         logging.debug("RELEASE_IP: %s, %s " % (out, err)) |     logging.debug("RELEASE_IP: %s, %s ", out, err) | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										85
									
								
								nova/tests/rpc_unittest.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										85
									
								
								nova/tests/rpc_unittest.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,85 @@ | |||||||
|  | # vim: tabstop=4 shiftwidth=4 softtabstop=4 | ||||||
|  |  | ||||||
|  | # Copyright 2010 United States Government as represented by the | ||||||
|  | # Administrator of the National Aeronautics and Space Administration. | ||||||
|  | # 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. | ||||||
|  | """ | ||||||
|  | Unit Tests for remote procedure calls using queue | ||||||
|  | """ | ||||||
|  | import logging | ||||||
|  |  | ||||||
|  | from twisted.internet import defer | ||||||
|  |  | ||||||
|  | from nova import flags | ||||||
|  | from nova import rpc | ||||||
|  | from nova import test | ||||||
|  |  | ||||||
|  |  | ||||||
|  | FLAGS = flags.FLAGS | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class RpcTestCase(test.BaseTestCase): | ||||||
|  |     """Test cases for rpc""" | ||||||
|  |     def setUp(self):  # pylint: disable-msg=C0103 | ||||||
|  |         super(RpcTestCase, self).setUp() | ||||||
|  |         self.conn = rpc.Connection.instance() | ||||||
|  |         self.receiver = TestReceiver() | ||||||
|  |         self.consumer = rpc.AdapterConsumer(connection=self.conn, | ||||||
|  |                                             topic='test', | ||||||
|  |                                             proxy=self.receiver) | ||||||
|  |  | ||||||
|  |         self.injected.append(self.consumer.attach_to_tornado(self.ioloop)) | ||||||
|  |  | ||||||
|  |     def test_call_succeed(self): | ||||||
|  |         """Get a value through rpc call""" | ||||||
|  |         value = 42 | ||||||
|  |         result = yield rpc.call('test', {"method": "echo", | ||||||
|  |                                          "args": {"value": value}}) | ||||||
|  |         self.assertEqual(value, result) | ||||||
|  |  | ||||||
|  |     def test_call_exception(self): | ||||||
|  |         """Test that exception gets passed back properly | ||||||
|  |  | ||||||
|  |         rpc.call returns a RemoteError object.  The value of the | ||||||
|  |         exception is converted to a string, so we convert it back | ||||||
|  |         to an int in the test. | ||||||
|  |         """ | ||||||
|  |         value = 42 | ||||||
|  |         self.assertFailure(rpc.call('test', {"method": "fail", | ||||||
|  |                                              "args": {"value": value}}), | ||||||
|  |                            rpc.RemoteError) | ||||||
|  |         try: | ||||||
|  |             yield rpc.call('test', {"method": "fail", | ||||||
|  |                                     "args": {"value": value}}) | ||||||
|  |             self.fail("should have thrown rpc.RemoteError") | ||||||
|  |         except rpc.RemoteError as exc: | ||||||
|  |             self.assertEqual(int(exc.value), value) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class TestReceiver(object): | ||||||
|  |     """Simple Proxy class so the consumer has methods to call | ||||||
|  |  | ||||||
|  |     Uses static methods because we aren't actually storing any state""" | ||||||
|  |  | ||||||
|  |     @staticmethod | ||||||
|  |     def echo(value): | ||||||
|  |         """Simply returns whatever value is sent in""" | ||||||
|  |         logging.debug("Received %s", value) | ||||||
|  |         return defer.succeed(value) | ||||||
|  |  | ||||||
|  |     @staticmethod | ||||||
|  |     def fail(value): | ||||||
|  |         """Raises an exception with the value sent in""" | ||||||
|  |         raise Exception(value) | ||||||
| @@ -16,36 +16,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 cloudservers | from nova import flags | ||||||
| 
 | 
 | ||||||
| class IdFake: | FLAGS = flags.FLAGS | ||||||
|     def __init__(self, id): |  | ||||||
|         self.id = id |  | ||||||
| 
 | 
 | ||||||
| # to get your access key: | flags.DEFINE_integer('runtime_answer', 54, 'test flag') | ||||||
| # from nova.auth import users |  | ||||||
| # users.UserManger.instance().get_users()[0].access |  | ||||||
| rscloud = cloudservers.CloudServers( |  | ||||||
|             'admin', |  | ||||||
|             '6cca875e-5ab3-4c60-9852-abf5c5c60cc6' |  | ||||||
|           ) |  | ||||||
| rscloud.client.AUTH_URL = 'http://localhost:8773/v1.0' |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| rv = rscloud.servers.list() |  | ||||||
| print "SERVERS: %s" % rv |  | ||||||
| 
 |  | ||||||
| if len(rv) == 0: |  | ||||||
|     server = rscloud.servers.create( |  | ||||||
|                "test-server", |  | ||||||
|                IdFake("ami-tiny"), |  | ||||||
|                IdFake("m1.tiny") |  | ||||||
|              ) |  | ||||||
|     print "LAUNCH: %s" % server |  | ||||||
| else: |  | ||||||
|     server = rv[0] |  | ||||||
|     print "Server to kill: %s" % server |  | ||||||
| 
 |  | ||||||
| raw_input("press enter key to kill the server") |  | ||||||
| 
 |  | ||||||
| server.delete() |  | ||||||
							
								
								
									
										69
									
								
								nova/tests/virt_unittest.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										69
									
								
								nova/tests/virt_unittest.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,69 @@ | |||||||
|  | # vim: tabstop=4 shiftwidth=4 softtabstop=4 | ||||||
|  | # | ||||||
|  | #    Copyright 2010 OpenStack LLC | ||||||
|  | # | ||||||
|  | #    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 nova import flags | ||||||
|  | from nova import test | ||||||
|  | from nova.virt import libvirt_conn | ||||||
|  |  | ||||||
|  | FLAGS = flags.FLAGS | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class LibvirtConnTestCase(test.TrialTestCase): | ||||||
|  |     def test_get_uri_and_template(self): | ||||||
|  |         class MockDataModel(object): | ||||||
|  |             def __init__(self): | ||||||
|  |                 self.datamodel = { 'name' : 'i-cafebabe', | ||||||
|  |                                    'memory_kb' : '1024000', | ||||||
|  |                                    'basepath' : '/some/path', | ||||||
|  |                                    'bridge_name' : 'br100', | ||||||
|  |                                    'mac_address' : '02:12:34:46:56:67', | ||||||
|  |                                    'vcpus' : 2 } | ||||||
|  |  | ||||||
|  |         type_uri_map = { 'qemu' : ('qemu:///system', | ||||||
|  |                                 [lambda s: '<domain type=\'qemu\'>' in s, | ||||||
|  |                                  lambda s: 'type>hvm</type' in s, | ||||||
|  |                                  lambda s: 'emulator>/usr/bin/kvm' not in s]), | ||||||
|  |                          'kvm' : ('qemu:///system', | ||||||
|  |                                 [lambda s: '<domain type=\'kvm\'>' in s, | ||||||
|  |                                  lambda s: 'type>hvm</type' in s, | ||||||
|  |                                  lambda s: 'emulator>/usr/bin/qemu<' not in s]), | ||||||
|  |                          'uml' : ('uml:///system', | ||||||
|  |                                 [lambda s: '<domain type=\'uml\'>' in s, | ||||||
|  |                                  lambda s: 'type>uml</type' in s]), | ||||||
|  |                           } | ||||||
|  |  | ||||||
|  |         for (libvirt_type,(expected_uri, checks)) in type_uri_map.iteritems(): | ||||||
|  |             FLAGS.libvirt_type = libvirt_type | ||||||
|  |             conn = libvirt_conn.LibvirtConnection(True) | ||||||
|  |  | ||||||
|  |             uri, template = conn.get_uri_and_template() | ||||||
|  |             self.assertEquals(uri, expected_uri) | ||||||
|  |  | ||||||
|  |             for i, check in enumerate(checks): | ||||||
|  |                 xml = conn.toXml(MockDataModel()) | ||||||
|  |                 self.assertTrue(check(xml), '%s failed check %d' % (xml, i)) | ||||||
|  |  | ||||||
|  |         # Deliberately not just assigning this string to FLAGS.libvirt_uri and | ||||||
|  |         # checking against that later on. This way we make sure the | ||||||
|  |         # implementation doesn't fiddle around with the FLAGS. | ||||||
|  |         testuri = 'something completely different' | ||||||
|  |         FLAGS.libvirt_uri = testuri | ||||||
|  |         for (libvirt_type,(expected_uri, checks)) in type_uri_map.iteritems(): | ||||||
|  |             FLAGS.libvirt_type = libvirt_type | ||||||
|  |             conn = libvirt_conn.LibvirtConnection(True) | ||||||
|  |             uri, template = conn.get_uri_and_template() | ||||||
|  |             self.assertEquals(uri, testuri) | ||||||
|  |  | ||||||
| @@ -17,6 +17,10 @@ | |||||||
| #    under the License. | #    under the License. | ||||||
|  |  | ||||||
| import logging | import logging | ||||||
|  | import shutil | ||||||
|  | import tempfile | ||||||
|  |  | ||||||
|  | from twisted.internet import defer | ||||||
|  |  | ||||||
| from nova import compute | from nova import compute | ||||||
| from nova import exception | from nova import exception | ||||||
| @@ -34,10 +38,16 @@ class VolumeTestCase(test.TrialTestCase): | |||||||
|         super(VolumeTestCase, self).setUp() |         super(VolumeTestCase, self).setUp() | ||||||
|         self.compute = compute.service.ComputeService() |         self.compute = compute.service.ComputeService() | ||||||
|         self.volume = None |         self.volume = None | ||||||
|  |         self.tempdir = tempfile.mkdtemp() | ||||||
|         self.flags(connection_type='fake', |         self.flags(connection_type='fake', | ||||||
|                    fake_storage=True) |                    fake_storage=True, | ||||||
|  |                    aoe_export_dir=self.tempdir) | ||||||
|         self.volume = volume_service.VolumeService() |         self.volume = volume_service.VolumeService() | ||||||
|  |  | ||||||
|  |     def tearDown(self): | ||||||
|  |         shutil.rmtree(self.tempdir) | ||||||
|  |  | ||||||
|  |     @defer.inlineCallbacks | ||||||
|     def test_run_create_volume(self): |     def test_run_create_volume(self): | ||||||
|         vol_size = '0' |         vol_size = '0' | ||||||
|         user_id = 'fake' |         user_id = 'fake' | ||||||
| @@ -48,34 +58,40 @@ class VolumeTestCase(test.TrialTestCase): | |||||||
|                          volume_service.get_volume(volume_id)['volume_id']) |                          volume_service.get_volume(volume_id)['volume_id']) | ||||||
|  |  | ||||||
|         rv = self.volume.delete_volume(volume_id) |         rv = self.volume.delete_volume(volume_id) | ||||||
|         self.assertFailure(volume_service.get_volume(volume_id), |         self.assertRaises(exception.Error, volume_service.get_volume, volume_id) | ||||||
|                            exception.Error) |  | ||||||
|  |  | ||||||
|  |     @defer.inlineCallbacks | ||||||
|     def test_too_big_volume(self): |     def test_too_big_volume(self): | ||||||
|         vol_size = '1001' |         vol_size = '1001' | ||||||
|         user_id = 'fake' |         user_id = 'fake' | ||||||
|         project_id = 'fake' |         project_id = 'fake' | ||||||
|         self.assertRaises(TypeError, |         try: | ||||||
|                           self.volume.create_volume, |             yield self.volume.create_volume(vol_size, user_id, project_id) | ||||||
|                           vol_size, user_id, project_id) |             self.fail("Should have thrown TypeError") | ||||||
|  |         except TypeError: | ||||||
|  |             pass | ||||||
|  |  | ||||||
|  |     @defer.inlineCallbacks | ||||||
|     def test_too_many_volumes(self): |     def test_too_many_volumes(self): | ||||||
|         vol_size = '1' |         vol_size = '1' | ||||||
|         user_id = 'fake' |         user_id = 'fake' | ||||||
|         project_id = 'fake' |         project_id = 'fake' | ||||||
|         num_shelves = FLAGS.last_shelf_id - FLAGS.first_shelf_id + 1 |         num_shelves = FLAGS.last_shelf_id - FLAGS.first_shelf_id + 1 | ||||||
|         total_slots = FLAGS.slots_per_shelf * num_shelves |         total_slots = FLAGS.blades_per_shelf * num_shelves | ||||||
|         vols = [] |         vols = [] | ||||||
|  |         from nova import datastore | ||||||
|  |         redis = datastore.Redis.instance() | ||||||
|         for i in xrange(total_slots): |         for i in xrange(total_slots): | ||||||
|             vid = yield self.volume.create_volume(vol_size, user_id, project_id) |             vid = yield self.volume.create_volume(vol_size, user_id, project_id) | ||||||
|             vols.append(vid) |             vols.append(vid) | ||||||
|         self.assertFailure(self.volume.create_volume(vol_size, |         self.assertFailure(self.volume.create_volume(vol_size, | ||||||
|                                                      user_id, |                                                      user_id, | ||||||
|                                                      project_id), |                                                      project_id), | ||||||
|                            volume_service.NoMoreVolumes) |                            volume_service.NoMoreBlades) | ||||||
|         for id in vols: |         for id in vols: | ||||||
|             yield self.volume.delete_volume(id) |             yield self.volume.delete_volume(id) | ||||||
|  |  | ||||||
|  |     @defer.inlineCallbacks | ||||||
|     def test_run_attach_detach_volume(self): |     def test_run_attach_detach_volume(self): | ||||||
|         # Create one volume and one compute to test with |         # Create one volume and one compute to test with | ||||||
|         instance_id = "storage-test" |         instance_id = "storage-test" | ||||||
| @@ -84,22 +100,26 @@ class VolumeTestCase(test.TrialTestCase): | |||||||
|         project_id = 'fake' |         project_id = 'fake' | ||||||
|         mountpoint = "/dev/sdf" |         mountpoint = "/dev/sdf" | ||||||
|         volume_id = yield self.volume.create_volume(vol_size, user_id, project_id) |         volume_id = yield self.volume.create_volume(vol_size, user_id, project_id) | ||||||
|  |  | ||||||
|         volume_obj = volume_service.get_volume(volume_id) |         volume_obj = volume_service.get_volume(volume_id) | ||||||
|         volume_obj.start_attach(instance_id, mountpoint) |         volume_obj.start_attach(instance_id, mountpoint) | ||||||
|         rv = yield self.compute.attach_volume(volume_id, |         if FLAGS.fake_tests: | ||||||
|                                           instance_id, |             volume_obj.finish_attach() | ||||||
|  |         else: | ||||||
|  |             rv = yield self.compute.attach_volume(instance_id, | ||||||
|  |                                                   volume_id, | ||||||
|                                                   mountpoint) |                                                   mountpoint) | ||||||
|         self.assertEqual(volume_obj['status'], "in-use") |         self.assertEqual(volume_obj['status'], "in-use") | ||||||
|         self.assertEqual(volume_obj['attachStatus'], "attached") |         self.assertEqual(volume_obj['attach_status'], "attached") | ||||||
|         self.assertEqual(volume_obj['instance_id'], instance_id) |         self.assertEqual(volume_obj['instance_id'], instance_id) | ||||||
|         self.assertEqual(volume_obj['mountpoint'], mountpoint) |         self.assertEqual(volume_obj['mountpoint'], mountpoint) | ||||||
|  |  | ||||||
|         self.assertRaises(exception.Error, |         self.assertFailure(self.volume.delete_volume(volume_id), exception.Error) | ||||||
|                           self.volume.delete_volume, |         volume_obj.start_detach() | ||||||
|  |         if FLAGS.fake_tests: | ||||||
|  |             volume_obj.finish_detach() | ||||||
|  |         else: | ||||||
|  |             rv = yield self.volume.detach_volume(instance_id, | ||||||
|                                                  volume_id) |                                                  volume_id) | ||||||
|  |  | ||||||
|         rv = yield self.volume.detach_volume(volume_id) |  | ||||||
|         volume_obj = volume_service.get_volume(volume_id) |         volume_obj = volume_service.get_volume(volume_id) | ||||||
|         self.assertEqual(volume_obj['status'], "available") |         self.assertEqual(volume_obj['status'], "available") | ||||||
|  |  | ||||||
| @@ -108,6 +128,27 @@ class VolumeTestCase(test.TrialTestCase): | |||||||
|                           volume_service.get_volume, |                           volume_service.get_volume, | ||||||
|                           volume_id) |                           volume_id) | ||||||
|  |  | ||||||
|  |     @defer.inlineCallbacks | ||||||
|  |     def test_multiple_volume_race_condition(self): | ||||||
|  |         vol_size = "5" | ||||||
|  |         user_id = "fake" | ||||||
|  |         project_id = 'fake' | ||||||
|  |         shelf_blades = [] | ||||||
|  |         def _check(volume_id): | ||||||
|  |             vol = volume_service.get_volume(volume_id) | ||||||
|  |             shelf_blade = '%s.%s' % (vol['shelf_id'], vol['blade_id']) | ||||||
|  |             self.assert_(shelf_blade not in shelf_blades) | ||||||
|  |             shelf_blades.append(shelf_blade) | ||||||
|  |             logging.debug("got %s" % shelf_blade) | ||||||
|  |             vol.destroy() | ||||||
|  |         deferreds = [] | ||||||
|  |         for i in range(5): | ||||||
|  |             d = self.volume.create_volume(vol_size, user_id, project_id) | ||||||
|  |             d.addCallback(_check) | ||||||
|  |             d.addErrback(self.fail) | ||||||
|  |             deferreds.append(d) | ||||||
|  |         yield defer.DeferredList(deferreds) | ||||||
|  |  | ||||||
|     def test_multi_node(self): |     def test_multi_node(self): | ||||||
|         # TODO(termie): Figure out how to test with two nodes, |         # TODO(termie): Figure out how to test with two nodes, | ||||||
|         # each of them having a different FLAG for storage_node |         # each of them having a different FLAG for storage_node | ||||||
|   | |||||||
| @@ -241,15 +241,7 @@ def serve(filename): | |||||||
|         print 'usage: %s [options] [start|stop|restart]' % argv[0] |         print 'usage: %s [options] [start|stop|restart]' % argv[0] | ||||||
|         sys.exit(1) |         sys.exit(1) | ||||||
|  |  | ||||||
|     class NoNewlineFormatter(logging.Formatter): |     formatter = logging.Formatter( | ||||||
|         """Strips newlines from default formatter""" |  | ||||||
|         def format(self, record): |  | ||||||
|             """Grabs default formatter's output and strips newlines""" |  | ||||||
|             data = logging.Formatter.format(self, record) |  | ||||||
|             return data.replace("\n", "--") |  | ||||||
|  |  | ||||||
|     # NOTE(vish): syslog-ng doesn't handle newlines from trackbacks very well |  | ||||||
|     formatter = NoNewlineFormatter( |  | ||||||
|         '(%(name)s): %(levelname)s %(message)s') |         '(%(name)s): %(levelname)s %(message)s') | ||||||
|     handler = logging.StreamHandler(log.StdioOnnaStick()) |     handler = logging.StreamHandler(log.StdioOnnaStick()) | ||||||
|     handler.setFormatter(formatter) |     handler.setFormatter(formatter) | ||||||
|   | |||||||
| @@ -57,6 +57,7 @@ def rangetest(**argchecks):                 # validate ranges for both+defaults | |||||||
|         return onCall |         return onCall | ||||||
|     return onDecorator |     return onDecorator | ||||||
|  |  | ||||||
|  |  | ||||||
| def typetest(**argchecks): | def typetest(**argchecks): | ||||||
|     def onDecorator(func): |     def onDecorator(func): | ||||||
|         import sys |         import sys | ||||||
|   | |||||||
							
								
								
									
										10
									
								
								run_tests.py
									
									
									
									
									
								
							
							
						
						
									
										10
									
								
								run_tests.py
									
									
									
									
									
								
							| @@ -38,11 +38,11 @@ Due to our use of multiprocessing it we frequently get some ignorable | |||||||
| 'Interrupted system call' exceptions after test completion. | 'Interrupted system call' exceptions after test completion. | ||||||
|  |  | ||||||
| """ | """ | ||||||
|  |  | ||||||
| import __main__ | import __main__ | ||||||
| import os | import os | ||||||
| import sys | import sys | ||||||
|  |  | ||||||
|  |  | ||||||
| from twisted.scripts import trial as trial_script | from twisted.scripts import trial as trial_script | ||||||
|  |  | ||||||
| from nova import datastore | from nova import datastore | ||||||
| @@ -54,21 +54,23 @@ from nova.tests.auth_unittest import * | |||||||
| from nova.tests.api_unittest import * | from nova.tests.api_unittest import * | ||||||
| from nova.tests.cloud_unittest import * | from nova.tests.cloud_unittest import * | ||||||
| from nova.tests.compute_unittest import * | from nova.tests.compute_unittest import * | ||||||
|  | from nova.tests.flags_unittest import * | ||||||
| from nova.tests.model_unittest import * | from nova.tests.model_unittest import * | ||||||
| from nova.tests.network_unittest import * | from nova.tests.network_unittest import * | ||||||
| from nova.tests.objectstore_unittest import * | from nova.tests.objectstore_unittest import * | ||||||
| from nova.tests.process_unittest import * | from nova.tests.process_unittest import * | ||||||
|  | from nova.tests.rpc_unittest import * | ||||||
| from nova.tests.validator_unittest import * | from nova.tests.validator_unittest import * | ||||||
| from nova.tests.volume_unittest import * | from nova.tests.volume_unittest import * | ||||||
|  |  | ||||||
|  |  | ||||||
| FLAGS = flags.FLAGS | FLAGS = flags.FLAGS | ||||||
|  |  | ||||||
| flags.DEFINE_bool('flush_db', True, | flags.DEFINE_bool('flush_db', True, | ||||||
|                   'Flush the database before running fake tests') |                   'Flush the database before running fake tests') | ||||||
|  |  | ||||||
| flags.DEFINE_string('tests_stderr', 'run_tests.err.log', | flags.DEFINE_string('tests_stderr', 'run_tests.err.log', | ||||||
|                     'Path to where to pipe STDERR during test runs. Default = "run_tests.err.log"') |                     'Path to where to pipe STDERR during test runs.' | ||||||
|  |                     ' Default = "run_tests.err.log"') | ||||||
|  |  | ||||||
|  |  | ||||||
| if __name__ == '__main__': | if __name__ == '__main__': | ||||||
|     OptionsClass = twistd.WrapTwistedOptions(trial_script.Options) |     OptionsClass = twistd.WrapTwistedOptions(trial_script.Options) | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user
	 Justin Santa Barbara
					Justin Santa Barbara