2013-12-21 17:34:51 -07:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
#
|
|
|
|
# Copyright (C) 2013 GNS3 Technologies Inc.
|
|
|
|
#
|
|
|
|
# This program is free software: you can redistribute it and/or modify
|
|
|
|
# it under the terms of the GNU General Public License as published by
|
|
|
|
# the Free Software Foundation, either version 3 of the License, or
|
|
|
|
# (at your option) any later version.
|
|
|
|
#
|
|
|
|
# This program is distributed in the hope that it will be useful,
|
|
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
# GNU General Public License for more details.
|
|
|
|
#
|
|
|
|
# You should have received a copy of the GNU General Public License
|
|
|
|
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
"""
|
|
|
|
JSON-RPC protocol implementation.
|
|
|
|
http://www.jsonrpc.org/specification
|
|
|
|
"""
|
|
|
|
|
|
|
|
import json
|
|
|
|
import uuid
|
|
|
|
|
|
|
|
|
|
|
|
class JSONRPCObject(object):
|
|
|
|
"""
|
|
|
|
Base object for JSON-RPC requests, responses,
|
|
|
|
notifications and errors.
|
|
|
|
"""
|
|
|
|
|
|
|
|
def __init__(self):
|
|
|
|
return JSONRPCEncoder().default(self)
|
|
|
|
|
|
|
|
def __str__(self, *args, **kwargs):
|
|
|
|
return json.dumps(self, cls=JSONRPCEncoder)
|
|
|
|
|
|
|
|
def __call__(self):
|
|
|
|
return JSONRPCEncoder().default(self)
|
|
|
|
|
|
|
|
|
|
|
|
class JSONRPCEncoder(json.JSONEncoder):
|
|
|
|
"""
|
|
|
|
Creates the JSON-RPC message.
|
|
|
|
"""
|
|
|
|
|
|
|
|
def default(self, obj):
|
|
|
|
"""
|
|
|
|
Returns a Python dictionary corresponding to a JSON-RPC message.
|
|
|
|
"""
|
|
|
|
|
|
|
|
if isinstance(obj, JSONRPCObject):
|
|
|
|
message = {"jsonrpc": 2.0}
|
|
|
|
for field in dir(obj):
|
|
|
|
if not field.startswith('_'):
|
|
|
|
value = getattr(obj, field)
|
|
|
|
message[field] = value
|
|
|
|
return message
|
|
|
|
return json.JSONEncoder.default(self, obj)
|
|
|
|
|
|
|
|
|
|
|
|
class JSONRPCInvalidRequest(JSONRPCObject):
|
|
|
|
"""
|
|
|
|
Error response for an invalid request.
|
|
|
|
"""
|
|
|
|
|
|
|
|
def __init__(self):
|
|
|
|
JSONRPCObject.__init__(self)
|
|
|
|
self.id = None
|
|
|
|
self.error = {"code": -32600, "message": "Invalid Request"}
|
|
|
|
|
|
|
|
|
|
|
|
class JSONRPCMethodNotFound(JSONRPCObject):
|
|
|
|
"""
|
|
|
|
Error response for an method not found.
|
|
|
|
|
|
|
|
:param request_id: JSON-RPC identifier
|
|
|
|
"""
|
|
|
|
|
|
|
|
def __init__(self, request_id):
|
|
|
|
JSONRPCObject.__init__(self)
|
|
|
|
self.id = request_id
|
|
|
|
self.error = {"code": -32601, "message": "Method not found"}
|
|
|
|
|
|
|
|
|
|
|
|
class JSONRPCInvalidParams(JSONRPCObject):
|
|
|
|
"""
|
|
|
|
Error response for invalid parameters.
|
|
|
|
|
|
|
|
:param request_id: JSON-RPC identifier
|
|
|
|
"""
|
|
|
|
|
|
|
|
def __init__(self, request_id):
|
|
|
|
JSONRPCObject.__init__(self)
|
|
|
|
self.id = request_id
|
|
|
|
self.error = {"code": -32602, "message": "Invalid params"}
|
|
|
|
|
|
|
|
|
|
|
|
class JSONRPCInternalError(JSONRPCObject):
|
|
|
|
"""
|
|
|
|
Error response for an internal error.
|
|
|
|
|
|
|
|
:param request_id: JSON-RPC identifier (optional)
|
|
|
|
"""
|
|
|
|
|
|
|
|
def __init__(self, request_id=None):
|
|
|
|
JSONRPCObject.__init__(self)
|
|
|
|
self.id = request_id
|
|
|
|
self.error = {"code": -32603, "message": "Internal error"}
|
|
|
|
|
|
|
|
|
|
|
|
class JSONRPCParseError(JSONRPCObject):
|
|
|
|
"""
|
|
|
|
Error response for parsing error.
|
|
|
|
"""
|
|
|
|
|
|
|
|
def __init__(self):
|
|
|
|
JSONRPCObject.__init__(self)
|
|
|
|
self.id = None
|
|
|
|
self.error = {"code": -32700, "message": "Parse error"}
|
|
|
|
|
|
|
|
|
|
|
|
class JSONRPCCustomError(JSONRPCObject):
|
|
|
|
"""
|
|
|
|
Error response for an custom error.
|
|
|
|
|
|
|
|
:param code: JSON-RPC error code
|
|
|
|
:param message: JSON-RPC error message
|
|
|
|
:param request_id: JSON-RPC identifier (optional)
|
|
|
|
"""
|
|
|
|
|
|
|
|
def __init__(self, code, message, request_id=None):
|
|
|
|
JSONRPCObject.__init__(self)
|
|
|
|
self.id = request_id
|
|
|
|
self.error = {"code": code, "message": message}
|
|
|
|
|
|
|
|
|
|
|
|
class JSONRPCResponse(JSONRPCObject):
|
|
|
|
"""
|
|
|
|
JSON-RPC successful response.
|
|
|
|
|
|
|
|
:param result: JSON-RPC result
|
|
|
|
:param request_id: JSON-RPC identifier
|
|
|
|
"""
|
|
|
|
|
|
|
|
def __init__(self, result, request_id):
|
|
|
|
JSONRPCObject.__init__(self)
|
|
|
|
self.id = request_id
|
|
|
|
self.result = result
|
|
|
|
|
|
|
|
|
|
|
|
class JSONRPCRequest(JSONRPCObject):
|
|
|
|
"""
|
|
|
|
JSON-RPC request.
|
|
|
|
|
|
|
|
:param method: JSON-RPC destination method
|
|
|
|
:param params: JSON-RPC params for the corresponding method (optional)
|
|
|
|
:param request_id: JSON-RPC identifier (generated by default)
|
|
|
|
"""
|
|
|
|
|
2014-01-31 16:31:34 -07:00
|
|
|
def __init__(self, method, params=None, request_id=None):
|
2013-12-21 17:34:51 -07:00
|
|
|
JSONRPCObject.__init__(self)
|
2014-05-28 06:26:20 -06:00
|
|
|
if request_id is None:
|
2014-01-31 16:31:34 -07:00
|
|
|
request_id = str(uuid.uuid4())
|
2013-12-21 17:34:51 -07:00
|
|
|
self.id = request_id
|
|
|
|
self.method = method
|
|
|
|
if params:
|
|
|
|
self.params = params
|
|
|
|
|
|
|
|
|
|
|
|
class JSONRPCNotification(JSONRPCObject):
|
|
|
|
"""
|
|
|
|
JSON-RPC notification.
|
|
|
|
|
|
|
|
:param method: JSON-RPC destination method
|
|
|
|
:param params: JSON-RPC params for the corresponding method (optional)
|
|
|
|
"""
|
|
|
|
|
|
|
|
def __init__(self, method, params=None):
|
|
|
|
JSONRPCObject.__init__(self)
|
|
|
|
self.method = method
|
|
|
|
if params:
|
|
|
|
self.params = params
|