blob: 887f6974e1102d798906ece3b7c4bb0e98b9e67b [file] [log] [blame]
Yegappan Lakshmanan9247a222022-03-30 10:16:05 +01001#!/usr/bin/env python
2#
3# Server that will accept connections from a Vim channel.
4# Used by test_channel.vim to test LSP functionality.
5#
6# This requires Python 2.6 or later.
7
8from __future__ import print_function
9import json
10import socket
11import sys
12import time
13import threading
14
15try:
16 # Python 3
17 import socketserver
18except ImportError:
19 # Python 2
20 import SocketServer as socketserver
21
22class ThreadedTCPRequestHandler(socketserver.BaseRequestHandler):
23
24 def setup(self):
25 self.request.setsockopt(socket.IPPROTO_TCP, socket.TCP_NODELAY, 1)
26
Yegappan Lakshmananbac9a9e2022-04-19 10:25:13 +010027 def debuglog(self, msg):
28 if self.debug:
29 with open("Xlspserver.log", "a") as myfile:
30 myfile.write(msg)
31
Yegappan Lakshmanan1926ae42023-09-21 16:36:28 +020032 def send_lsp_req(self, msgid, method, params):
33 v = {'jsonrpc': '2.0', 'id': msgid, 'method': method}
34 if len(params) != 0:
35 v['params'] = params
36 s = json.dumps(v)
37 req = "Content-Length: " + str(len(s)) + "\r\n"
38 req += "Content-Type: application/vscode-jsonrpc; charset=utf-8\r\n"
39 req += "\r\n"
40 req += s
41 if self.debug:
42 self.debuglog("SEND: ({0} bytes) '{1}'\n".format(len(req), req))
43 self.request.sendall(req.encode('utf-8'))
44
45 def send_lsp_resp(self, msgid, resp_dict):
Yegappan Lakshmanan9247a222022-03-30 10:16:05 +010046 v = {'jsonrpc': '2.0', 'result': resp_dict}
47 if msgid != -1:
48 v['id'] = msgid
49 s = json.dumps(v)
50 resp = "Content-Length: " + str(len(s)) + "\r\n"
Yegappan Lakshmananc3eddd22023-04-25 14:54:54 +010051 resp += "Content-Type: application/vscode-jsonrpc; charset=utf-8\r\n"
Yegappan Lakshmanan9247a222022-03-30 10:16:05 +010052 resp += "\r\n"
53 resp += s
54 if self.debug:
Yegappan Lakshmananbac9a9e2022-04-19 10:25:13 +010055 self.debuglog("SEND: ({0} bytes) '{1}'\n".format(len(resp), resp))
Yegappan Lakshmanan9247a222022-03-30 10:16:05 +010056 self.request.sendall(resp.encode('utf-8'))
57
58 def send_wrong_payload(self):
59 v = 'wrong-payload'
60 s = json.dumps(v)
61 resp = "Content-Length: " + str(len(s)) + "\r\n"
Yegappan Lakshmananc3eddd22023-04-25 14:54:54 +010062 resp += "Content-Type: application/vscode-jsonrpc; charset=utf-8\r\n"
Yegappan Lakshmanan9247a222022-03-30 10:16:05 +010063 resp += "\r\n"
64 resp += s
65 self.request.sendall(resp.encode('utf-8'))
66
67 def send_empty_header(self, msgid, resp_dict):
68 v = {'jsonrpc': '2.0', 'id': msgid, 'result': resp_dict}
69 s = json.dumps(v)
70 resp = "\r\n"
71 resp += s
72 self.request.sendall(resp.encode('utf-8'))
73
74 def send_empty_payload(self):
75 resp = "Content-Length: 0\r\n"
Yegappan Lakshmananc3eddd22023-04-25 14:54:54 +010076 resp += "Content-Type: application/vscode-jsonrpc; charset=utf-8\r\n"
Yegappan Lakshmanan9247a222022-03-30 10:16:05 +010077 resp += "\r\n"
78 self.request.sendall(resp.encode('utf-8'))
79
80 def send_extra_hdr_fields(self, msgid, resp_dict):
81 # test for sending extra fields in the http header
82 v = {'jsonrpc': '2.0', 'id': msgid, 'result': resp_dict}
83 s = json.dumps(v)
84 resp = "Host: abc.vim.org\r\n"
85 resp += "User-Agent: Python\r\n"
86 resp += "Accept-Language: en-US,en\r\n"
Yegappan Lakshmananc3eddd22023-04-25 14:54:54 +010087 resp += "Content-Type: application/vscode-jsonrpc; charset=utf-8\r\n"
Yegappan Lakshmanan9247a222022-03-30 10:16:05 +010088 resp += "Content-Length: " + str(len(s)) + "\r\n"
89 resp += "\r\n"
90 resp += s
91 self.request.sendall(resp.encode('utf-8'))
92
Yegappan Lakshmanan03cca292022-04-18 14:07:46 +010093 def send_delayed_payload(self, msgid, resp_dict):
94 # test for sending the hdr first and then after some delay, send the
95 # payload
96 v = {'jsonrpc': '2.0', 'id': msgid, 'result': resp_dict}
97 s = json.dumps(v)
98 resp = "Content-Length: " + str(len(s)) + "\r\n"
99 resp += "\r\n"
100 self.request.sendall(resp.encode('utf-8'))
101 time.sleep(0.05)
102 resp = s
103 self.request.sendall(resp.encode('utf-8'))
104
Yegappan Lakshmanan9247a222022-03-30 10:16:05 +0100105 def send_hdr_without_len(self, msgid, resp_dict):
106 # test for sending the http header without length
107 v = {'jsonrpc': '2.0', 'id': msgid, 'result': resp_dict}
108 s = json.dumps(v)
Yegappan Lakshmananc3eddd22023-04-25 14:54:54 +0100109 resp = "Content-Type: application/vscode-jsonrpc; charset=utf-8\r\n"
Yegappan Lakshmanan9247a222022-03-30 10:16:05 +0100110 resp += "\r\n"
111 resp += s
112 self.request.sendall(resp.encode('utf-8'))
113
114 def send_hdr_with_wrong_len(self, msgid, resp_dict):
115 # test for sending the http header with wrong length
116 v = {'jsonrpc': '2.0', 'id': msgid, 'result': resp_dict}
117 s = json.dumps(v)
118 resp = "Content-Length: 1000\r\n"
119 resp += "\r\n"
120 resp += s
121 self.request.sendall(resp.encode('utf-8'))
122
123 def send_hdr_with_negative_len(self, msgid, resp_dict):
124 # test for sending the http header with negative length
125 v = {'jsonrpc': '2.0', 'id': msgid, 'result': resp_dict}
126 s = json.dumps(v)
127 resp = "Content-Length: -1\r\n"
128 resp += "\r\n"
129 resp += s
130 self.request.sendall(resp.encode('utf-8'))
131
132 def do_ping(self, payload):
133 time.sleep(0.2)
Yegappan Lakshmanan1926ae42023-09-21 16:36:28 +0200134 self.send_lsp_resp(payload['id'], 'alive')
Yegappan Lakshmanan9247a222022-03-30 10:16:05 +0100135
136 def do_echo(self, payload):
Yegappan Lakshmanan1926ae42023-09-21 16:36:28 +0200137 self.send_lsp_resp(-1, payload)
Yegappan Lakshmanan9247a222022-03-30 10:16:05 +0100138
139 def do_simple_rpc(self, payload):
140 # test for a simple RPC request
Yegappan Lakshmanan1926ae42023-09-21 16:36:28 +0200141 self.send_lsp_resp(payload['id'], 'simple-rpc')
Yegappan Lakshmanan9247a222022-03-30 10:16:05 +0100142
143 def do_rpc_with_notif(self, payload):
144 # test for sending a notification before replying to a request message
Yegappan Lakshmanan1926ae42023-09-21 16:36:28 +0200145 self.send_lsp_resp(-1, 'rpc-with-notif-notif')
Yegappan Lakshmanan9247a222022-03-30 10:16:05 +0100146 # sleep for some time to make sure the notification is delivered
147 time.sleep(0.2)
Yegappan Lakshmanan1926ae42023-09-21 16:36:28 +0200148 self.send_lsp_resp(payload['id'], 'rpc-with-notif-resp')
Yegappan Lakshmanan9247a222022-03-30 10:16:05 +0100149
150 def do_wrong_payload(self, payload):
151 # test for sending a non dict payload
152 self.send_wrong_payload()
153 time.sleep(0.2)
Yegappan Lakshmanan1926ae42023-09-21 16:36:28 +0200154 self.send_lsp_resp(-1, 'wrong-payload')
Yegappan Lakshmanan9247a222022-03-30 10:16:05 +0100155
Yegappan Lakshmananbac9a9e2022-04-19 10:25:13 +0100156 def do_large_payload(self, payload):
157 # test for sending a large (> 64K) payload
Yegappan Lakshmanan1926ae42023-09-21 16:36:28 +0200158 self.send_lsp_resp(payload['id'], payload)
Yegappan Lakshmananbac9a9e2022-04-19 10:25:13 +0100159
Yegappan Lakshmanan9247a222022-03-30 10:16:05 +0100160 def do_rpc_resp_incorrect_id(self, payload):
Yegappan Lakshmanan1926ae42023-09-21 16:36:28 +0200161 self.send_lsp_resp(-1, 'rpc-resp-incorrect-id-1')
162 self.send_lsp_resp(-1, 'rpc-resp-incorrect-id-2')
163 self.send_lsp_resp(1, 'rpc-resp-incorrect-id-3')
Yegappan Lakshmanan9247a222022-03-30 10:16:05 +0100164 time.sleep(0.2)
Yegappan Lakshmanan1926ae42023-09-21 16:36:28 +0200165 self.send_lsp_resp(payload['id'], 'rpc-resp-incorrect-id-4')
Yegappan Lakshmanan9247a222022-03-30 10:16:05 +0100166
167 def do_simple_notif(self, payload):
168 # notification message test
Yegappan Lakshmanan1926ae42023-09-21 16:36:28 +0200169 self.send_lsp_resp(-1, 'simple-notif')
Yegappan Lakshmanan9247a222022-03-30 10:16:05 +0100170
171 def do_multi_notif(self, payload):
172 # send multiple notifications
Yegappan Lakshmanan1926ae42023-09-21 16:36:28 +0200173 self.send_lsp_resp(-1, 'multi-notif1')
174 self.send_lsp_resp(-1, 'multi-notif2')
Yegappan Lakshmanan9247a222022-03-30 10:16:05 +0100175
176 def do_msg_with_id(self, payload):
Yegappan Lakshmanan1926ae42023-09-21 16:36:28 +0200177 self.send_lsp_resp(payload['id'], 'msg-with-id')
Yegappan Lakshmanan9247a222022-03-30 10:16:05 +0100178
179 def do_msg_specific_cb(self, payload):
Yegappan Lakshmanan1926ae42023-09-21 16:36:28 +0200180 self.send_lsp_resp(payload['id'], 'msg-specific-cb')
Yegappan Lakshmanan9247a222022-03-30 10:16:05 +0100181
182 def do_server_req(self, payload):
Yegappan Lakshmanan1926ae42023-09-21 16:36:28 +0200183 self.send_lsp_resp(201, {'method': 'checkhealth', 'params': {'a': 20}})
Yegappan Lakshmanan9247a222022-03-30 10:16:05 +0100184
185 def do_extra_hdr_fields(self, payload):
186 self.send_extra_hdr_fields(payload['id'], 'extra-hdr-fields')
187
Magnus Groß8fbd9442023-08-27 00:49:51 +0200188 def do_delayed_payload(self, payload):
Yegappan Lakshmanan03cca292022-04-18 14:07:46 +0100189 self.send_delayed_payload(payload['id'], 'delayed-payload')
190
Yegappan Lakshmanan9247a222022-03-30 10:16:05 +0100191 def do_hdr_without_len(self, payload):
192 self.send_hdr_without_len(payload['id'], 'hdr-without-len')
193
194 def do_hdr_with_wrong_len(self, payload):
195 self.send_hdr_with_wrong_len(payload['id'], 'hdr-with-wrong-len')
196
197 def do_hdr_with_negative_len(self, payload):
198 self.send_hdr_with_negative_len(payload['id'], 'hdr-with-negative-len')
199
200 def do_empty_header(self, payload):
201 self.send_empty_header(payload['id'], 'empty-header')
202
203 def do_empty_payload(self, payload):
204 self.send_empty_payload()
205
Yegappan Lakshmanan1926ae42023-09-21 16:36:28 +0200206 def do_server_req_in_middle(self, payload):
207 # Send a notification message to the client in the middle of processing
208 # a request message from the client
209 self.send_lsp_req(-1, 'server-req-in-middle', {'text': 'server-notif'})
210 # Send a request message to the client in the middle of processing a
211 # request message from the client.
212 self.send_lsp_req(payload['id'], 'server-req-in-middle', {'text': 'server-req'})
213
214 def do_server_req_in_middle_resp(self, payload):
215 # After receiving a response from the client send the response to the
216 # client request.
217 self.send_lsp_resp(payload['id'], {'text': 'server-resp'})
218
Yegappan Lakshmanan9247a222022-03-30 10:16:05 +0100219 def process_msg(self, msg):
220 try:
221 decoded = json.loads(msg)
Yegappan Lakshmanan9247a222022-03-30 10:16:05 +0100222 if 'method' in decoded:
223 test_map = {
224 'ping': self.do_ping,
225 'echo': self.do_echo,
226 'simple-rpc': self.do_simple_rpc,
227 'rpc-with-notif': self.do_rpc_with_notif,
228 'wrong-payload': self.do_wrong_payload,
Yegappan Lakshmananbac9a9e2022-04-19 10:25:13 +0100229 'large-payload': self.do_large_payload,
Yegappan Lakshmanan9247a222022-03-30 10:16:05 +0100230 'rpc-resp-incorrect-id': self.do_rpc_resp_incorrect_id,
231 'simple-notif': self.do_simple_notif,
232 'multi-notif': self.do_multi_notif,
233 'msg-with-id': self.do_msg_with_id,
dundargocc57b5bc2022-11-02 13:30:51 +0000234 'msg-specific-cb': self.do_msg_specific_cb,
Yegappan Lakshmanan9247a222022-03-30 10:16:05 +0100235 'server-req': self.do_server_req,
236 'extra-hdr-fields': self.do_extra_hdr_fields,
Magnus Groß8fbd9442023-08-27 00:49:51 +0200237 'delayed-payload': self.do_delayed_payload,
Yegappan Lakshmanan9247a222022-03-30 10:16:05 +0100238 'hdr-without-len': self.do_hdr_without_len,
239 'hdr-with-wrong-len': self.do_hdr_with_wrong_len,
240 'hdr-with-negative-len': self.do_hdr_with_negative_len,
241 'empty-header': self.do_empty_header,
Yegappan Lakshmanan1926ae42023-09-21 16:36:28 +0200242 'empty-payload': self.do_empty_payload,
243 'server-req-in-middle': self.do_server_req_in_middle,
244 'server-req-in-middle-resp': self.do_server_req_in_middle_resp,
Yegappan Lakshmanan9247a222022-03-30 10:16:05 +0100245 }
246 if decoded['method'] in test_map:
247 test_map[decoded['method']](decoded)
248 else:
Yegappan Lakshmananbac9a9e2022-04-19 10:25:13 +0100249 self.debuglog("Error: Unsupported method - " + decoded['method'] + "\n")
Yegappan Lakshmanan9247a222022-03-30 10:16:05 +0100250 else:
Yegappan Lakshmananbac9a9e2022-04-19 10:25:13 +0100251 self.debuglog("Error: 'method' field is not found\n")
Yegappan Lakshmanan9247a222022-03-30 10:16:05 +0100252
253 except ValueError:
Yegappan Lakshmananbac9a9e2022-04-19 10:25:13 +0100254 self.debuglog("Error: json decoding failed\n")
Yegappan Lakshmanan9247a222022-03-30 10:16:05 +0100255
256 def process_msgs(self, msgbuf):
257 while True:
258 sidx = msgbuf.find('Content-Length: ')
259 if sidx == -1:
Yegappan Lakshmananbac9a9e2022-04-19 10:25:13 +0100260 # partial message received
Yegappan Lakshmanan9247a222022-03-30 10:16:05 +0100261 return msgbuf
262 sidx += 16
263 eidx = msgbuf.find('\r\n')
264 if eidx == -1:
Yegappan Lakshmananbac9a9e2022-04-19 10:25:13 +0100265 # partial message received
Yegappan Lakshmanan9247a222022-03-30 10:16:05 +0100266 return msgbuf
267 msglen = int(msgbuf[sidx:eidx])
268
269 hdrend = msgbuf.find('\r\n\r\n')
270 if hdrend == -1:
Yegappan Lakshmananbac9a9e2022-04-19 10:25:13 +0100271 # partial message received
Yegappan Lakshmanan9247a222022-03-30 10:16:05 +0100272 return msgbuf
273
Yegappan Lakshmananbac9a9e2022-04-19 10:25:13 +0100274 if msglen > len(msgbuf[hdrend + 4:]):
275 if self.debug:
276 self.debuglog("Partial message ({0} bytes)\n".format(len(msgbuf)))
277 # partial message received
278 return msgbuf
279
280 if self.debug:
281 self.debuglog("Complete message ({0} bytes) received\n".format(msglen))
282
Yegappan Lakshmanan9247a222022-03-30 10:16:05 +0100283 # Remove the header
284 msgbuf = msgbuf[hdrend + 4:]
285 payload = msgbuf[:msglen]
286
287 self.process_msg(payload)
288
289 # Remove the processed message
290 msgbuf = msgbuf[msglen:]
291
292 def handle(self):
Yegappan Lakshmanan9247a222022-03-30 10:16:05 +0100293 self.debug = False
Yegappan Lakshmananbac9a9e2022-04-19 10:25:13 +0100294 self.debuglog("=== socket opened ===\n")
Yegappan Lakshmanan9247a222022-03-30 10:16:05 +0100295 msgbuf = ''
296 while True:
297 try:
298 received = self.request.recv(4096).decode('utf-8')
299 except socket.error:
Yegappan Lakshmananbac9a9e2022-04-19 10:25:13 +0100300 self.debuglog("=== socket error ===\n")
Yegappan Lakshmanan9247a222022-03-30 10:16:05 +0100301 break
302 except IOError:
Yegappan Lakshmananbac9a9e2022-04-19 10:25:13 +0100303 self.debuglog("=== socket closed ===\n")
Yegappan Lakshmanan9247a222022-03-30 10:16:05 +0100304 break
305 if received == '':
Yegappan Lakshmananbac9a9e2022-04-19 10:25:13 +0100306 self.debuglog("=== socket closed ===\n")
Yegappan Lakshmanan9247a222022-03-30 10:16:05 +0100307 break
Yegappan Lakshmanan9247a222022-03-30 10:16:05 +0100308
309 # Write the received lines into the file for debugging
310 if self.debug:
Yegappan Lakshmananbac9a9e2022-04-19 10:25:13 +0100311 self.debuglog("RECV: ({0} bytes) '{1}'\n".format(len(received), received))
Yegappan Lakshmanan9247a222022-03-30 10:16:05 +0100312
313 # Can receive more than one line in a response or a partial line.
314 # Accumulate all the received characters and process one line at
315 # a time.
316 msgbuf += received
317 msgbuf = self.process_msgs(msgbuf)
318
319class ThreadedTCPServer(socketserver.ThreadingMixIn, socketserver.TCPServer):
320 pass
321
322def writePortInFile(port):
323 # Write the port number in Xportnr, so that the test knows it.
324 f = open("Xportnr", "w")
325 f.write("{0}".format(port))
326 f.close()
327
328def main(host, port, server_class=ThreadedTCPServer):
329 # Wait half a second before opening the port to test waittime in ch_open().
330 # We do want to get the port number, get that first. We cannot open the
331 # socket, guess a port is free.
332 if len(sys.argv) >= 2 and sys.argv[1] == 'delay':
333 port = 13684
334 writePortInFile(port)
Yegappan Lakshmanan9247a222022-03-30 10:16:05 +0100335 time.sleep(0.5)
336
James McCoy765d82a2023-01-09 16:25:59 +0000337 addrs = socket.getaddrinfo(host, port, 0, 0, socket.IPPROTO_TCP)
338 # Each addr is a (family, type, proto, canonname, sockaddr) tuple
339 sockaddr = addrs[0][4]
340 server_class.address_family = addrs[0][0]
341
342 server = server_class(sockaddr[0:2], ThreadedTCPRequestHandler)
Yegappan Lakshmanan9247a222022-03-30 10:16:05 +0100343 ip, port = server.server_address[0:2]
344
345 # Start a thread with the server. That thread will then start a new thread
346 # for each connection.
347 server_thread = threading.Thread(target=server.serve_forever)
348 server_thread.start()
349
350 writePortInFile(port)
351
Yegappan Lakshmanan9247a222022-03-30 10:16:05 +0100352 # Main thread terminates, but the server continues running
353 # until server.shutdown() is called.
354 try:
355 while server_thread.is_alive():
356 server_thread.join(1)
357 except (KeyboardInterrupt, SystemExit):
358 server.shutdown()
359
360if __name__ == "__main__":
361 main("localhost", 0)