Simple per-user instance quota checking in the create call. PEP8 fixes.

This commit is contained in:
Ed Cranford 2012-06-15 11:55:50 -05:00
parent 04110bd7ea
commit 25c97fa6c5
12 changed files with 51 additions and 21 deletions

View File

@ -50,6 +50,7 @@ block_device_mapping = /var/lib/mysql
device_path = /var/lib/mysql device_path = /var/lib/mysql
mount_point = /var/lib/mysql mount_point = /var/lib/mysql
max_accepted_volume_size = 10 max_accepted_volume_size = 10
max_instances_per_user = 5
volume_time_out=30 volume_time_out=30
# Reddwarf DNS # Reddwarf DNS

View File

@ -61,7 +61,8 @@ nova_volume_service_type = volume
nova_volume_service_name = Volume Service nova_volume_service_name = Volume Service
device_path = /dev/vdb device_path = /dev/vdb
mount_point = /var/lib/mysql mount_point = /var/lib/mysql
max_accepted_volume_size = 10 max_accepted_volume_size = 25
max_instances_per_user = 5
volume_time_out=30 volume_time_out=30
# Auth # Auth

View File

@ -82,6 +82,16 @@ class OverLimit(ReddwarfError):
"rate.") "rate.")
class QuotaExceeded(ReddwarfError):
message = _("User instance quota exceeded.")
class VolumeQuotaExceeded(QuotaExceeded):
message = _("Instance volume quota exceeded.")
class GuestError(ReddwarfError): class GuestError(ReddwarfError):
message = _("An error occurred communicating with the guest: " message = _("An error occurred communicating with the guest: "

View File

@ -56,6 +56,8 @@ class BaseController(wsgi.Controller):
], ],
webob.exc.HTTPRequestEntityTooLarge: [ webob.exc.HTTPRequestEntityTooLarge: [
exception.OverLimit, exception.OverLimit,
exception.QuotaExceeded,
exception.VolumeQuotaExceeded,
], ],
webob.exc.HTTPServerError: [ webob.exc.HTTPServerError: [
exception.VolumeCreationFailure exception.VolumeCreationFailure
@ -246,6 +248,17 @@ class InstanceController(BaseController):
raise exception.BadValue(msg=e) raise exception.BadValue(msg=e)
else: else:
volume_size = None volume_size = None
instance_max = int(config.Config.get('max_instances_per_user', 5))
number_instances = models.DBInstance.find_all(tenant_id=tenant_id,
deleted=False).count()
if number_instances >= instance_max:
# That's too many, pal. Got to cut you off.
LOG.error(_("New instance would exceed user instance quota."))
msg = "User instance quota of %d would be exceeded."
raise exception.QuotaExceeded(msg % instance_max)
instance = models.Instance.create(context, name, flavor_id, instance = models.Instance.create(context, name, flavor_id,
image_id, databases, users, image_id, databases, users,
service_type, volume_size) service_type, volume_size)
@ -285,18 +298,18 @@ class InstanceController(BaseController):
"integer value, %s cannot be accepted." "integer value, %s cannot be accepted."
% volume_size) % volume_size)
raise exception.ReddwarfError(msg) raise exception.ReddwarfError(msg)
max_size = int(config.Config.get('max_accepted_volume_size', max_size = int(config.Config.get('max_accepted_volume_size', 1))
1))
if int(volume_size) > max_size: if int(volume_size) > max_size:
msg = ("Volume 'size' cannot exceed maximum " msg = ("Volume 'size' cannot exceed maximum "
"of %d Gb, %s cannot be accepted." "of %d Gb, %s cannot be accepted."
% (max_size, volume_size)) % (max_size, volume_size))
raise exception.ReddwarfError(msg) raise exception.VolumeQuotaExceeded(msg)
@staticmethod @staticmethod
def _validate(body): def _validate(body):
"""Validate that the request has all the required parameters""" """Validate that the request has all the required parameters"""
InstanceController._validate_body_not_empty(body) InstanceController._validate_body_not_empty(body)
try: try:
body['instance'] body['instance']
body['instance']['flavorRef'] body['instance']['flavorRef']
@ -315,6 +328,7 @@ class InstanceController(BaseController):
raise exception.MissingKey(key="size") raise exception.MissingKey(key="size")
elif must_have_vol: elif must_have_vol:
raise exception.MissingKey(key="volume") raise exception.MissingKey(key="volume")
except KeyError as e: except KeyError as e:
LOG.error(_("Create Instance Required field(s) - %s") % e) LOG.error(_("Create Instance Required field(s) - %s") % e)
raise exception.ReddwarfError("Required element/key - %s " raise exception.ReddwarfError("Required element/key - %s "

View File

@ -59,6 +59,7 @@ class API(object):
logging.error("Error running async task:") logging.error("Error running async task:")
logging.error((traceback.format_exception(type_, value, tb))) logging.error((traceback.format_exception(type_, value, tb)))
raise type_, value, tb raise type_, value, tb
eventlet.spawn_after(0, func) eventlet.spawn_after(0, func)
def _get_routing_key(self): def _get_routing_key(self):

View File

@ -80,4 +80,3 @@ class TaskManager(service.Manager):
instance_tasks.create_instance(flavor_id, flavor_ram, image_id, instance_tasks.create_instance(flavor_id, flavor_ram, image_id,
databases, users, service_type, databases, users, service_type,
volume_size) volume_size)

View File

@ -148,7 +148,8 @@ class FreshInstanceTasks(FreshInstance):
if utils.bool_from_string(dns_support): if utils.bool_from_string(dns_support):
def get_server(): def get_server():
return nova_client.servers.get(self.db_info.compute_instance_id) c_id = self.db_info.compute_instance_id
return nova_client.servers.get(c_id)
def ip_is_available(server): def ip_is_available(server):
LOG.info("Polling for ip addresses: $%s " % server.addresses) LOG.info("Polling for ip addresses: $%s " % server.addresses)

View File

@ -21,4 +21,3 @@ from rsdns.client.dns_client import DNSaas
from rsdns.client.dns_client import DNSaasClient from rsdns.client.dns_client import DNSaasClient
from rsdns.client.domains import DomainsManager from rsdns.client.domains import DomainsManager
from rsdns.client.records import RecordsManager from rsdns.client.records import RecordsManager

View File

@ -48,7 +48,8 @@ class DNSaasClient(HTTPClient):
def authenticate(self): def authenticate(self):
"""Set the management url and auth token""" """Set the management url and auth token"""
req_body = {'credentials':{'username':self.user, 'key':self.api_key}} req_body = {'credentials': {'username': self.user,
'key': self.api_key}}
resp, body = self.request(self.auth_url, "POST", body=req_body) resp, body = self.request(self.auth_url, "POST", body=req_body)
if 'access' in body: if 'access' in body:
if not self.management_url: if not self.management_url:
@ -103,6 +104,7 @@ class DNSaasClient(HTTPClient):
return resp, body return resp, body
class DNSaas(Client): class DNSaas(Client):
""" """
Top-level object to access the DNSaas service Top-level object to access the DNSaas service
@ -115,7 +117,8 @@ class DNSaas(Client):
from rsdns.client.domains import DomainsManager from rsdns.client.domains import DomainsManager
from rsdns.client.records import RecordsManager from rsdns.client.records import RecordsManager
super(DNSaas, self).__init__(self, accountId, username, apikey, auth_url, management_base_url) super(DNSaas, self).__init__(self, accountId, username, apikey,
auth_url, management_base_url)
self.client = DNSaasClient(accountId, username, apikey, auth_url, self.client = DNSaasClient(accountId, username, apikey, auth_url,
management_base_url) management_base_url)
self.domains = DomainsManager(self) self.domains = DomainsManager(self)

View File

@ -31,6 +31,7 @@ class Domain(base.Resource):
def response_list_name(self): def response_list_name(self):
return "domains" return "domains"
class FutureDomain(FutureResource): class FutureDomain(FutureResource):
def convert_callback(self, resp, body): def convert_callback(self, resp, body):

View File

@ -1,4 +1,5 @@
class RsDnsError(RuntimeError): class RsDnsError(RuntimeError):
def __init__(self, error): def __init__(self, error):
@ -13,6 +14,7 @@ class RsDnsError(RuntimeError):
def __str__(self): def __str__(self):
return self.message return self.message
class FutureResource(object): class FutureResource(object):
"""Polls a callback url to return a resource.""" """Polls a callback url to return a resource."""
@ -41,8 +43,6 @@ class FutureResource(object):
return None return None
resp_list = body['response'][self.response_list_name()] resp_list = body['response'][self.response_list_name()]
self.result = self.manager.create_from_list(resp_list) self.result = self.manager.create_from_list(resp_list)
#self.resource_class(self, res) for res in list]
#self.result = Domain(self.manager, body['self.convert_callback(resp, body)
return self.result return self.result
@property @property

View File

@ -53,7 +53,8 @@ class RecordsManager(base.ManagerWithFind):
""" """
resource_class = Record resource_class = Record
def create(self, domain, record_name, record_data, record_type, record_ttl): def create(self, domain, record_name, record_data, record_type,
record_ttl):
""" """
Create a new Record on the given domain Create a new Record on the given domain
@ -67,8 +68,8 @@ class RecordsManager(base.ManagerWithFind):
base.getid(domain), body=data) base.getid(domain), body=data)
if resp.status == 202: if resp.status == 202:
return FutureRecord(self, **body) return FutureRecord(self, **body)
raise RuntimeError("Did not expect response when creating a DNS record " raise RuntimeError("Did not expect response when creating a DNS "
"%s" % str(resp.status)) "record %s" % str(resp.status))
def create_from_list(self, list): def create_from_list(self, list):
return [self.resource_class(self, res) for res in list] return [self.resource_class(self, res) for res in list]
@ -143,4 +144,3 @@ class RecordsManager(base.ManagerWithFind):
else: else:
raise RuntimeError("Next href had multiple offset params!") raise RuntimeError("Next href had multiple offset params!")
return (list, next_offset) return (list, next_offset)