blob: 9c1bb98348c745f1745d43f522cc30236022f674 [file] [log] [blame]
# Licensed to the Apache Software Foundation (ASF) under one or more
# contributor license agreements. See the NOTICE file distributed with
# this work for additional information regarding copyright ownership.
# The ASF licenses this file to You under the Apache License, Version 2.0
# (the "License"); you may not use this file except in compliance with
# the License. You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
import sys
import time
from libcloud.utils.py3 import httplib
try:
import simplejson as json
except ImportError:
import json
from libcloud.utils.misc import reverse_dict
from libcloud.common.types import LibcloudError
from libcloud.common.gogrid import GoGridConnection, GoGridResponse,\
BaseGoGridDriver
from libcloud.loadbalancer.base import LoadBalancer, Member, Driver, Algorithm
from libcloud.loadbalancer.base import DEFAULT_ALGORITHM
from libcloud.loadbalancer.types import State, LibcloudLBImmutableError
class GoGridLBResponse(GoGridResponse):
def success(self):
if self.status == httplib.INTERNAL_SERVER_ERROR:
# Hack, but at least this error message is more useful than
# "unexpected server error"
body = json.loads(self.body)
if body['method'] == '/grid/loadbalancer/add' and \
len(body['list']) >= 1 and \
body['list'][0]['message'].find(
'unexpected server error') != -1:
raise LibcloudError(
value='You mostly likely tried to add a member with an IP'
' address not assigned to your account', driver=self)
return super(GoGridLBResponse, self).success()
class GoGridLBConnection(GoGridConnection):
"""
Connection class for the GoGrid load-balancer driver.
"""
responseCls = GoGridLBResponse
class GoGridLBDriver(BaseGoGridDriver, Driver):
connectionCls = GoGridLBConnection
api_name = 'gogrid_lb'
name = 'GoGrid LB'
website = 'http://www.gogrid.com/'
LB_STATE_MAP = {'On': State.RUNNING,
'Unknown': State.UNKNOWN}
_VALUE_TO_ALGORITHM_MAP = {
'round robin': Algorithm.ROUND_ROBIN,
'least connect': Algorithm.LEAST_CONNECTIONS
}
_ALGORITHM_TO_VALUE_MAP = reverse_dict(_VALUE_TO_ALGORITHM_MAP)
def __init__(self, *args, **kwargs):
"""
@inherits: L{Driver.__init__}
"""
super(GoGridLBDriver, self).__init__(*args, **kwargs)
def list_protocols(self):
# GoGrid only supports http
return ['http']
def list_balancers(self):
return self._to_balancers(
self.connection.request('/api/grid/loadbalancer/list').object)
def ex_create_balancer_nowait(self, name, members, protocol='http',
port=80, algorithm=DEFAULT_ALGORITHM):
"""
@inherits: L{Driver.create_balancer}
"""
algorithm = self._algorithm_to_value(algorithm)
params = {'name': name,
'loadbalancer.type': algorithm,
'virtualip.ip': self._get_first_ip(),
'virtualip.port': port}
params.update(self._members_to_params(members))
resp = self.connection.request('/api/grid/loadbalancer/add',
method='GET',
params=params)
return self._to_balancers(resp.object)[0]
def create_balancer(self, name, members, protocol='http', port=80,
algorithm=DEFAULT_ALGORITHM):
balancer = self.ex_create_balancer_nowait(name, members, protocol,
port, algorithm)
timeout = 60 * 20
waittime = 0
interval = 2 * 15
if balancer.id is not None:
return balancer
else:
while waittime < timeout:
balancers = self.list_balancers()
for i in balancers:
if i.name == balancer.name and i.id is not None:
return i
waittime += interval
time.sleep(interval)
raise Exception('Failed to get id')
def destroy_balancer(self, balancer):
try:
resp = self.connection.request(
'/api/grid/loadbalancer/delete', method='POST',
params={'id': balancer.id})
except Exception:
e = sys.exc_info()[1]
if "Update request for LoadBalancer" in str(e):
raise LibcloudLBImmutableError(
"Cannot delete immutable object", GoGridLBDriver)
else:
raise
return resp.status == 200
def get_balancer(self, **kwargs):
params = {}
try:
params['name'] = kwargs['ex_balancer_name']
except KeyError:
balancer_id = kwargs['balancer_id']
params['id'] = balancer_id
resp = self.connection.request('/api/grid/loadbalancer/get',
params=params)
return self._to_balancers(resp.object)[0]
def balancer_attach_member(self, balancer, member):
members = self.balancer_list_members(balancer)
members.append(member)
params = {"id": balancer.id}
params.update(self._members_to_params(members))
resp = self._update_balancer(params)
return [m for m in
self._to_members(resp.object["list"][0]["realiplist"], balancer)
if m.ip == member.ip][0]
def balancer_detach_member(self, balancer, member):
members = self.balancer_list_members(balancer)
remaining_members = [n for n in members if n.id != member.id]
params = {"id": balancer.id}
params.update(self._members_to_params(remaining_members))
resp = self._update_balancer(params)
return resp.status == 200
def balancer_list_members(self, balancer):
resp = self.connection.request('/api/grid/loadbalancer/get',
params={'id': balancer.id})
return self._to_members(resp.object["list"][0]["realiplist"], balancer)
def _update_balancer(self, params):
try:
return self.connection.request('/api/grid/loadbalancer/edit',
method='POST',
params=params)
except Exception:
e = sys.exc_info()[1]
if "Update already pending" in str(e):
raise LibcloudLBImmutableError(
"Balancer is immutable", GoGridLBDriver)
raise LibcloudError(value='Exception: %s' % str(e), driver=self)
def _members_to_params(self, members):
"""
Helper method to convert list of L{Member} objects
to GET params.
"""
params = {}
i = 0
for member in members:
params["realiplist.%s.ip" % i] = member.ip
params["realiplist.%s.port" % i] = member.port
i += 1
return params
def _to_balancers(self, object):
return [self._to_balancer(el) for el in object["list"]]
def _to_balancer(self, el):
lb = LoadBalancer(id=el.get("id"),
name=el["name"],
state=self.LB_STATE_MAP.get(
el["state"]["name"], State.UNKNOWN),
ip=el["virtualip"]["ip"]["ip"],
port=el["virtualip"]["port"],
driver=self.connection.driver)
return lb
def _to_members(self, object, balancer=None):
return [self._to_member(el, balancer) for el in object]
def _to_member(self, el, balancer=None):
member = Member(id=el["ip"]["id"],
ip=el["ip"]["ip"],
port=el["port"],
balancer=balancer)
return member