51129f6ef3
Also more closely mirror AWS by not returning ELB BackendServerDescriptions that don't have policies. Signed-off-by: Michael Nussbaum <michael.nussbaum@getbraintree.com>
363 lines
15 KiB
Python
363 lines
15 KiB
Python
from __future__ import unicode_literals
|
|
|
|
import boto.ec2.elb
|
|
from boto.ec2.elb.attributes import (
|
|
LbAttributes,
|
|
ConnectionSettingAttribute,
|
|
ConnectionDrainingAttribute,
|
|
AccessLogAttribute,
|
|
CrossZoneLoadBalancingAttribute,
|
|
)
|
|
from boto.ec2.elb.policies import (
|
|
Policies,
|
|
OtherPolicy,
|
|
)
|
|
from moto.core import BaseBackend
|
|
from moto.ec2.models import ec2_backends
|
|
from .exceptions import (
|
|
LoadBalancerNotFoundError,
|
|
TooManyTagsError,
|
|
BadHealthCheckDefinition,
|
|
DuplicateLoadBalancerName,
|
|
)
|
|
|
|
|
|
|
|
class FakeHealthCheck(object):
|
|
def __init__(self, timeout, healthy_threshold, unhealthy_threshold,
|
|
interval, target):
|
|
self.timeout = timeout
|
|
self.healthy_threshold = healthy_threshold
|
|
self.unhealthy_threshold = unhealthy_threshold
|
|
self.interval = interval
|
|
self.target = target
|
|
if not target.startswith(('HTTP', 'TCP', 'HTTPS', 'SSL')):
|
|
raise BadHealthCheckDefinition
|
|
|
|
|
|
class FakeListener(object):
|
|
def __init__(self, load_balancer_port, instance_port, protocol, ssl_certificate_id):
|
|
self.load_balancer_port = load_balancer_port
|
|
self.instance_port = instance_port
|
|
self.protocol = protocol.upper()
|
|
self.ssl_certificate_id = ssl_certificate_id
|
|
self.policy_names = []
|
|
|
|
def __repr__(self):
|
|
return "FakeListener(lbp: %s, inp: %s, pro: %s, cid: %s, policies: %s)" % (self.load_balancer_port, self.instance_port, self.protocol, self.ssl_certificate_id, self.policy_names)
|
|
|
|
|
|
class FakeBackend(object):
|
|
def __init__(self, instance_port):
|
|
self.instance_port = instance_port
|
|
self.policy_names = []
|
|
|
|
def __repr__(self):
|
|
return "FakeBackend(inp: %s, policies: %s)" % (self.instance_port, self.policy_names)
|
|
|
|
|
|
class FakeLoadBalancer(object):
|
|
def __init__(self, name, zones, ports, scheme='internet-facing', vpc_id=None, subnets=None):
|
|
self.name = name
|
|
self.health_check = None
|
|
self.instance_ids = []
|
|
self.zones = zones
|
|
self.listeners = []
|
|
self.backends = []
|
|
self.scheme = scheme
|
|
self.attributes = FakeLoadBalancer.get_default_attributes()
|
|
self.policies = Policies()
|
|
self.policies.other_policies = []
|
|
self.policies.app_cookie_stickiness_policies = []
|
|
self.policies.lb_cookie_stickiness_policies = []
|
|
self.subnets = subnets or []
|
|
self.vpc_id = vpc_id or 'vpc-56e10e3d'
|
|
self.tags = {}
|
|
self.dns_name = "%s.us-east-1.elb.amazonaws.com" % (name)
|
|
|
|
for port in ports:
|
|
listener = FakeListener(
|
|
protocol=(port.get('protocol') or port['Protocol']),
|
|
load_balancer_port=(port.get('load_balancer_port') or port['LoadBalancerPort']),
|
|
instance_port=(port.get('instance_port') or port['InstancePort']),
|
|
ssl_certificate_id=port.get('sslcertificate_id', port.get('SSLCertificateId')),
|
|
)
|
|
self.listeners.append(listener)
|
|
|
|
# it is unclear per the AWS documentation as to when or how backend
|
|
# information gets set, so let's guess and set it here *shrug*
|
|
backend = FakeBackend(
|
|
instance_port=(port.get('instance_port') or port['InstancePort']),
|
|
)
|
|
self.backends.append(backend)
|
|
|
|
@classmethod
|
|
def create_from_cloudformation_json(cls, resource_name, cloudformation_json, region_name):
|
|
properties = cloudformation_json['Properties']
|
|
|
|
elb_backend = elb_backends[region_name]
|
|
new_elb = elb_backend.create_load_balancer(
|
|
name=properties.get('LoadBalancerName', resource_name),
|
|
zones=properties.get('AvailabilityZones', []),
|
|
ports=properties['Listeners'],
|
|
scheme=properties.get('Scheme', 'internet-facing'),
|
|
)
|
|
|
|
instance_ids = properties.get('Instances', [])
|
|
for instance_id in instance_ids:
|
|
elb_backend.register_instances(new_elb.name, [instance_id])
|
|
|
|
policies = properties.get('Policies', [])
|
|
port_policies = {}
|
|
for policy in policies:
|
|
policy_name = policy["PolicyName"]
|
|
other_policy = OtherPolicy()
|
|
other_policy.policy_name = policy_name
|
|
elb_backend.create_lb_other_policy(new_elb.name, other_policy)
|
|
for port in policy.get("InstancePorts", []):
|
|
policies_for_port = port_policies.get(port, set())
|
|
policies_for_port.add(policy_name)
|
|
port_policies[port] = policies_for_port
|
|
|
|
for port, policies in port_policies.items():
|
|
elb_backend.set_load_balancer_policies_of_backend_server(new_elb.name, port, list(policies))
|
|
|
|
health_check = properties.get('HealthCheck')
|
|
if health_check:
|
|
elb_backend.configure_health_check(
|
|
load_balancer_name=new_elb.name,
|
|
timeout=health_check['Timeout'],
|
|
healthy_threshold=health_check['HealthyThreshold'],
|
|
unhealthy_threshold=health_check['UnhealthyThreshold'],
|
|
interval=health_check['Interval'],
|
|
target=health_check['Target'],
|
|
)
|
|
|
|
return new_elb
|
|
|
|
@classmethod
|
|
def update_from_cloudformation_json(cls, original_resource, new_resource_name, cloudformation_json, region_name):
|
|
cls.delete_from_cloudformation_json(original_resource.name, cloudformation_json, region_name)
|
|
return cls.create_from_cloudformation_json(new_resource_name, cloudformation_json, region_name)
|
|
|
|
@classmethod
|
|
def delete_from_cloudformation_json(cls, resource_name, cloudformation_json, region_name):
|
|
elb_backend = elb_backends[region_name]
|
|
try:
|
|
elb_backend.delete_load_balancer(resource_name)
|
|
except KeyError:
|
|
pass
|
|
|
|
@property
|
|
def physical_resource_id(self):
|
|
return self.name
|
|
|
|
def get_cfn_attribute(self, attribute_name):
|
|
from moto.cloudformation.exceptions import UnformattedGetAttTemplateException
|
|
if attribute_name == 'CanonicalHostedZoneName':
|
|
raise NotImplementedError('"Fn::GetAtt" : [ "{0}" , "CanonicalHostedZoneName" ]"')
|
|
elif attribute_name == 'CanonicalHostedZoneNameID':
|
|
raise NotImplementedError('"Fn::GetAtt" : [ "{0}" , "CanonicalHostedZoneNameID" ]"')
|
|
elif attribute_name == 'DNSName':
|
|
return self.dns_name
|
|
elif attribute_name == 'SourceSecurityGroup.GroupName':
|
|
raise NotImplementedError('"Fn::GetAtt" : [ "{0}" , "SourceSecurityGroup.GroupName" ]"')
|
|
elif attribute_name == 'SourceSecurityGroup.OwnerAlias':
|
|
raise NotImplementedError('"Fn::GetAtt" : [ "{0}" , "SourceSecurityGroup.OwnerAlias" ]"')
|
|
raise UnformattedGetAttTemplateException()
|
|
|
|
@classmethod
|
|
def get_default_attributes(cls):
|
|
attributes = LbAttributes()
|
|
|
|
cross_zone_load_balancing = CrossZoneLoadBalancingAttribute()
|
|
cross_zone_load_balancing.enabled = False
|
|
attributes.cross_zone_load_balancing = cross_zone_load_balancing
|
|
|
|
connection_draining = ConnectionDrainingAttribute()
|
|
connection_draining.enabled = False
|
|
attributes.connection_draining = connection_draining
|
|
|
|
access_log = AccessLogAttribute()
|
|
access_log.enabled = False
|
|
attributes.access_log = access_log
|
|
|
|
connection_settings = ConnectionSettingAttribute()
|
|
connection_settings.idle_timeout = 60
|
|
attributes.connecting_settings = connection_settings
|
|
|
|
return attributes
|
|
|
|
def add_tag(self, key, value):
|
|
if len(self.tags) >= 10 and key not in self.tags:
|
|
raise TooManyTagsError()
|
|
self.tags[key] = value
|
|
|
|
def list_tags(self):
|
|
return self.tags
|
|
|
|
def remove_tag(self, key):
|
|
if key in self.tags:
|
|
del self.tags[key]
|
|
|
|
def delete(self, region):
|
|
''' Not exposed as part of the ELB API - used for CloudFormation. '''
|
|
elb_backends[region].delete_load_balancer(self.name)
|
|
|
|
|
|
class ELBBackend(BaseBackend):
|
|
|
|
def __init__(self, region_name=None):
|
|
self.region_name = region_name
|
|
self.load_balancers = {}
|
|
|
|
def reset(self):
|
|
region_name = self.region_name
|
|
self.__dict__ = {}
|
|
self.__init__(region_name)
|
|
|
|
def create_load_balancer(self, name, zones, ports, scheme='internet-facing', subnets=None):
|
|
vpc_id = None
|
|
ec2_backend = ec2_backends[self.region_name]
|
|
if subnets:
|
|
subnet = ec2_backend.get_subnet(subnets[0])
|
|
vpc_id = subnet.vpc_id
|
|
if name in self.load_balancers:
|
|
raise DuplicateLoadBalancerName(name)
|
|
new_load_balancer = FakeLoadBalancer(name=name, zones=zones, ports=ports, scheme=scheme, subnets=subnets, vpc_id=vpc_id)
|
|
self.load_balancers[name] = new_load_balancer
|
|
return new_load_balancer
|
|
|
|
def create_load_balancer_listeners(self, name, ports):
|
|
balancer = self.load_balancers.get(name, None)
|
|
if balancer:
|
|
for port in ports:
|
|
protocol = port['protocol']
|
|
instance_port = port['instance_port']
|
|
lb_port = port['load_balancer_port']
|
|
ssl_certificate_id = port.get('sslcertificate_id')
|
|
for listener in balancer.listeners:
|
|
if lb_port == listener.load_balancer_port:
|
|
break
|
|
else:
|
|
balancer.listeners.append(FakeListener(lb_port, instance_port, protocol, ssl_certificate_id))
|
|
|
|
return balancer
|
|
|
|
def describe_load_balancers(self, names):
|
|
balancers = self.load_balancers.values()
|
|
if names:
|
|
matched_balancers = [balancer for balancer in balancers if balancer.name in names]
|
|
if len(names) != len(matched_balancers):
|
|
missing_elb = list(set(names) - set(matched_balancers))[0]
|
|
raise LoadBalancerNotFoundError(missing_elb)
|
|
return matched_balancers
|
|
else:
|
|
return balancers
|
|
|
|
def delete_load_balancer_listeners(self, name, ports):
|
|
balancer = self.load_balancers.get(name, None)
|
|
listeners = []
|
|
if balancer:
|
|
for lb_port in ports:
|
|
for listener in balancer.listeners:
|
|
if int(lb_port) == int(listener.load_balancer_port):
|
|
continue
|
|
else:
|
|
listeners.append(listener)
|
|
balancer.listeners = listeners
|
|
return balancer
|
|
|
|
def delete_load_balancer(self, load_balancer_name):
|
|
self.load_balancers.pop(load_balancer_name, None)
|
|
|
|
def get_load_balancer(self, load_balancer_name):
|
|
return self.load_balancers.get(load_balancer_name)
|
|
|
|
def configure_health_check(self, load_balancer_name, timeout,
|
|
healthy_threshold, unhealthy_threshold, interval,
|
|
target):
|
|
check = FakeHealthCheck(timeout, healthy_threshold, unhealthy_threshold,
|
|
interval, target)
|
|
load_balancer = self.get_load_balancer(load_balancer_name)
|
|
load_balancer.health_check = check
|
|
return check
|
|
|
|
def set_load_balancer_listener_sslcertificate(self, name, lb_port, ssl_certificate_id):
|
|
balancer = self.load_balancers.get(name, None)
|
|
if balancer:
|
|
for idx, listener in enumerate(balancer.listeners):
|
|
if lb_port == listener.load_balancer_port:
|
|
balancer.listeners[idx].ssl_certificate_id = ssl_certificate_id
|
|
|
|
return balancer
|
|
|
|
def register_instances(self, load_balancer_name, instance_ids):
|
|
load_balancer = self.get_load_balancer(load_balancer_name)
|
|
load_balancer.instance_ids.extend(instance_ids)
|
|
return load_balancer
|
|
|
|
def deregister_instances(self, load_balancer_name, instance_ids):
|
|
load_balancer = self.get_load_balancer(load_balancer_name)
|
|
new_instance_ids = [instance_id for instance_id in load_balancer.instance_ids if instance_id not in instance_ids]
|
|
load_balancer.instance_ids = new_instance_ids
|
|
return load_balancer
|
|
|
|
def set_cross_zone_load_balancing_attribute(self, load_balancer_name, attribute):
|
|
load_balancer = self.get_load_balancer(load_balancer_name)
|
|
load_balancer.attributes.cross_zone_load_balancing = attribute
|
|
return load_balancer
|
|
|
|
def set_access_log_attribute(self, load_balancer_name, attribute):
|
|
load_balancer = self.get_load_balancer(load_balancer_name)
|
|
load_balancer.attributes.access_log = attribute
|
|
return load_balancer
|
|
|
|
def set_connection_draining_attribute(self, load_balancer_name, attribute):
|
|
load_balancer = self.get_load_balancer(load_balancer_name)
|
|
load_balancer.attributes.connection_draining = attribute
|
|
return load_balancer
|
|
|
|
def set_connection_settings_attribute(self, load_balancer_name, attribute):
|
|
load_balancer = self.get_load_balancer(load_balancer_name)
|
|
load_balancer.attributes.connecting_settings = attribute
|
|
return load_balancer
|
|
|
|
def create_lb_other_policy(self, load_balancer_name, other_policy):
|
|
load_balancer = self.get_load_balancer(load_balancer_name)
|
|
if other_policy.policy_name not in [p.policy_name for p in load_balancer.policies.other_policies]:
|
|
load_balancer.policies.other_policies.append(other_policy)
|
|
|
|
return load_balancer
|
|
|
|
def create_app_cookie_stickiness_policy(self, load_balancer_name, policy):
|
|
load_balancer = self.get_load_balancer(load_balancer_name)
|
|
load_balancer.policies.app_cookie_stickiness_policies.append(policy)
|
|
return load_balancer
|
|
|
|
def create_lb_cookie_stickiness_policy(self, load_balancer_name, policy):
|
|
load_balancer = self.get_load_balancer(load_balancer_name)
|
|
load_balancer.policies.lb_cookie_stickiness_policies.append(policy)
|
|
return load_balancer
|
|
|
|
def set_load_balancer_policies_of_backend_server(self, load_balancer_name, instance_port, policies):
|
|
load_balancer = self.get_load_balancer(load_balancer_name)
|
|
backend = [b for b in load_balancer.backends if int(b.instance_port) == instance_port][0]
|
|
backend_idx = load_balancer.backends.index(backend)
|
|
backend.policy_names = policies
|
|
load_balancer.backends[backend_idx] = backend
|
|
return load_balancer
|
|
|
|
def set_load_balancer_policies_of_listener(self, load_balancer_name, load_balancer_port, policies):
|
|
load_balancer = self.get_load_balancer(load_balancer_name)
|
|
listener = [l for l in load_balancer.listeners if int(l.load_balancer_port) == load_balancer_port][0]
|
|
listener_idx = load_balancer.listeners.index(listener)
|
|
listener.policy_names = policies
|
|
load_balancer.listeners[listener_idx] = listener
|
|
return load_balancer
|
|
|
|
|
|
elb_backends = {}
|
|
for region in ec2_backends.keys():
|
|
elb_backends[region] = ELBBackend(region)
|