Add session persistence support for NVP advanced LBaaS

Change-Id: I2042894755cdaf54b2bc39e58028746aa7c1e8ea
Closes-Bug: #1256243
This commit is contained in:
berlin 2013-11-29 17:45:33 +08:00 committed by Gerrit Code Review
parent 6945f525f4
commit 19f299d6f7
6 changed files with 212 additions and 23 deletions

View File

@ -1240,6 +1240,8 @@ class NsxAdvancedPlugin(sr_db.ServiceRouter_mixin,
def update_vip(self, context, id, vip): def update_vip(self, context, id, vip):
edge_id = self._get_edge_id_by_vip_id(context, id) edge_id = self._get_edge_id_by_vip_id(context, id)
old_vip = self.get_vip(context, id) old_vip = self.get_vip(context, id)
session_persistence_update = bool(
vip['vip'].get('session_persistence'))
vip['vip']['status'] = service_constants.PENDING_UPDATE vip['vip']['status'] = service_constants.PENDING_UPDATE
v = super(NsxAdvancedPlugin, self).update_vip(context, id, vip) v = super(NsxAdvancedPlugin, self).update_vip(context, id, vip)
v[rsi.ROUTER_ID] = self._get_resource_router_id_binding( v[rsi.ROUTER_ID] = self._get_resource_router_id_binding(
@ -1262,7 +1264,7 @@ class NsxAdvancedPlugin(sr_db.ServiceRouter_mixin,
self.vcns_driver.create_vip(context, edge_id, v) self.vcns_driver.create_vip(context, edge_id, v)
return v return v
try: try:
self.vcns_driver.update_vip(context, v) self.vcns_driver.update_vip(context, v, session_persistence_update)
except Exception: except Exception:
with excutils.save_and_reraise_exception(): with excutils.save_and_reraise_exception():
LOG.exception(_("Failed to update vip with id: %s!"), id) LOG.exception(_("Failed to update vip with id: %s!"), id)

View File

@ -37,6 +37,13 @@ PROTOCOL_MAP = {
lb_constants.PROTOCOL_HTTP: 'http', lb_constants.PROTOCOL_HTTP: 'http',
lb_constants.PROTOCOL_HTTPS: 'tcp' lb_constants.PROTOCOL_HTTPS: 'tcp'
} }
SESSION_PERSISTENCE_METHOD_MAP = {
lb_constants.SESSION_PERSISTENCE_SOURCE_IP: 'sourceip',
lb_constants.SESSION_PERSISTENCE_APP_COOKIE: 'cookie',
lb_constants.SESSION_PERSISTENCE_HTTP_COOKIE: 'cookie'}
SESSION_PERSISTENCE_COOKIE_MAP = {
lb_constants.SESSION_PERSISTENCE_APP_COOKIE: 'app',
lb_constants.SESSION_PERSISTENCE_HTTP_COOKIE: 'insert'}
class EdgeLbDriver(): class EdgeLbDriver():
@ -51,9 +58,11 @@ class EdgeLbDriver():
pool_vseid = poolid_map['pool_vseid'] pool_vseid = poolid_map['pool_vseid']
return { return {
'name': vip.get('name'), 'name': vip.get('name'),
'description': vip.get('description'),
'ipAddress': vip.get('address'), 'ipAddress': vip.get('address'),
'protocol': vip.get('protocol'), 'protocol': vip.get('protocol'),
'port': vip.get('protocol_port'), 'port': vip.get('protocol_port'),
'connectionLimit': max(0, vip.get('connection_limit')),
'defaultPoolId': pool_vseid, 'defaultPoolId': pool_vseid,
'applicationProfileId': app_profileid 'applicationProfileId': app_profileid
} }
@ -75,15 +84,18 @@ class EdgeLbDriver():
def _convert_lb_pool(self, context, edge_id, pool, members): def _convert_lb_pool(self, context, edge_id, pool, members):
vsepool = { vsepool = {
'name': pool.get('name'), 'name': pool.get('name'),
'description': pool.get('description'),
'algorithm': BALANCE_MAP.get( 'algorithm': BALANCE_MAP.get(
pool.get('lb_method'), pool.get('lb_method'),
'round-robin'), 'round-robin'),
'transparent': True,
'member': [], 'member': [],
'monitorId': [] 'monitorId': []
} }
for member in members: for member in members:
vsepool['member'].append({ vsepool['member'].append({
'ipAddress': member['address'], 'ipAddress': member['address'],
'weight': member['weight'],
'port': member['protocol_port'] 'port': member['protocol_port']
}) })
##TODO(linb) right now, vse only accept at most one monitor per pool ##TODO(linb) right now, vse only accept at most one monitor per pool
@ -121,23 +133,45 @@ class EdgeLbDriver():
'id': monitor_vse['name'] 'id': monitor_vse['name']
} }
def _convert_app_profile(self, name, app_profile): def _convert_app_profile(self, name, sess_persist, protocol):
#TODO(linb): convert the session_persistence to vcns_app_profile = {
#corresponding app_profile 'insertXForwardedFor': False,
return { 'name': name,
"insertXForwardedFor": False, 'serverSslEnabled': False,
"name": name, 'sslPassthrough': False,
"persistence": { 'template': protocol,
"method": "sourceip"
},
"serverSslEnabled": False,
"sslPassthrough": False,
"template": "HTTP"
} }
# Since SSL Termination is not supported right now, so just use
# sslPassthrough mehtod if the protocol is HTTPS.
if protocol == lb_constants.PROTOCOL_HTTPS:
vcns_app_profile['sslPassthrough'] = True
if sess_persist.get('type'):
# If protocol is not HTTP, only sourceip is supported
if (protocol != lb_constants.PROTOCOL_HTTP and
sess_persist['type'] != (
lb_constants.SESSION_PERSISTENCE_SOURCE_IP)):
msg = (_("Invalid %(protocol)s persistence method: %(type)s") %
{'protocol': protocol,
'type': sess_persist['type']})
raise vcns_exc.VcnsBadRequest(resource='sess_persist', msg=msg)
persistence = {
'method': SESSION_PERSISTENCE_METHOD_MAP.get(
sess_persist['type'])}
if sess_persist['type'] in SESSION_PERSISTENCE_COOKIE_MAP:
if sess_persist.get('cookie_name'):
persistence['cookieName'] = sess_persist['cookie_name']
else:
persistence['cookieName'] = 'default_cookie_name'
persistence['cookieMode'] = SESSION_PERSISTENCE_COOKIE_MAP.get(
sess_persist['type'])
vcns_app_profile['persistence'] = persistence
return vcns_app_profile
def create_vip(self, context, edge_id, vip): def create_vip(self, context, edge_id, vip):
app_profile = self._convert_app_profile( app_profile = self._convert_app_profile(
vip['name'], vip.get('session_persistence')) vip['name'], (vip.get('session_persistence') or {}),
vip.get('protocol'))
try: try:
header, response = self.vcns.create_app_profile( header, response = self.vcns.create_app_profile(
edge_id, app_profile) edge_id, app_profile)
@ -156,6 +190,7 @@ class EdgeLbDriver():
with excutils.save_and_reraise_exception(): with excutils.save_and_reraise_exception():
LOG.exception(_("Failed to create vip on vshield edge: %s"), LOG.exception(_("Failed to create vip on vshield edge: %s"),
edge_id) edge_id)
self.vcns.delete_app_profile(edge_id, app_profileid)
objuri = header['location'] objuri = header['location']
vip_vseid = objuri[objuri.rfind("/") + 1:] vip_vseid = objuri[objuri.rfind("/") + 1:]
@ -168,6 +203,18 @@ class EdgeLbDriver():
} }
vcns_db.add_vcns_edge_vip_binding(context.session, map_info) vcns_db.add_vcns_edge_vip_binding(context.session, map_info)
def _get_vip_binding(self, session, id):
vip_binding = vcns_db.get_vcns_edge_vip_binding(session, id)
if not vip_binding:
msg = (_("vip_binding not found with id: %(id)s "
"edge_id: %(edge_id)s") % {
'id': id,
'edge_id': vip_binding[vcns_const.EDGE_ID]})
LOG.error(msg)
raise vcns_exc.VcnsNotFound(
resource='router_service_binding', msg=msg)
return vip_binding
def get_vip(self, context, id): def get_vip(self, context, id):
vip_binding = vcns_db.get_vcns_edge_vip_binding(context.session, id) vip_binding = vcns_db.get_vcns_edge_vip_binding(context.session, id)
edge_id = vip_binding[vcns_const.EDGE_ID] edge_id = vip_binding[vcns_const.EDGE_ID]
@ -179,33 +226,53 @@ class EdgeLbDriver():
LOG.exception(_("Failed to get vip on edge")) LOG.exception(_("Failed to get vip on edge"))
return self._restore_lb_vip(context, edge_id, response) return self._restore_lb_vip(context, edge_id, response)
def update_vip(self, context, vip): def update_vip(self, context, vip, session_persistence_update=True):
vip_binding = vcns_db.get_vcns_edge_vip_binding( vip_binding = self._get_vip_binding(context.session, vip['id'])
context.session, vip['id'])
edge_id = vip_binding[vcns_const.EDGE_ID] edge_id = vip_binding[vcns_const.EDGE_ID]
vip_vseid = vip_binding.get('vip_vseid') vip_vseid = vip_binding.get('vip_vseid')
if session_persistence_update:
app_profileid = vip_binding.get('app_profileid') app_profileid = vip_binding.get('app_profileid')
app_profile = self._convert_app_profile(
vip['name'], vip.get('session_persistence', {}),
vip.get('protocol'))
try:
self.vcns.update_app_profile(
edge_id, app_profileid, app_profile)
except vcns_exc.VcnsApiException:
with excutils.save_and_reraise_exception():
LOG.exception(_("Failed to update app profile on "
"edge: %s") % edge_id)
vip_new = self._convert_lb_vip(context, edge_id, vip, app_profileid) vip_new = self._convert_lb_vip(context, edge_id, vip, app_profileid)
try: try:
self.vcns.update_vip(edge_id, vip_vseid, vip_new) self.vcns.update_vip(edge_id, vip_vseid, vip_new)
except vcns_exc.VcnsApiException: except vcns_exc.VcnsApiException:
with excutils.save_and_reraise_exception(): with excutils.save_and_reraise_exception():
LOG.exception(_("Failed to update vip on edge: %s"), edge_id) LOG.exception(_("Failed to update vip on edge: %s") % edge_id)
def delete_vip(self, context, id): def delete_vip(self, context, id):
vip_binding = vcns_db.get_vcns_edge_vip_binding( vip_binding = self._get_vip_binding(context.session, id)
context.session, id)
edge_id = vip_binding[vcns_const.EDGE_ID] edge_id = vip_binding[vcns_const.EDGE_ID]
vip_vseid = vip_binding['vip_vseid'] vip_vseid = vip_binding['vip_vseid']
app_profileid = vip_binding['app_profileid'] app_profileid = vip_binding['app_profileid']
try: try:
self.vcns.delete_vip(edge_id, vip_vseid) self.vcns.delete_vip(edge_id, vip_vseid)
self.vcns.delete_app_profile(edge_id, app_profileid) except vcns_exc.ResourceNotFound:
LOG.exception(_("vip not found on edge: %s") % edge_id)
except vcns_exc.VcnsApiException: except vcns_exc.VcnsApiException:
with excutils.save_and_reraise_exception(): with excutils.save_and_reraise_exception():
LOG.exception(_("Failed to delete vip on edge: %s"), edge_id) LOG.exception(_("Failed to delete vip on edge: %s") % edge_id)
try:
self.vcns.delete_app_profile(edge_id, app_profileid)
except vcns_exc.ResourceNotFound:
LOG.exception(_("app profile not found on edge: %s") % edge_id)
except vcns_exc.VcnsApiException:
with excutils.save_and_reraise_exception():
LOG.exception(_("Failed to delete app profile on edge: %s") %
edge_id)
vcns_db.delete_vcns_edge_vip_binding(context.session, id) vcns_db.delete_vcns_edge_vip_binding(context.session, id)
def create_pool(self, context, edge_id, pool, members): def create_pool(self, context, edge_id, pool, members):

View File

@ -260,6 +260,12 @@ class Vcns(object):
APP_PROFILE_RESOURCE) APP_PROFILE_RESOURCE)
return self.do_request(HTTP_POST, uri, app_profile) return self.do_request(HTTP_POST, uri, app_profile)
def update_app_profile(self, edge_id, app_profileid, app_profile):
uri = self._build_uri_path(
edge_id, LOADBALANCER_SERVICE,
APP_PROFILE_RESOURCE, app_profileid)
return self.do_request(HTTP_PUT, uri, app_profile)
def delete_app_profile(self, edge_id, app_profileid): def delete_app_profile(self, edge_id, app_profileid):
uri = self._build_uri_path( uri = self._build_uri_path(
edge_id, LOADBALANCER_SERVICE, edge_id, LOADBALANCER_SERVICE,

View File

@ -510,6 +510,17 @@ class FakeVcns(object):
response = "" response = ""
return self.return_helper(header, response) return self.return_helper(header, response)
def update_app_profile(self, edge_id, app_profileid, app_profile):
header = {'status': 404}
response = ""
if not self._fake_app_profiles_dict.get(edge_id) or (
not self._fake_app_profiles_dict[edge_id].get(app_profileid)):
return self.return_helper(header, response)
header = {'status': 204}
self._fake_app_profiles_dict[edge_id][app_profileid].update(
app_profile)
return self.return_helper(header, response)
def delete_app_profile(self, edge_id, app_profileid): def delete_app_profile(self, edge_id, app_profileid):
header = {'status': 404} header = {'status': 404}
response = "" response = ""

View File

@ -15,6 +15,7 @@
import contextlib import contextlib
import testtools
from webob import exc as web_exc from webob import exc as web_exc
from neutron.api.v2 import attributes from neutron.api.v2 import attributes
@ -83,6 +84,8 @@ class TestLoadbalancerPlugin(
self.fc2.delete_health_monitor) self.fc2.delete_health_monitor)
instance.return_value.create_app_profile.side_effect = ( instance.return_value.create_app_profile.side_effect = (
self.fc2.create_app_profile) self.fc2.create_app_profile)
instance.return_value.update_app_profile.side_effect = (
self.fc2.update_app_profile)
instance.return_value.delete_app_profile.side_effect = ( instance.return_value.delete_app_profile.side_effect = (
self.fc2.delete_app_profile) self.fc2.delete_app_profile)
@ -189,6 +192,15 @@ class TestLoadbalancerPlugin(
expected expected
) )
def test_create_vip_with_session_persistence(self):
self.test_create_vip(session_persistence={'type': 'HTTP_COOKIE'})
def test_create_vip_with_invalid_persistence_method(self):
with testtools.ExpectedException(web_exc.HTTPClientError):
self.test_create_vip(
protocol='TCP',
session_persistence={'type': 'HTTP_COOKIE'})
def test_update_vip(self): def test_update_vip(self):
name = 'new_vip' name = 'new_vip'
router_id = self._create_and_get_router() router_id = self._create_and_get_router()

View File

@ -22,6 +22,7 @@ from neutron.openstack.common import uuidutils
from neutron.plugins.vmware.dbexts import vcns_db from neutron.plugins.vmware.dbexts import vcns_db
from neutron.plugins.vmware.vshield.common import exceptions as vcns_exc from neutron.plugins.vmware.vshield.common import exceptions as vcns_exc
from neutron.plugins.vmware.vshield import vcns_driver from neutron.plugins.vmware.vshield import vcns_driver
from neutron.services.loadbalancer import constants as lb_constants
from neutron.tests.unit.db.loadbalancer import test_db_loadbalancer from neutron.tests.unit.db.loadbalancer import test_db_loadbalancer
from neutron.tests.unit.vmware import get_fake_conf from neutron.tests.unit.vmware import get_fake_conf
from neutron.tests.unit.vmware import VCNS_NAME from neutron.tests.unit.vmware import VCNS_NAME
@ -68,6 +69,8 @@ class VcnsDriverTestCase(test_db_loadbalancer.LoadBalancerPluginDbTestCase):
self.fc2.delete_health_monitor) self.fc2.delete_health_monitor)
instance.return_value.create_app_profile.side_effect = ( instance.return_value.create_app_profile.side_effect = (
self.fc2.create_app_profile) self.fc2.create_app_profile)
instance.return_value.update_app_profile.side_effect = (
self.fc2.update_app_profile)
instance.return_value.delete_app_profile.side_effect = ( instance.return_value.delete_app_profile.side_effect = (
self.fc2.delete_app_profile) self.fc2.delete_app_profile)
self.pool_id = None self.pool_id = None
@ -106,6 +109,94 @@ class TestEdgeLbDriver(VcnsDriverTestCase):
for k, v in vip_get.iteritems(): for k, v in vip_get.iteritems():
self.assertEqual(vip_create[k], v) self.assertEqual(vip_create[k], v)
def test_convert_app_profile(self):
app_profile_name = 'app_profile_name'
sess_persist1 = {'type': "SOURCE_IP"}
sess_persist2 = {'type': "HTTP_COOKIE"}
sess_persist3 = {'type': "APP_COOKIE",
'cookie_name': "app_cookie_name"}
# protocol is HTTP and type is SOURCE_IP
expect_vcns_app_profile1 = {
'insertXForwardedFor': False,
'name': app_profile_name,
'serverSslEnabled': False,
'sslPassthrough': False,
'template': lb_constants.PROTOCOL_HTTP,
'persistence': {'method': 'sourceip'}}
vcns_app_profile = self.driver._convert_app_profile(
app_profile_name, sess_persist1, lb_constants.PROTOCOL_HTTP)
for k, v in expect_vcns_app_profile1.iteritems():
self.assertEqual(vcns_app_profile[k], v)
# protocol is HTTP and type is HTTP_COOKIE and APP_COOKIE
expect_vcns_app_profile2 = {
'insertXForwardedFor': False,
'name': app_profile_name,
'serverSslEnabled': False,
'sslPassthrough': False,
'template': lb_constants.PROTOCOL_HTTP,
'persistence': {'method': 'cookie',
'cookieName': 'default_cookie_name',
'cookieMode': 'insert'}}
vcns_app_profile = self.driver._convert_app_profile(
app_profile_name, sess_persist2, lb_constants.PROTOCOL_HTTP)
for k, v in expect_vcns_app_profile2.iteritems():
self.assertEqual(vcns_app_profile[k], v)
expect_vcns_app_profile3 = {
'insertXForwardedFor': False,
'name': app_profile_name,
'serverSslEnabled': False,
'sslPassthrough': False,
'template': lb_constants.PROTOCOL_HTTP,
'persistence': {'method': 'cookie',
'cookieName': sess_persist3['cookie_name'],
'cookieMode': 'app'}}
vcns_app_profile = self.driver._convert_app_profile(
app_profile_name, sess_persist3, lb_constants.PROTOCOL_HTTP)
for k, v in expect_vcns_app_profile3.iteritems():
self.assertEqual(vcns_app_profile[k], v)
# protocol is HTTPS and type is SOURCE_IP
expect_vcns_app_profile1 = {
'insertXForwardedFor': False,
'name': app_profile_name,
'serverSslEnabled': False,
'sslPassthrough': True,
'template': lb_constants.PROTOCOL_HTTPS,
'persistence': {'method': 'sourceip'}}
vcns_app_profile = self.driver._convert_app_profile(
app_profile_name, sess_persist1, lb_constants.PROTOCOL_HTTPS)
for k, v in expect_vcns_app_profile1.iteritems():
self.assertEqual(vcns_app_profile[k], v)
# protocol is HTTPS, and type isn't SOURCE_IP
self.assertRaises(vcns_exc.VcnsBadRequest,
self.driver._convert_app_profile,
app_profile_name,
sess_persist2, lb_constants.PROTOCOL_HTTPS)
self.assertRaises(vcns_exc.VcnsBadRequest,
self.driver._convert_app_profile,
app_profile_name,
sess_persist3, lb_constants.PROTOCOL_HTTPS)
# protocol is TCP and type is SOURCE_IP
expect_vcns_app_profile1 = {
'insertXForwardedFor': False,
'name': app_profile_name,
'serverSslEnabled': False,
'sslPassthrough': False,
'template': lb_constants.PROTOCOL_TCP,
'persistence': {'method': 'sourceip'}}
vcns_app_profile = self.driver._convert_app_profile(
app_profile_name, sess_persist1, lb_constants.PROTOCOL_TCP)
for k, v in expect_vcns_app_profile1.iteritems():
self.assertEqual(vcns_app_profile[k], v)
# protocol is TCP, and type isn't SOURCE_IP
self.assertRaises(vcns_exc.VcnsBadRequest,
self.driver._convert_app_profile,
app_profile_name,
sess_persist2, lb_constants.PROTOCOL_TCP)
self.assertRaises(vcns_exc.VcnsBadRequest,
self.driver._convert_app_profile,
app_profile_name,
sess_persist3, lb_constants.PROTOCOL_TCP)
def test_update_vip(self): def test_update_vip(self):
ctx = context.get_admin_context() ctx = context.get_admin_context()
with self.pool(no_delete=True) as pool: with self.pool(no_delete=True) as pool: