Refactoring of ZMQ pubsub

To work around the ZMQ issues (publisher binds to a port, thus only one
publisher can work on a given IP), we had a lot of junk code in our
repo.
Removed all redundant mechanisms (use_multiproc, is_neutron_server) and
modified to have 2 types of configurable ZMQ drivers:
1. zmq_remote_pubsub_driver - Has TCP publisher and IPC subscriber, This
   should be used in the publisher service.
2. zmq_pubsub_driver - Has IPC publisher and TCP subscriberi, This
   should be used in all other uses.

The way to set the one to use is via configuration, thus it is up to the
deployment to make sure this is configured correctly.

Change-Id: Ibf7894e608187e87bdeb7774749bfa0cc15eae56
This commit is contained in:
Shachar Snapiri 2018-06-10 16:45:36 +03:00 committed by Omer Anson
parent 4ae2336114
commit bbe1347e4e
35 changed files with 135 additions and 260 deletions

View File

@ -28,6 +28,7 @@ if is_service_enabled df-metadata ; then
fi fi
DRAGONFLOW_CONF=/etc/neutron/dragonflow.ini DRAGONFLOW_CONF=/etc/neutron/dragonflow.ini
DRAGONFLOW_PUBLISHER_CONF=/etc/neutron/dragonflow_publisher.ini
DRAGONFLOW_DATAPATH=/etc/neutron/dragonflow_datapath_layout.yaml DRAGONFLOW_DATAPATH=/etc/neutron/dragonflow_datapath_layout.yaml
Q_PLUGIN_EXTRA_CONF_PATH=/etc/neutron Q_PLUGIN_EXTRA_CONF_PATH=/etc/neutron
Q_PLUGIN_EXTRA_CONF_FILES=(dragonflow.ini) Q_PLUGIN_EXTRA_CONF_FILES=(dragonflow.ini)

View File

@ -155,13 +155,11 @@ fi
if is_service_enabled df-etcd-pubsub-service ; then if is_service_enabled df-etcd-pubsub-service ; then
init_pubsub init_pubsub
DF_PUB_SUB_USE_MULTIPROC="False"
source $DEST/dragonflow/devstack/etcd_pubsub_driver source $DEST/dragonflow/devstack/etcd_pubsub_driver
fi fi
if [[ "$DF_REDIS_PUBSUB" == "True" ]]; then if [[ "$DF_REDIS_PUBSUB" == "True" ]]; then
init_pubsub init_pubsub
DF_PUB_SUB_USE_MULTIPROC="False"
source $DEST/dragonflow/devstack/redis_pubsub_driver source $DEST/dragonflow/devstack/redis_pubsub_driver
fi fi
@ -316,8 +314,7 @@ function configure_df_plugin {
iniset $DRAGONFLOW_CONF df apps_list "$DF_APPS_LIST" iniset $DRAGONFLOW_CONF df apps_list "$DF_APPS_LIST"
iniset $DRAGONFLOW_CONF df_l2_app l2_responder "$DF_L2_RESPONDER" iniset $DRAGONFLOW_CONF df_l2_app l2_responder "$DF_L2_RESPONDER"
iniset $DRAGONFLOW_CONF df enable_df_pub_sub "$DF_PUB_SUB" iniset $DRAGONFLOW_CONF df enable_df_pub_sub "$DF_PUB_SUB"
iniset $DRAGONFLOW_CONF df pub_sub_use_multiproc "$DF_PUB_SUB_USE_MULTIPROC" iniset $DRAGONFLOW_CONF df_zmq ipc_socket "$DF_ZMQ_IPC_SOCKET"
iniset $DRAGONFLOW_CONF df publisher_multiproc_socket "$DF_PUBLISHER_MULTIPROC_SOCKET"
if [[ ! -z ${EXTERNAL_HOST_IP} ]]; then if [[ ! -z ${EXTERNAL_HOST_IP} ]]; then
iniset $DRAGONFLOW_CONF df external_host_ip "$EXTERNAL_HOST_IP" iniset $DRAGONFLOW_CONF df external_host_ip "$EXTERNAL_HOST_IP"
iniset $DRAGONFLOW_CONF df_snat_app external_network_bridge "$PUBLIC_BRIDGE" iniset $DRAGONFLOW_CONF df_snat_app external_network_bridge "$PUBLIC_BRIDGE"
@ -483,9 +480,9 @@ function verify_ryu_version {
} }
function start_pubsub_service { function start_pubsub_service {
echo "Starting Dragonflow publisher service"
if is_service_enabled df-publisher-service ; then if is_service_enabled df-publisher-service ; then
run_process df-publisher-service "$DF_PUBLISHER_SERVICE_BINARY --config-file $NEUTRON_CONF --config-file $DRAGONFLOW_CONF" echo "Starting Dragonflow publisher service"
run_process df-publisher-service "$DF_PUBLISHER_SERVICE_BINARY --config-file $NEUTRON_CONF --config-file $DRAGONFLOW_CONF --config-file $DRAGONFLOW_PUBLISHER_CONF"
fi fi
} }
@ -587,7 +584,6 @@ function handle_df_stack_post_install {
if [ -z $PUB_SUB_DRIVER ]; then if [ -z $PUB_SUB_DRIVER ]; then
die $LINENO "pub-sub enabled, but no pub-sub driver selected" die $LINENO "pub-sub enabled, but no pub-sub driver selected"
fi fi
PUB_SUB_MULTIPROC_DRIVER=${PUB_SUB_MULTIPROC_DRIVER:-$PUB_SUB_DRIVER}
fi fi
if is_service_enabled nova; then if is_service_enabled nova; then

View File

@ -11,7 +11,7 @@ DF_INSTALL_DEBUG_ROOTWRAP_CONF=${DF_INSTALL_DEBUG_ROOTWRAP_CONF:-"True"}
DF_L3_BINARY=$NEUTRON_BIN_DIR/df-l3-agent DF_L3_BINARY=$NEUTRON_BIN_DIR/df-l3-agent
DF_LOCAL_CONTROLLER_BINARY=$NEUTRON_BIN_DIR/df-local-controller DF_LOCAL_CONTROLLER_BINARY=$NEUTRON_BIN_DIR/df-local-controller
DF_PUBLISHER_SERVICE_BINARY=$NEUTRON_BIN_DIR/df-publisher-service DF_PUBLISHER_SERVICE_BINARY=$NEUTRON_BIN_DIR/df-publisher-service
DF_PUBLISHER_MULTIPROC_SOCKET=${DF_PUBLISHER_MULTIPROC_SOCKET:-"/var/run/zmq_pubsub/zmq-publisher-socket"} DF_ZMQ_IPC_SOCKET=${DF_ZMQ_IPC_SOCKET:-"/var/run/zmq_pubsub/zmq-publisher-socket"}
DF_AUTO_DETECT_PORT_BEHIND_PORT=${DF_AUTO_DETECT_PORT_BEHIND_PORT:-"False"} DF_AUTO_DETECT_PORT_BEHIND_PORT=${DF_AUTO_DETECT_PORT_BEHIND_PORT:-"False"}
DF_LBAAS_AUTO_ENABLE_VIP_PORTS=${DF_LBAAS_AUTO_ENABLE_VIP_PORTS:-"True"} DF_LBAAS_AUTO_ENABLE_VIP_PORTS=${DF_LBAAS_AUTO_ENABLE_VIP_PORTS:-"True"}
@ -36,7 +36,6 @@ DF_L2_RESPONDER=${DF_L2_RESPONDER:-'True'}
DF_MONITOR_TABLE_POLL_TIME=${DF_MONITOR_TABLE_POLL_TIME:-30} DF_MONITOR_TABLE_POLL_TIME=${DF_MONITOR_TABLE_POLL_TIME:-30}
DF_PUB_SUB=${DF_PUB_SUB:-"False"} DF_PUB_SUB=${DF_PUB_SUB:-"False"}
DF_PUB_SUB_USE_MULTIPROC=${DF_PUB_SUB_USE_MULTIPROC:-"True"}
DF_Q_SVC_MASTER=${DF_Q_SVC_MASTER:-"True"} DF_Q_SVC_MASTER=${DF_Q_SVC_MASTER:-"True"}
PUBLISHER_RATE_LIMIT_TIMEOUT=${PUBLISHER_RATE_LIMIT_TIMEOUT:-180} PUBLISHER_RATE_LIMIT_TIMEOUT=${PUBLISHER_RATE_LIMIT_TIMEOUT:-180}

View File

@ -1,15 +1,13 @@
#!/bin/bash #!/bin/bash
ZMQ_IPC_SOCKET=$DF_PUBLISHER_MULTIPROC_SOCKET
function configure_pubsub_service_plugin { function configure_pubsub_service_plugin {
NEUTRON_CONF=${NEUTRON_CONF:-"/etc/neutron/neutron.conf"} NEUTRON_CONF=${NEUTRON_CONF:-"/etc/neutron/neutron.conf"}
PUB_SUB_DRIVER=${PUB_SUB_DRIVER:-"zmq_pubsub_driver"} PUB_SUB_DRIVER=${PUB_SUB_DRIVER:-"zmq_pubsub_driver"}
PUB_SUB_MULTIPROC_DRIVER=${PUB_SUB_MULTIPROC_DRIVER:-"zmq_pubsub_multiproc_driver"}
iniset $DRAGONFLOW_CONF df pub_sub_driver $PUB_SUB_DRIVER iniset $DRAGONFLOW_CONF df pub_sub_driver $PUB_SUB_DRIVER
iniset $DRAGONFLOW_CONF df pub_sub_multiproc_driver $PUB_SUB_MULTIPROC_DRIVER DF_PUBLISHER_DRIVER=${DF_PUBLISHER_DRIVER:-"zmq_bind_pubsub_driver"}
iniset $DRAGONFLOW_PUBLISHER_CONF df pub_sub_driver $DF_PUBLISHER_DRIVER
ZMQ_IPC_SOCKET_DIR=`dirname $ZMQ_IPC_SOCKET` ZMQ_IPC_SOCKET_DIR=`dirname $DF_ZMQ_IPC_SOCKET`
sudo mkdir -p $ZMQ_IPC_SOCKET_DIR sudo mkdir -p $ZMQ_IPC_SOCKET_DIR
sudo chown $STACK_USER $ZMQ_IPC_SOCKET_DIR sudo chown $STACK_USER $ZMQ_IPC_SOCKET_DIR
} }

View File

@ -130,7 +130,7 @@ Next you need to change the configuration, for example, etcd:
Pub/Sub Driver Pub/Sub Driver
-------------- --------------
Dragonflow supports zeromq and redis. You need to change the configuration, for example, zeromq: Dragonflow supports etcd, redis and zeromq. You need to change the configuration, for example, etcd:
/etc/neutron/dragonflow.ini: /etc/neutron/dragonflow.ini:
@ -138,10 +138,7 @@ Dragonflow supports zeromq and redis. You need to change the configuration, for
[df] [df]
enable_df_pub_sub = True enable_df_pub_sub = True
pub_sub_driver = zmq_pubsub_driver pub_sub_driver = etcd_pubsub_driver
publisher_multiproc_socket = /var/run/zmq_pubsub/zmq-publisher-socket
pub_sub_multiproc_driver = zmq_pubsub_multiproc_driver
pub_sub_use_multiproc = True
publisher_rate_limit_count = 1 publisher_rate_limit_count = 1
publisher_rate_limit_timeout = 180 publisher_rate_limit_timeout = 180
monitor_table_poll_time = 30 monitor_table_poll_time = 30

View File

@ -76,36 +76,21 @@ __ _COMMON_PARAMS
1. pub_sub_driver - The alias to the class implementing ``PubSubApi`` for 1. pub_sub_driver - The alias to the class implementing ``PubSubApi`` for
network-based pub/sub. network-based pub/sub.
2. pub_sub_multiproc_driver - The alias to the class implementing ``PubSubApi`` 2. publisher_port - The port to which the network publisher should bind. It is
for IPC-based pub/sub.
3. publisher_port - The port to which the network publisher should bind. It is
also the port the network subscribers connect. also the port the network subscribers connect.
4. publisher_transport - The transport protocol (e.g. TCP, UDP) over which 3. publisher_transport - The transport protocol (e.g. TCP, UDP) over which
pub/sub netwrok communication is passed. pub/sub netwrok communication is passed.
5. publisher_bind_address - The local address to which the network publisher 4. publisher_bind_address - The local address to which the network publisher
should bind. '*' means all addresses. should bind. '*' means all addresses.
6. publisher_multiproc_socket - The local socket over which the multi-proc
pub/sub implementation should communicate. The actual value is
implementation specific, since different implementations may use different
IPC mechanisms.
Some publish-subscribe drivers do not need to use a publisher service. Some publish-subscribe drivers do not need to use a publisher service.
This can be the case if e.g. the publisher does not bind to the communication This can be the case if e.g. the publisher does not bind to the communication
socket. socket.
In this case, the pub_sub_multiproc_driver and publisher_multiproc_socket All publishers are created using the pub_sub_driver.
options are ignored. All publishers are created using the pub_sub_driver.
In case this is what you want, disable the following option.
1. pub_sub_use_multiproc - Use inter-process publish/subscribe. Publishers
send events via the publisher service. When disabled, publishers send
events directly to the network.
======================== ========================
Reference Implementation Reference Implementation

View File

@ -430,7 +430,7 @@ def main():
df_utils.config_parse() df_utils.config_parse()
global nb_api global nb_api
nb_api = api_nb.NbApi.get_instance(False) nb_api = api_nb.NbApi.get_instance()
args.handle(args) args.handle(args)

View File

@ -279,7 +279,7 @@ def start(is_service):
"""main method""" """main method"""
df_config.init(sys.argv) df_config.init(sys.argv)
df_utils.config_parse() df_utils.config_parse()
nb_api = api_nb.NbApi.get_instance(False) nb_api = api_nb.NbApi.get_instance()
if is_service: if is_service:
df_service.register_service('df-skydive-service', nb_api) df_service.register_service('df-skydive-service', nb_api)
service_manager = cotyledon.ServiceManager() service_manager = cotyledon.ServiceManager()

View File

@ -81,7 +81,7 @@ def main():
config.init(sys.argv[1:]) config.init(sys.argv[1:])
config.setup_logging() config.setup_logging()
environment_setup() environment_setup()
nb_api = api_nb.NbApi.get_instance(False) nb_api = api_nb.NbApi.get_instance()
service_instance = metadata_service.DFMetadataProxyHandler( service_instance = metadata_service.DFMetadataProxyHandler(
cfg.CONF, nb_api) cfg.CONF, nb_api)
df_service.register_service('df-metadata-service', nb_api) df_service.register_service('df-metadata-service', nb_api)

View File

@ -27,6 +27,7 @@ from dragonflow.conf import df_redis
from dragonflow.conf import df_ryu from dragonflow.conf import df_ryu
from dragonflow.conf import df_skydive from dragonflow.conf import df_skydive
from dragonflow.conf import df_snat from dragonflow.conf import df_snat
from dragonflow.conf import df_zmq
CONF = cfg.CONF CONF = cfg.CONF
@ -42,6 +43,7 @@ df_l2.register_opts()
df_l3.register_opts() df_l3.register_opts()
df_dnat.register_opts() df_dnat.register_opts()
df_redis.register_opts() df_redis.register_opts()
df_zmq.register_opts()
df_ryu.register_opts() df_ryu.register_opts()
df_provider_networks.register_opts() df_provider_networks.register_opts()
df_snat.register_opts() df_snat.register_opts()

View File

@ -62,9 +62,6 @@ df_opts = [
cfg.StrOpt('pub_sub_driver', cfg.StrOpt('pub_sub_driver',
default='zmq_pubsub_driver', default='zmq_pubsub_driver',
help=_('Drivers to use for the Dragonflow pub/sub')), help=_('Drivers to use for the Dragonflow pub/sub')),
cfg.StrOpt('pub_sub_multiproc_driver',
default='zmq_pubsub_multiproc_driver',
help=_('Drivers to use for the Dragonflow pub/sub')),
cfg.BoolOpt('enable_neutron_notifier', cfg.BoolOpt('enable_neutron_notifier',
default=False, default=False,
help=_('Enable notifier for Dragonflow controller sending ' help=_('Enable notifier for Dragonflow controller sending '
@ -84,19 +81,6 @@ df_opts = [
cfg.StrOpt('publisher_bind_address', cfg.StrOpt('publisher_bind_address',
default='*', default='*',
help=_('Neutron Server Publishers bind address')), help=_('Neutron Server Publishers bind address')),
cfg.BoolOpt(
'pub_sub_use_multiproc',
default=True,
help=_(
'Use inter-process publish/subscribe. '
'Publishers send events via the publisher service.'
)
),
cfg.StrOpt(
'publisher_multiproc_socket',
default='/var/run/zmq_pubsub/zmq-publisher-socket',
help=_('Neutron Server Publisher inter-process socket address')
),
cfg.IntOpt( cfg.IntOpt(
'publisher_timeout', 'publisher_timeout',
default=300, default=300,

36
dragonflow/conf/df_zmq.py Normal file
View File

@ -0,0 +1,36 @@
# Copyright (c) 2016 OpenStack Foundation.
# 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 oslo_config import cfg
from dragonflow._i18n import _
df_zmq_opts = [
cfg.StrOpt(
'ipc_socket',
default='/var/run/zmq_pubsub/zmq-publisher-socket',
help=_('Neutron Server Publisher inter-process socket address')
),
]
def register_opts():
cfg.CONF.register_opts(df_zmq_opts, group='df_zmq')
def list_opts():
return {'df_zmq': df_zmq_opts}

View File

@ -165,7 +165,7 @@ class BGPService(service.Service):
def main(): def main():
df_config.init(sys.argv) df_config.init(sys.argv)
nb_api = api_nb.NbApi.get_instance(False) nb_api = api_nb.NbApi.get_instance()
server = BGPService(nb_api) server = BGPService(nb_api)
df_service.register_service('df-bgp-service', nb_api) df_service.register_service('df-bgp-service', nb_api)
service.launch(cfg.CONF, server).wait() service.launch(cfg.CONF, server).wait()

View File

@ -473,7 +473,7 @@ def main():
df_config.init(sys.argv) df_config.init(sys.argv)
init_ryu_config() init_ryu_config()
nb_api = api_nb.NbApi.get_instance(False) nb_api = api_nb.NbApi.get_instance()
controller = DfLocalController(chassis_name, nb_api) controller = DfLocalController(chassis_name, nb_api)
service.register_service('df-local-controller', nb_api) service.register_service('df-local-controller', nb_api)
controller.run() controller.run()

View File

@ -42,7 +42,7 @@ class PublisherService(object):
def __init__(self, nb_api): def __init__(self, nb_api):
self._queue = queue.Queue() self._queue = queue.Queue()
self.publisher = _get_publisher() self.publisher = _get_publisher()
self.multiproc_subscriber = self._get_multiproc_subscriber() self.subscriber = self._get_subscriber()
self.nb_api = nb_api self.nb_api = nb_api
self.db = self.nb_api.driver self.db = self.nb_api.driver
self.uuid = pub_sub_api.generate_publisher_uuid() self.uuid = pub_sub_api.generate_publisher_uuid()
@ -51,21 +51,18 @@ class PublisherService(object):
cfg.CONF.df.publisher_rate_limit_timeout, cfg.CONF.df.publisher_rate_limit_timeout,
) )
def _get_multiproc_subscriber(self): def _get_subscriber(self):
""" """
Return the subscriber for inter-process communication. If multi-proc Return the subscriber for inter-process communication. If multi-proc
communication is not use (i.e. disabled from config), return None. communication is not use (i.e. disabled from config), return None.
""" """
if not cfg.CONF.df.pub_sub_use_multiproc:
return None
pub_sub_driver = df_utils.load_driver( pub_sub_driver = df_utils.load_driver(
cfg.CONF.df.pub_sub_multiproc_driver, cfg.CONF.df.pub_sub_driver,
df_utils.DF_PUBSUB_DRIVER_NAMESPACE) df_utils.DF_PUBSUB_DRIVER_NAMESPACE)
return pub_sub_driver.get_subscriber() return pub_sub_driver.get_subscriber()
def initialize(self): def initialize(self):
if self.multiproc_subscriber: self.subscriber.initialize(self._append_event_to_queue)
self.multiproc_subscriber.initialize(self._append_event_to_queue)
self.publisher.initialize() self.publisher.initialize()
def _append_event_to_queue(self, table, key, action, value, topic): def _append_event_to_queue(self, table, key, action, value, topic):
@ -74,8 +71,7 @@ class PublisherService(object):
time.sleep(0) time.sleep(0)
def run(self): def run(self):
if self.multiproc_subscriber: self.subscriber.daemonize()
self.multiproc_subscriber.daemonize()
self._register_as_publisher() self._register_as_publisher()
self._start_db_table_monitors() self._start_db_table_monitors()
while True: while True:
@ -155,7 +151,7 @@ def main():
# PATCH(snapiri): Disable pub_sub as it creates a publisher in nb_api # PATCH(snapiri): Disable pub_sub as it creates a publisher in nb_api
# which collides with the publisher we create here. # which collides with the publisher we create here.
cfg.CONF.set_override('enable_df_pub_sub', False, group='df') cfg.CONF.set_override('enable_df_pub_sub', False, group='df')
nb_api = api_nb.NbApi.get_instance(False) nb_api = api_nb.NbApi.get_instance()
service = PublisherService(nb_api) service = PublisherService(nb_api)
df_service.register_service('df-publisher-service', nb_api) df_service.register_service('df-publisher-service', nb_api)
service.initialize() service.initialize()

View File

@ -55,35 +55,24 @@ def _get_topic(obj):
class NbApi(object): class NbApi(object):
def __init__(self, db_driver, use_pubsub=False, is_neutron_server=False): def __init__(self, db_driver):
super(NbApi, self).__init__() super(NbApi, self).__init__()
self.driver = db_driver self.driver = db_driver
self.controller = None self.controller = None
self.use_pubsub = use_pubsub self.use_pubsub = cfg.CONF.df.enable_df_pub_sub
self.publisher = None self.publisher = None
self.subscriber = None self.subscriber = None
self.is_neutron_server = is_neutron_server
self.enable_selective_topo_dist = \ self.enable_selective_topo_dist = \
cfg.CONF.df.enable_selective_topology_distribution cfg.CONF.df.enable_selective_topology_distribution
self.pub_sub_use_multiproc = False
if self.is_neutron_server:
# multiproc pub/sub is only supported in neutron server
self.pub_sub_use_multiproc = cfg.CONF.df.pub_sub_use_multiproc
@staticmethod @staticmethod
def get_instance(is_neutron_server): def get_instance():
global _nb_api global _nb_api
if _nb_api is None: if _nb_api is None:
nb_driver = df_utils.load_driver( nb_driver = df_utils.load_driver(
cfg.CONF.df.nb_db_class, cfg.CONF.df.nb_db_class,
df_utils.DF_NB_DB_DRIVER_NAMESPACE) df_utils.DF_NB_DB_DRIVER_NAMESPACE)
# Do not use pubsub for external apps - this causes issues with nb_api = NbApi(nb_driver)
# threads and other issues.
use_pubsub = cfg.CONF.df.enable_df_pub_sub
nb_api = NbApi(
nb_driver,
use_pubsub=use_pubsub,
is_neutron_server=is_neutron_server)
ip, port = get_db_ip_port() ip, port = get_db_ip_port()
nb_api._initialize(db_ip=ip, db_port=port) nb_api._initialize(db_ip=ip, db_port=port)
_nb_api = nb_api _nb_api = nb_api
@ -94,25 +83,17 @@ class NbApi(object):
if self.use_pubsub: if self.use_pubsub:
self.publisher = self._get_publisher() self.publisher = self._get_publisher()
self.subscriber = self._get_subscriber() self.subscriber = self._get_subscriber()
if self.is_neutron_server:
self.publisher.initialize() self.publisher.initialize()
# Start a thread to detect DB failover in Plugin # Start a thread to detect DB failover in Plugin
self.publisher.set_publisher_for_failover( self.publisher.set_publisher_for_failover(
self.publisher, self.publisher,
self.db_recover_callback) self.db_recover_callback)
self.publisher.start_detect_for_failover() self.publisher.start_detect_for_failover()
self.driver.set_neutron_server(True)
else:
# FIXME(nick-ma-z): if active-detection is enabled,
# we initialize the publisher here. Make sure it
# only supports redis-based pub/sub driver.
if "active_port_detection" in cfg.CONF.df.apps_list:
self.publisher.initialize()
def set_db_change_callback(self, db_change_callback): def set_db_change_callback(self, db_change_callback):
if self.use_pubsub and not self.is_neutron_server: if self.use_pubsub:
# NOTE(gampel) we want to start queuing event as soon # This is here to not allow multiple subscribers to be started
# as possible # under the same process. One should be more than enough.
if not self.subscriber.is_running: if not self.subscriber.is_running:
self._start_subscriber(db_change_callback) self._start_subscriber(db_change_callback)
# Register for DB Failover detection in NB Plugin # Register for DB Failover detection in NB Plugin
@ -134,16 +115,11 @@ class NbApi(object):
self.driver.process_ha() self.driver.process_ha()
self.publisher.process_ha() self.publisher.process_ha()
self.subscriber.process_ha() self.subscriber.process_ha()
if not self.is_neutron_server:
self.controller.sync() self.controller.sync()
def _get_publisher(self): def _get_publisher(self):
if self.pub_sub_use_multiproc:
pubsub_driver_name = cfg.CONF.df.pub_sub_multiproc_driver
else:
pubsub_driver_name = cfg.CONF.df.pub_sub_driver
pub_sub_driver = df_utils.load_driver( pub_sub_driver = df_utils.load_driver(
pubsub_driver_name, cfg.CONF.df.pub_sub_driver,
df_utils.DF_PUBSUB_DRIVER_NAMESPACE) df_utils.DF_PUBSUB_DRIVER_NAMESPACE)
return pub_sub_driver.get_publisher() return pub_sub_driver.get_publisher()

View File

@ -163,10 +163,3 @@ class DbApi(object):
:returns: None :returns: None
""" """
@abc.abstractmethod
def set_neutron_server(self, is_neutron_server):
"""Set neutron server flag
:returns: None
"""

View File

@ -193,6 +193,3 @@ class CassandraDbDriver(db_api.DbApi):
def process_ha(self): def process_ha(self):
pass pass
def set_neutron_server(self, is_neutron_server):
pass

View File

@ -178,7 +178,3 @@ class EtcdDbDriver(db_api.DbApi):
def process_ha(self): def process_ha(self):
# Not needed in etcd # Not needed in etcd
pass pass
def set_neutron_server(self, is_neutron_server):
# Not needed in etcd
pass

View File

@ -97,7 +97,3 @@ class RamCloudDbDriver(db_api.DbApi):
def process_ha(self): def process_ha(self):
# Not needed in rmc # Not needed in rmc
pass pass
def set_neutron_server(self, is_neutron_server):
# Not needed in rmc
pass

View File

@ -394,6 +394,3 @@ class RedisDbDriver(db_api.DbApi):
def process_ha(self): def process_ha(self):
pass pass
def set_neutron_server(self, is_neutron_server):
pass

View File

@ -134,6 +134,3 @@ class RethinkDbDriver(db_api.DbApi):
def process_ha(self): def process_ha(self):
pass pass
def set_neutron_server(self, is_neutron_server):
pass # Not implemented

View File

@ -187,7 +187,3 @@ class ZookeeperDbDriver(db_api.DbApi):
def process_ha(self): def process_ha(self):
# Not needed in zookeeper # Not needed in zookeeper
pass pass
def set_neutron_server(self, is_neutron_server):
# Not needed in zookeeper
pass

View File

@ -55,5 +55,5 @@ class UniqueKey(mf.MixinBase):
# Relevant bp: # Relevant bp:
# https://blueprints.launchpad.net/dragonflow/+spec/pub-sub-v2 # https://blueprints.launchpad.net/dragonflow/+spec/pub-sub-v2
from dragonflow.db import api_nb from dragonflow.db import api_nb
nb_api = api_nb.NbApi.get_instance(True) nb_api = api_nb.NbApi.get_instance()
self.unique_key = nb_api.driver.allocate_unique_key(self.table_name) self.unique_key = nb_api.driver.allocate_unique_key(self.table_name)

View File

@ -12,6 +12,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 abc
import six
import traceback import traceback
import eventlet import eventlet
@ -27,41 +29,6 @@ LOG = logging.getLogger(__name__)
SUPPORTED_TRANSPORTS = set(['tcp', 'epgm']) SUPPORTED_TRANSPORTS = set(['tcp', 'epgm'])
class ZMQPubSub(pub_sub_api.PubSubApi):
def __init__(self):
super(ZMQPubSub, self).__init__()
transport = cfg.CONF.df.publisher_transport
if transport not in SUPPORTED_TRANSPORTS:
message = ("zmq_pub_sub: Unsupported publisher_transport value "
"%(transport)s, expected %(expected)s")
LOG.error(message, {
'transport': transport,
'expected': SUPPORTED_TRANSPORTS
})
raise exceptions.UnsupportedTransportException(transport=transport)
self.subscriber = ZMQSubscriberAgent()
self.publisher = ZMQPublisherAgent()
def get_publisher(self):
return self.publisher
def get_subscriber(self):
return self.subscriber
class ZMQPubSubMultiproc(pub_sub_api.PubSubApi):
def __init__(self):
super(ZMQPubSubMultiproc, self).__init__()
self.subscriber = ZMQSubscriberMultiprocAgent()
self.publisher = ZMQPublisherMultiprocAgent()
def get_publisher(self):
return self.publisher
def get_subscriber(self):
return self.subscriber
class ZMQPublisherAgentBase(pub_sub_api.PublisherAgentBase): class ZMQPublisherAgentBase(pub_sub_api.PublisherAgentBase):
def __init__(self): def __init__(self):
self.socket = None self.socket = None
@ -111,7 +78,7 @@ class ZMQPublisherMultiprocAgent(ZMQPublisherAgentBase):
def _connect(self): def _connect(self):
self.socket = self.context.socket(zmq.PUSH) self.socket = self.context.socket(zmq.PUSH)
ipc_socket = cfg.CONF.df.publisher_multiproc_socket ipc_socket = cfg.CONF.df_zmq.ipc_socket
LOG.debug("About to connect to IPC socket: %s", ipc_socket) LOG.debug("About to connect to IPC socket: %s", ipc_socket)
self.socket.connect('ipc://%s' % ipc_socket) self.socket.connect('ipc://%s' % ipc_socket)
@ -173,7 +140,7 @@ class ZMQSubscriberAgentBase(pub_sub_api.SubscriberAgentBase):
class ZMQSubscriberMultiprocAgent(ZMQSubscriberAgentBase): class ZMQSubscriberMultiprocAgent(ZMQSubscriberAgentBase):
def connect(self): def connect(self):
self.sub_socket = self.context.socket(zmq.PULL) self.sub_socket = self.context.socket(zmq.PULL)
ipc_socket = cfg.CONF.df.publisher_multiproc_socket ipc_socket = cfg.CONF.df_zmq.ipc_socket
LOG.debug("About to bind to IPC socket: %s", ipc_socket) LOG.debug("About to bind to IPC socket: %s", ipc_socket)
self.sub_socket.bind('ipc://%s' % ipc_socket) self.sub_socket.bind('ipc://%s' % ipc_socket)
@ -187,3 +154,42 @@ class ZMQSubscriberAgent(ZMQSubscriberAgentBase):
self.sub_socket.connect(uri) self.sub_socket.connect(uri)
for topic in self.topic_list: for topic in self.topic_list:
self.sub_socket.setsockopt(zmq.SUBSCRIBE, topic) self.sub_socket.setsockopt(zmq.SUBSCRIBE, topic)
@six.add_metaclass(abc.ABCMeta)
class ZMQPubSubBase(pub_sub_api.PubSubApi):
def __init__(self):
super(ZMQPubSubBase, self).__init__()
transport = cfg.CONF.df.publisher_transport
if transport not in SUPPORTED_TRANSPORTS:
message = ("zmq_pub_sub: Unsupported publisher_transport value "
"%(transport)s, expected %(expected)s")
LOG.error(message, {
'transport': transport,
'expected': SUPPORTED_TRANSPORTS
})
raise exceptions.UnsupportedTransportException(transport=transport)
self.subscriber = None
self.publisher = None
def get_publisher(self):
return self.publisher
def get_subscriber(self):
return self.subscriber
class ZMQPubSubBind(ZMQPubSubBase):
"""Has IPC subscriber and TCP/PGM publisher"""
def __init__(self):
super(ZMQPubSubBind, self).__init__()
self.subscriber = ZMQSubscriberMultiprocAgent()
self.publisher = ZMQPublisherAgent()
class ZMQPubSubConnect(ZMQPubSubBase):
"""Has TCP/PGM subscriber and IPC publisher"""
def __init__(self):
super(ZMQPubSubConnect, self).__init__()
self.subscriber = ZMQSubscriberAgent()
self.publisher = ZMQPublisherMultiprocAgent()

View File

@ -79,7 +79,7 @@ class DFMechDriver(api.MechanismDriver):
def post_fork_initialize(self, resource, event, trigger, **kwargs): def post_fork_initialize(self, resource, event, trigger, **kwargs):
# NOTE(nick-ma-z): This will initialize all workers (API, RPC, # NOTE(nick-ma-z): This will initialize all workers (API, RPC,
# plugin service, etc) and threads with network connections. # plugin service, etc) and threads with network connections.
self.nb_api = api_nb.NbApi.get_instance(True) self.nb_api = api_nb.NbApi.get_instance()
df_qos.initialize(self.nb_api) df_qos.initialize(self.nb_api)
if cfg.CONF.df.enable_neutron_notifier: if cfg.CONF.df.enable_neutron_notifier:
neutron_notifier = df_utils.load_driver( neutron_notifier = df_utils.load_driver(

View File

@ -79,7 +79,3 @@ class _DummyDbDriver(db_api.DbApi):
def process_ha(self): def process_ha(self):
# Do nothing # Do nothing
pass pass
def set_neutron_server(self, is_neutron_server):
# Do nothing
pass

View File

@ -156,7 +156,7 @@ def main():
elif sys.argv[1] != 'client': elif sys.argv[1] != 'client':
raise Exception('Bad parameter #1: Expected \'server\' or \'client\',' raise Exception('Bad parameter #1: Expected \'server\' or \'client\','
' found: %s' % sys.argv[1]) ' found: %s' % sys.argv[1])
nb_api = api_nb.NbApi.get_instance(is_server) nb_api = api_nb.NbApi.get_instance()
if is_server: if is_server:
run_server(nb_api) run_server(nb_api)
else: else:

View File

@ -61,7 +61,7 @@ class DFTestBase(base.BaseTestCase):
self.integration_bridge = self.conf.integration_bridge self.integration_bridge = self.conf.integration_bridge
self._queue = queue.PriorityQueue() self._queue = queue.PriorityQueue()
self.nb_api = api_nb.NbApi.get_instance(False) self.nb_api = api_nb.NbApi.get_instance()
# As we are running in the same process over and over, # As we are running in the same process over and over,
# do not perform redundant calls to the subscriber # do not perform redundant calls to the subscriber
if not self.nb_api.subscriber.is_running: if not self.nb_api.subscriber.is_running:
@ -112,9 +112,6 @@ class DFTestBase(base.BaseTestCase):
return publisher return publisher
def get_publisher(self, port=None): def get_publisher(self, port=None):
if cfg.CONF.df.pub_sub_use_multiproc:
pubsub_driver_name = cfg.CONF.df.pub_sub_multiproc_driver
else:
pubsub_driver_name = cfg.CONF.df.pub_sub_driver pubsub_driver_name = cfg.CONF.df.pub_sub_driver
if port is not None: if port is not None:
cfg.CONF.set_override('publisher_port', port, group='df') cfg.CONF.set_override('publisher_port', port, group='df')

View File

@ -22,7 +22,6 @@ from dragonflow.db import db_common
from dragonflow.db.models import core from dragonflow.db.models import core
from dragonflow.db import pub_sub_api from dragonflow.db import pub_sub_api
from dragonflow.tests.common import constants as const from dragonflow.tests.common import constants as const
from dragonflow.tests.common import utils as test_utils
from dragonflow.tests.fullstack import test_base from dragonflow.tests.fullstack import test_base
from dragonflow.tests.fullstack import test_objects as objects from dragonflow.tests.fullstack import test_objects as objects
@ -302,66 +301,6 @@ class TestPubSub(PubSubTestBase):
self.assertEqual(ns.events_action, action) self.assertEqual(ns.events_action, action)
class TestMultiprocPubSub(PubSubTestBase):
def setUp(self):
super(TestMultiprocPubSub, self).setUp()
self.do_test = cfg.CONF.df.pub_sub_use_multiproc
self.key = 'key-{}'.format(random.random())
self.event = db_common.DbUpdate(
'info',
None,
"log",
"TestMultiprocPubSub value",
topic=db_common.SEND_ALL_TOPIC,
)
self.publisher = None
self.subscriber = None
def tearDown(self):
if self.subscriber:
self.subscriber.close()
self._stop_publisher(self.publisher)
super(TestMultiprocPubSub, self).tearDown()
def _handle_received_event(self, table, key, action, value, topic):
self.event_received_info = db_common.DbUpdate(
table,
key,
action,
value,
topic=topic)
self.event_received = True
def test_multiproc_pub_sub(self):
if not self.do_test:
self.skipTest('pub/sub is not enabled')
return
self.event_received = False
cfg.CONF.set_override('publisher_multiproc_socket',
'/tmp/ipc_test_socket', group='df')
pub_sub_driver = df_utils.load_driver(
cfg.CONF.df.pub_sub_multiproc_driver,
df_utils.DF_PUBSUB_DRIVER_NAMESPACE)
publisher = pub_sub_driver.get_publisher()
publisher.initialize()
self.subscriber = pub_sub_driver.get_subscriber()
self.subscriber.initialize(self._handle_received_event)
self.subscriber.daemonize()
publisher.send_event(self.event)
test_utils.wait_until_true(lambda: self.event_received)
self.subscriber.close()
self.subscriber = None
# Check that we received the same event
self.assertEqual(self.event.table, self.event_received_info.table)
self.assertEqual(self.event.key, self.event_received_info.key)
self.assertEqual(self.event.action, self.event_received_info.action)
# Value is not tested, since it's currently set to None
# self.assertEqual(self.event.value, self.event_received_info.value)
self.assertEqual(self.event.topic, self.event_received_info.topic)
class TestDbTableMonitors(PubSubTestBase): class TestDbTableMonitors(PubSubTestBase):
def setUp(self): def setUp(self):
super(TestDbTableMonitors, self).setUp() super(TestDbTableMonitors, self).setUp()

View File

@ -12,6 +12,8 @@
from jsonmodels import fields from jsonmodels import fields
import mock import mock
from oslo_config import cfg
from dragonflow.common import exceptions from dragonflow.common import exceptions
from dragonflow.db import api_nb from dragonflow.db import api_nb
from dragonflow.db import db_common from dragonflow.db import db_common
@ -40,11 +42,8 @@ class TopicModelTest(mf.ModelBase, mixins.Topic):
class TestNbApi(tests_base.BaseTestCase): class TestNbApi(tests_base.BaseTestCase):
def setUp(self): def setUp(self):
super(TestNbApi, self).setUp() super(TestNbApi, self).setUp()
self.api_nb = api_nb.NbApi( cfg.CONF.set_override('enable_df_pub_sub', True, group='df')
db_driver=mock.Mock(), self.api_nb = api_nb.NbApi(db_driver=mock.Mock())
use_pubsub=True,
is_neutron_server=True
)
self.api_nb.publisher = mock.Mock() self.api_nb.publisher = mock.Mock()
self.api_nb.enable_selective_topo_dist = True self.api_nb.enable_selective_topo_dist = True

View File

@ -80,7 +80,7 @@ class DFAppTestBase(tests_base.BaseTestCase):
# CLear old objects from cache # CLear old objects from cache
db_store.get_instance().clear() db_store.get_instance().clear()
self.nb_api = api_nb.NbApi.get_instance(False) self.nb_api = api_nb.NbApi.get_instance()
self.controller = df_local_controller.DfLocalController( self.controller = df_local_controller.DfLocalController(
fake_chassis1.id, self.nb_api) fake_chassis1.id, self.nb_api)
self.vswitch_api = self.controller.vswitch_api = mock.MagicMock() self.vswitch_api = self.controller.vswitch_api = mock.MagicMock()

View File

@ -77,7 +77,7 @@ class TestDFBGPService(tests_base.BaseTestCase):
mock_nb_api = mock.patch('dragonflow.db.api_nb.NbApi.get_instance') mock_nb_api = mock.patch('dragonflow.db.api_nb.NbApi.get_instance')
mock_nb_api.start() mock_nb_api.start()
self.addCleanup(mock_nb_api.stop) self.addCleanup(mock_nb_api.stop)
nb_api = api_nb.NbApi.get_instance(False) nb_api = api_nb.NbApi.get_instance()
self.bgp_service = df_bgp_service.BGPService(nb_api) self.bgp_service = df_bgp_service.BGPService(nb_api)
self.bgp_service.bgp_driver = mock.Mock() self.bgp_service.bgp_driver = mock.Mock()
self.bgp_service.bgp_pulse = LoopingCallByEvent( self.bgp_service.bgp_pulse = LoopingCallByEvent(

View File

@ -58,11 +58,7 @@ class TestSync(tests_base.BaseTestCase):
self._db_store = db_store.get_instance() self._db_store = db_store.get_instance()
self._db_store.clear() self._db_store.clear()
self.nb_api = api_nb.NbApi( self.nb_api = api_nb.NbApi(db_driver=mock.Mock())
db_driver=mock.Mock(),
use_pubsub=True,
is_neutron_server=True
)
self.nb_api.publisher = mock.Mock() self.nb_api.publisher = mock.Mock()
self.nb_api.enable_selective_topo_dist = True self.nb_api.enable_selective_topo_dist = True
self._update = mock.Mock(side_effect=self._db_store.update) self._update = mock.Mock(side_effect=self._db_store.update)

View File

@ -68,8 +68,8 @@ console_scripts =
df-bgp-service = dragonflow.cmd.eventlet.df_bgp_service:main df-bgp-service = dragonflow.cmd.eventlet.df_bgp_service:main
df-skydive-service= dragonflow.cmd.df_skydive_service:service_main df-skydive-service= dragonflow.cmd.df_skydive_service:service_main
dragonflow.pubsub_driver = dragonflow.pubsub_driver =
zmq_pubsub_driver = dragonflow.db.pubsub_drivers.zmq_pubsub_driver:ZMQPubSub zmq_pubsub_driver = dragonflow.db.pubsub_drivers.zmq_pubsub_driver:ZMQPubSubConnect
zmq_pubsub_multiproc_driver = dragonflow.db.pubsub_drivers.zmq_pubsub_driver:ZMQPubSubMultiproc zmq_bind_pubsub_driver = dragonflow.db.pubsub_drivers.zmq_pubsub_driver:ZMQPubSubBind
redis_db_pubsub_driver = dragonflow.db.pubsub_drivers.redis_db_pubsub_driver:RedisPubSub redis_db_pubsub_driver = dragonflow.db.pubsub_drivers.redis_db_pubsub_driver:RedisPubSub
etcd_pubsub_driver = dragonflow.db.pubsub_drivers.etcd_pubsub_driver:EtcdPubSub etcd_pubsub_driver = dragonflow.db.pubsub_drivers.etcd_pubsub_driver:EtcdPubSub
dragonflow.nb_db_driver = dragonflow.nb_db_driver =