41a52adaf8
- Topology change is notified - JSON-RPC/WebSocket Signed-off-by: Satoshi Kobayashi <satoshi-k@stratosphere.co.jp> Reviewed-by: YAMADA Hideki <yamada.hideki@po.ntts.co.jp> Signed-off-by: FUJITA Tomonori <fujita.tomonori@lab.ntt.co.jp>
302 lines
9.0 KiB
Python
302 lines
9.0 KiB
Python
# Copyright (C) 2012 Nippon Telegraph and Telephone Corporation.
|
|
# Copyright (C) 2012 Isaku Yamahata <yamahata at private email ne jp>
|
|
#
|
|
# 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.
|
|
|
|
import inspect
|
|
from types import MethodType
|
|
|
|
import webob.dec
|
|
from webob.response import Response
|
|
from ryu import cfg
|
|
from ryu.lib import hub
|
|
from routes import Mapper
|
|
from routes.util import URLGenerator
|
|
from tinyrpc.server import RPCServer
|
|
from tinyrpc.dispatch import RPCDispatcher
|
|
from tinyrpc.dispatch import public as rpc_public
|
|
from tinyrpc.protocols.jsonrpc import JSONRPCProtocol
|
|
from tinyrpc.transports import ServerTransport, ClientTransport
|
|
from tinyrpc.client import RPCClient
|
|
|
|
CONF = cfg.CONF
|
|
CONF.register_cli_opts([
|
|
cfg.StrOpt('wsapi-host', default='', help='webapp listen host'),
|
|
cfg.IntOpt('wsapi-port', default=8080, help='webapp listen port')
|
|
])
|
|
|
|
HEX_PATTERN = r'0x[0-9a-z]+'
|
|
DIGIT_PATTERN = r'[1-9][0-9]*'
|
|
|
|
|
|
def route(name, path, methods=None, requirements=None):
|
|
def _route(controller_method):
|
|
controller_method.routing_info = {
|
|
'name': name,
|
|
'path': path,
|
|
'methods': methods,
|
|
'requirements': requirements,
|
|
}
|
|
return controller_method
|
|
return _route
|
|
|
|
|
|
class WebSocketRegistrationWrapper(object):
|
|
|
|
def __init__(self, func, controller):
|
|
self._controller = controller
|
|
self._controller_method = MethodType(func, controller)
|
|
|
|
def __call__(self, ws):
|
|
wsgi_application = self._controller.parent
|
|
ws_manager = wsgi_application.websocketmanager
|
|
ws_manager.add_connection(ws)
|
|
try:
|
|
self._controller_method(ws)
|
|
finally:
|
|
ws_manager.delete_connection(ws)
|
|
|
|
|
|
class _AlreadyHandledResponse(Response):
|
|
# XXX: Eventlet API should not be used directly.
|
|
from eventlet.wsgi import ALREADY_HANDLED
|
|
_ALREADY_HANDLED = ALREADY_HANDLED
|
|
|
|
def __call__(self, environ, start_response):
|
|
return self._ALREADY_HANDLED
|
|
|
|
|
|
def websocket(name, path):
|
|
def _websocket(controller_func):
|
|
def __websocket(self, req, **kwargs):
|
|
wrapper = WebSocketRegistrationWrapper(controller_func, self)
|
|
ws_wsgi = hub.WebSocketWSGI(wrapper)
|
|
ws_wsgi(req.environ, req.start_response)
|
|
# XXX: In order to prevent the writing to a already closed socket.
|
|
# This issue is caused by combined use:
|
|
# - webob.dec.wsgify()
|
|
# - eventlet.wsgi.HttpProtocol.handle_one_response()
|
|
return _AlreadyHandledResponse()
|
|
__websocket.routing_info = {
|
|
'name': name,
|
|
'path': path,
|
|
'methods': None,
|
|
'requirements': None,
|
|
}
|
|
return __websocket
|
|
return _websocket
|
|
|
|
|
|
class ControllerBase(object):
|
|
special_vars = ['action', 'controller']
|
|
|
|
def __init__(self, req, link, data, **config):
|
|
self.req = req
|
|
self.link = link
|
|
self.parent = None
|
|
for name, value in config.items():
|
|
setattr(self, name, value)
|
|
|
|
def __call__(self, req):
|
|
action = self.req.urlvars.get('action', 'index')
|
|
if hasattr(self, '__before__'):
|
|
self.__before__()
|
|
|
|
kwargs = self.req.urlvars.copy()
|
|
for attr in self.special_vars:
|
|
if attr in kwargs:
|
|
del kwargs[attr]
|
|
|
|
return getattr(self, action)(req, **kwargs)
|
|
|
|
|
|
class WebSocketDisconnectedError(Exception):
|
|
pass
|
|
|
|
|
|
class WebSocketServerTransport(ServerTransport):
|
|
def __init__(self, ws):
|
|
self.ws = ws
|
|
|
|
def receive_message(self):
|
|
message = self.ws.wait()
|
|
if message is None:
|
|
raise WebSocketDisconnectedError()
|
|
context = None
|
|
return (context, message)
|
|
|
|
def send_reply(self, context, reply):
|
|
self.ws.send(unicode(reply))
|
|
|
|
|
|
class WebSocketRPCServer(RPCServer):
|
|
def __init__(self, ws, rpc_callback):
|
|
dispatcher = RPCDispatcher()
|
|
dispatcher.register_instance(rpc_callback)
|
|
super(WebSocketRPCServer, self).__init__(
|
|
WebSocketServerTransport(ws),
|
|
JSONRPCProtocol(),
|
|
dispatcher,
|
|
)
|
|
|
|
def serve_forever(self):
|
|
try:
|
|
super(WebSocketRPCServer, self).serve_forever()
|
|
except WebSocketDisconnectedError:
|
|
return
|
|
|
|
def _spawn(self, func, *args, **kwargs):
|
|
hub.spawn(func, *args, **kwargs)
|
|
|
|
|
|
class WebSocketClientTransport(ClientTransport):
|
|
|
|
def __init__(self, ws, queue):
|
|
self.ws = ws
|
|
self.queue = queue
|
|
|
|
def send_message(self, message, expect_reply=True):
|
|
self.ws.send(unicode(message))
|
|
|
|
if expect_reply:
|
|
return self.queue.get()
|
|
|
|
|
|
class WebSocketRPCClient(RPCClient):
|
|
|
|
def __init__(self, ws):
|
|
self.ws = ws
|
|
self.queue = hub.Queue()
|
|
super(WebSocketRPCClient, self).__init__(
|
|
JSONRPCProtocol(),
|
|
WebSocketClientTransport(ws, self.queue),
|
|
)
|
|
|
|
def serve_forever(self):
|
|
while True:
|
|
msg = self.ws.wait()
|
|
if msg is None:
|
|
break
|
|
self.queue.put(msg)
|
|
|
|
|
|
class wsgify_hack(webob.dec.wsgify):
|
|
def __call__(self, environ, start_response):
|
|
self.kwargs['start_response'] = start_response
|
|
return super(wsgify_hack, self).__call__(environ, start_response)
|
|
|
|
|
|
class WebSocketManager(object):
|
|
|
|
def __init__(self):
|
|
self._connections = []
|
|
|
|
def add_connection(self, ws):
|
|
self._connections.append(ws)
|
|
|
|
def delete_connection(self, ws):
|
|
self._connections.remove(ws)
|
|
|
|
def broadcast(self, msg):
|
|
for connection in self._connections:
|
|
connection.send(msg)
|
|
|
|
|
|
class WSGIApplication(object):
|
|
def __init__(self, **config):
|
|
self.config = config
|
|
self.mapper = Mapper()
|
|
self.registory = {}
|
|
self._wsmanager = WebSocketManager()
|
|
super(WSGIApplication, self).__init__()
|
|
# XXX: Switch how to call the API of Routes for every version
|
|
match_argspec = inspect.getargspec(self.mapper.match)
|
|
if 'environ' in match_argspec.args:
|
|
# New API
|
|
self._match = self._match_with_environ
|
|
else:
|
|
# Old API
|
|
self._match = self._match_with_path_info
|
|
|
|
def _match_with_environ(self, req):
|
|
match = self.mapper.match(environ=req.environ)
|
|
return match
|
|
|
|
def _match_with_path_info(self, req):
|
|
self.mapper.environ = req.environ
|
|
match = self.mapper.match(req.path_info)
|
|
return match
|
|
|
|
@wsgify_hack
|
|
def __call__(self, req, start_response):
|
|
match = self._match(req)
|
|
|
|
if not match:
|
|
return webob.exc.HTTPNotFound()
|
|
|
|
req.start_response = start_response
|
|
req.urlvars = match
|
|
link = URLGenerator(self.mapper, req.environ)
|
|
|
|
data = None
|
|
name = match['controller'].__name__
|
|
if name in self.registory:
|
|
data = self.registory[name]
|
|
|
|
controller = match['controller'](req, link, data, **self.config)
|
|
controller.parent = self
|
|
return controller(req)
|
|
|
|
def register(self, controller, data=None):
|
|
methods = inspect.getmembers(controller,
|
|
lambda v: inspect.ismethod(v) and
|
|
hasattr(v, 'routing_info'))
|
|
for method_name, method in methods:
|
|
routing_info = getattr(method, 'routing_info')
|
|
name = routing_info['name']
|
|
path = routing_info['path']
|
|
conditions = {}
|
|
if routing_info.get('methods'):
|
|
conditions['method'] = routing_info['methods']
|
|
requirements = routing_info.get('requirements') or {}
|
|
self.mapper.connect(name,
|
|
path,
|
|
controller=controller,
|
|
requirements=requirements,
|
|
action=method_name,
|
|
conditions=conditions)
|
|
if data:
|
|
self.registory[controller.__name__] = data
|
|
|
|
@property
|
|
def websocketmanager(self):
|
|
return self._wsmanager
|
|
|
|
|
|
class WSGIServer(hub.WSGIServer):
|
|
def __init__(self, application, **config):
|
|
super(WSGIServer, self).__init__((CONF.wsapi_host, CONF.wsapi_port),
|
|
application, **config)
|
|
|
|
def __call__(self):
|
|
self.serve_forever()
|
|
|
|
|
|
def start_service(app_mgr):
|
|
for instance in app_mgr.contexts.values():
|
|
if instance.__class__ == WSGIApplication:
|
|
return WSGIServer(instance)
|
|
|
|
return None
|