2018-11-12 22:13:15 +00:00
|
|
|
from __future__ import print_function
|
|
|
|
import json
|
|
|
|
import re
|
|
|
|
from pylspclient import lsp_structs
|
|
|
|
import threading
|
|
|
|
|
2018-11-15 22:41:21 +00:00
|
|
|
JSON_RPC_REQ_FORMAT = "Content-Length: {json_string_len}\r\n\r\n{json_string}"
|
2019-05-12 18:55:43 +00:00
|
|
|
LEN_HEADER = "Content-Length: "
|
|
|
|
TYPE_HEADER = "Content-Type: "
|
|
|
|
|
|
|
|
|
2018-11-12 22:13:15 +00:00
|
|
|
# TODO: add content-type
|
|
|
|
|
|
|
|
|
2019-05-12 18:55:43 +00:00
|
|
|
class MyEncoder(json.JSONEncoder):
|
2018-11-12 22:13:15 +00:00
|
|
|
"""
|
|
|
|
Encodes an object in JSON
|
|
|
|
"""
|
2019-05-12 18:55:43 +00:00
|
|
|
def default(self, o): # pylint: disable=E0202
|
2018-11-12 22:13:15 +00:00
|
|
|
return o.__dict__
|
|
|
|
|
|
|
|
|
|
|
|
class JsonRpcEndpoint(object):
|
|
|
|
'''
|
|
|
|
Thread safe JSON RPC endpoint implementation. Responsible to recieve and send JSON RPC messages, as described in the
|
|
|
|
protocol. More information can be found: https://www.jsonrpc.org/
|
|
|
|
'''
|
|
|
|
def __init__(self, stdin, stdout):
|
|
|
|
self.stdin = stdin
|
|
|
|
self.stdout = stdout
|
|
|
|
self.read_lock = threading.Lock()
|
|
|
|
self.write_lock = threading.Lock()
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def __add_header(json_string):
|
2018-11-15 22:41:21 +00:00
|
|
|
'''
|
|
|
|
Adds a header for the given json string
|
|
|
|
|
|
|
|
:param str json_string: The string
|
|
|
|
:return: the string with the header
|
|
|
|
'''
|
2018-11-12 22:13:15 +00:00
|
|
|
return JSON_RPC_REQ_FORMAT.format(json_string_len=len(json_string), json_string=json_string)
|
|
|
|
|
|
|
|
|
|
|
|
def send_request(self, message):
|
|
|
|
'''
|
2018-11-15 22:41:21 +00:00
|
|
|
Sends the given message.
|
|
|
|
|
2018-11-12 22:13:15 +00:00
|
|
|
:param dict message: The message to send.
|
|
|
|
'''
|
|
|
|
json_string = json.dumps(message, cls=MyEncoder)
|
|
|
|
jsonrpc_req = self.__add_header(json_string)
|
|
|
|
with self.write_lock:
|
|
|
|
self.stdin.write(jsonrpc_req.encode())
|
|
|
|
self.stdin.flush()
|
|
|
|
|
|
|
|
|
|
|
|
def recv_response(self):
|
2018-11-15 22:41:21 +00:00
|
|
|
'''
|
2019-01-11 22:55:02 +00:00
|
|
|
Recives a message.
|
2018-11-15 22:41:21 +00:00
|
|
|
|
|
|
|
:return: a message
|
2018-11-12 22:13:15 +00:00
|
|
|
'''
|
|
|
|
with self.read_lock:
|
2019-05-12 18:55:43 +00:00
|
|
|
message_size = None
|
|
|
|
while True:
|
|
|
|
#read header
|
|
|
|
line = self.stdout.readline()
|
|
|
|
if not line:
|
|
|
|
# server quit
|
|
|
|
return None
|
|
|
|
line = line.decode("utf-8")
|
|
|
|
if not line.endswith("\r\n"):
|
|
|
|
raise lsp_structs.ResponseError(lsp_structs.ErrorCodes.ParseError, "Bad header: missing newline")
|
|
|
|
#remove the "\r\n"
|
|
|
|
line = line[:-2]
|
|
|
|
if line == "":
|
|
|
|
# done with the headers
|
|
|
|
break
|
|
|
|
elif line.startswith(LEN_HEADER):
|
|
|
|
line = line[len(LEN_HEADER):]
|
|
|
|
if not line.isdigit():
|
|
|
|
raise lsp_structs.ResponseError(lsp_structs.ErrorCodes.ParseError, "Bad header: size is not int")
|
|
|
|
message_size = int(line)
|
|
|
|
elif line.startswith(TYPE_HEADER):
|
|
|
|
# nothing todo with type for now.
|
|
|
|
pass
|
|
|
|
else:
|
|
|
|
raise lsp_structs.ResponseError(lsp_structs.ErrorCodes.ParseError, "Bad header: unkown header")
|
|
|
|
if not message_size:
|
|
|
|
raise lsp_structs.ResponseError(lsp_structs.ErrorCodes.ParseError, "Bad header: missing size")
|
|
|
|
|
|
|
|
jsonrpc_res = self.stdout.read(message_size).decode("utf-8")
|
2018-11-12 22:13:15 +00:00
|
|
|
return json.loads(jsonrpc_res)
|