Yegappan Lakshmanan | 9247a22 | 2022-03-30 10:16:05 +0100 | [diff] [blame] | 1 | #!/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 | |
| 8 | from __future__ import print_function |
| 9 | import json |
| 10 | import socket |
| 11 | import sys |
| 12 | import time |
| 13 | import threading |
| 14 | |
| 15 | try: |
| 16 | # Python 3 |
| 17 | import socketserver |
| 18 | except ImportError: |
| 19 | # Python 2 |
| 20 | import SocketServer as socketserver |
| 21 | |
| 22 | class ThreadedTCPRequestHandler(socketserver.BaseRequestHandler): |
| 23 | |
| 24 | def setup(self): |
| 25 | self.request.setsockopt(socket.IPPROTO_TCP, socket.TCP_NODELAY, 1) |
| 26 | |
Yegappan Lakshmanan | bac9a9e | 2022-04-19 10:25:13 +0100 | [diff] [blame] | 27 | def debuglog(self, msg): |
| 28 | if self.debug: |
| 29 | with open("Xlspserver.log", "a") as myfile: |
| 30 | myfile.write(msg) |
| 31 | |
Yegappan Lakshmanan | 1926ae4 | 2023-09-21 16:36:28 +0200 | [diff] [blame] | 32 | 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 Lakshmanan | 9247a22 | 2022-03-30 10:16:05 +0100 | [diff] [blame] | 46 | 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 Lakshmanan | c3eddd2 | 2023-04-25 14:54:54 +0100 | [diff] [blame] | 51 | resp += "Content-Type: application/vscode-jsonrpc; charset=utf-8\r\n" |
Yegappan Lakshmanan | 9247a22 | 2022-03-30 10:16:05 +0100 | [diff] [blame] | 52 | resp += "\r\n" |
| 53 | resp += s |
| 54 | if self.debug: |
Yegappan Lakshmanan | bac9a9e | 2022-04-19 10:25:13 +0100 | [diff] [blame] | 55 | self.debuglog("SEND: ({0} bytes) '{1}'\n".format(len(resp), resp)) |
Yegappan Lakshmanan | 9247a22 | 2022-03-30 10:16:05 +0100 | [diff] [blame] | 56 | 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 Lakshmanan | c3eddd2 | 2023-04-25 14:54:54 +0100 | [diff] [blame] | 62 | resp += "Content-Type: application/vscode-jsonrpc; charset=utf-8\r\n" |
Yegappan Lakshmanan | 9247a22 | 2022-03-30 10:16:05 +0100 | [diff] [blame] | 63 | 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 Lakshmanan | c3eddd2 | 2023-04-25 14:54:54 +0100 | [diff] [blame] | 76 | resp += "Content-Type: application/vscode-jsonrpc; charset=utf-8\r\n" |
Yegappan Lakshmanan | 9247a22 | 2022-03-30 10:16:05 +0100 | [diff] [blame] | 77 | 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 Lakshmanan | c3eddd2 | 2023-04-25 14:54:54 +0100 | [diff] [blame] | 87 | resp += "Content-Type: application/vscode-jsonrpc; charset=utf-8\r\n" |
Yegappan Lakshmanan | 9247a22 | 2022-03-30 10:16:05 +0100 | [diff] [blame] | 88 | 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 Lakshmanan | 03cca29 | 2022-04-18 14:07:46 +0100 | [diff] [blame] | 93 | 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 Lakshmanan | 9247a22 | 2022-03-30 10:16:05 +0100 | [diff] [blame] | 105 | 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 Lakshmanan | c3eddd2 | 2023-04-25 14:54:54 +0100 | [diff] [blame] | 109 | resp = "Content-Type: application/vscode-jsonrpc; charset=utf-8\r\n" |
Yegappan Lakshmanan | 9247a22 | 2022-03-30 10:16:05 +0100 | [diff] [blame] | 110 | 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 Lakshmanan | 1926ae4 | 2023-09-21 16:36:28 +0200 | [diff] [blame] | 134 | self.send_lsp_resp(payload['id'], 'alive') |
Yegappan Lakshmanan | 9247a22 | 2022-03-30 10:16:05 +0100 | [diff] [blame] | 135 | |
| 136 | def do_echo(self, payload): |
Yegappan Lakshmanan | 1926ae4 | 2023-09-21 16:36:28 +0200 | [diff] [blame] | 137 | self.send_lsp_resp(-1, payload) |
Yegappan Lakshmanan | 9247a22 | 2022-03-30 10:16:05 +0100 | [diff] [blame] | 138 | |
| 139 | def do_simple_rpc(self, payload): |
| 140 | # test for a simple RPC request |
Yegappan Lakshmanan | 1926ae4 | 2023-09-21 16:36:28 +0200 | [diff] [blame] | 141 | self.send_lsp_resp(payload['id'], 'simple-rpc') |
Yegappan Lakshmanan | 9247a22 | 2022-03-30 10:16:05 +0100 | [diff] [blame] | 142 | |
| 143 | def do_rpc_with_notif(self, payload): |
| 144 | # test for sending a notification before replying to a request message |
Yegappan Lakshmanan | 1926ae4 | 2023-09-21 16:36:28 +0200 | [diff] [blame] | 145 | self.send_lsp_resp(-1, 'rpc-with-notif-notif') |
Yegappan Lakshmanan | 9247a22 | 2022-03-30 10:16:05 +0100 | [diff] [blame] | 146 | # sleep for some time to make sure the notification is delivered |
| 147 | time.sleep(0.2) |
Yegappan Lakshmanan | 1926ae4 | 2023-09-21 16:36:28 +0200 | [diff] [blame] | 148 | self.send_lsp_resp(payload['id'], 'rpc-with-notif-resp') |
Yegappan Lakshmanan | 9247a22 | 2022-03-30 10:16:05 +0100 | [diff] [blame] | 149 | |
| 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 Lakshmanan | 1926ae4 | 2023-09-21 16:36:28 +0200 | [diff] [blame] | 154 | self.send_lsp_resp(-1, 'wrong-payload') |
Yegappan Lakshmanan | 9247a22 | 2022-03-30 10:16:05 +0100 | [diff] [blame] | 155 | |
Yegappan Lakshmanan | bac9a9e | 2022-04-19 10:25:13 +0100 | [diff] [blame] | 156 | def do_large_payload(self, payload): |
| 157 | # test for sending a large (> 64K) payload |
Yegappan Lakshmanan | 1926ae4 | 2023-09-21 16:36:28 +0200 | [diff] [blame] | 158 | self.send_lsp_resp(payload['id'], payload) |
Yegappan Lakshmanan | bac9a9e | 2022-04-19 10:25:13 +0100 | [diff] [blame] | 159 | |
Yegappan Lakshmanan | 9247a22 | 2022-03-30 10:16:05 +0100 | [diff] [blame] | 160 | def do_rpc_resp_incorrect_id(self, payload): |
Yegappan Lakshmanan | 1926ae4 | 2023-09-21 16:36:28 +0200 | [diff] [blame] | 161 | 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 Lakshmanan | 9247a22 | 2022-03-30 10:16:05 +0100 | [diff] [blame] | 164 | time.sleep(0.2) |
Yegappan Lakshmanan | 1926ae4 | 2023-09-21 16:36:28 +0200 | [diff] [blame] | 165 | self.send_lsp_resp(payload['id'], 'rpc-resp-incorrect-id-4') |
Yegappan Lakshmanan | 9247a22 | 2022-03-30 10:16:05 +0100 | [diff] [blame] | 166 | |
| 167 | def do_simple_notif(self, payload): |
| 168 | # notification message test |
Yegappan Lakshmanan | 1926ae4 | 2023-09-21 16:36:28 +0200 | [diff] [blame] | 169 | self.send_lsp_resp(-1, 'simple-notif') |
Yegappan Lakshmanan | 9247a22 | 2022-03-30 10:16:05 +0100 | [diff] [blame] | 170 | |
| 171 | def do_multi_notif(self, payload): |
| 172 | # send multiple notifications |
Yegappan Lakshmanan | 1926ae4 | 2023-09-21 16:36:28 +0200 | [diff] [blame] | 173 | self.send_lsp_resp(-1, 'multi-notif1') |
| 174 | self.send_lsp_resp(-1, 'multi-notif2') |
Yegappan Lakshmanan | 9247a22 | 2022-03-30 10:16:05 +0100 | [diff] [blame] | 175 | |
| 176 | def do_msg_with_id(self, payload): |
Yegappan Lakshmanan | 1926ae4 | 2023-09-21 16:36:28 +0200 | [diff] [blame] | 177 | self.send_lsp_resp(payload['id'], 'msg-with-id') |
Yegappan Lakshmanan | 9247a22 | 2022-03-30 10:16:05 +0100 | [diff] [blame] | 178 | |
| 179 | def do_msg_specific_cb(self, payload): |
Yegappan Lakshmanan | 1926ae4 | 2023-09-21 16:36:28 +0200 | [diff] [blame] | 180 | self.send_lsp_resp(payload['id'], 'msg-specific-cb') |
Yegappan Lakshmanan | 9247a22 | 2022-03-30 10:16:05 +0100 | [diff] [blame] | 181 | |
| 182 | def do_server_req(self, payload): |
Yegappan Lakshmanan | 1926ae4 | 2023-09-21 16:36:28 +0200 | [diff] [blame] | 183 | self.send_lsp_resp(201, {'method': 'checkhealth', 'params': {'a': 20}}) |
Yegappan Lakshmanan | 9247a22 | 2022-03-30 10:16:05 +0100 | [diff] [blame] | 184 | |
| 185 | def do_extra_hdr_fields(self, payload): |
| 186 | self.send_extra_hdr_fields(payload['id'], 'extra-hdr-fields') |
| 187 | |
Magnus Groß | 8fbd944 | 2023-08-27 00:49:51 +0200 | [diff] [blame] | 188 | def do_delayed_payload(self, payload): |
Yegappan Lakshmanan | 03cca29 | 2022-04-18 14:07:46 +0100 | [diff] [blame] | 189 | self.send_delayed_payload(payload['id'], 'delayed-payload') |
| 190 | |
Yegappan Lakshmanan | 9247a22 | 2022-03-30 10:16:05 +0100 | [diff] [blame] | 191 | 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 Lakshmanan | 1926ae4 | 2023-09-21 16:36:28 +0200 | [diff] [blame] | 206 | 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 Lakshmanan | 9247a22 | 2022-03-30 10:16:05 +0100 | [diff] [blame] | 219 | def process_msg(self, msg): |
| 220 | try: |
| 221 | decoded = json.loads(msg) |
Yegappan Lakshmanan | 9247a22 | 2022-03-30 10:16:05 +0100 | [diff] [blame] | 222 | 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 Lakshmanan | bac9a9e | 2022-04-19 10:25:13 +0100 | [diff] [blame] | 229 | 'large-payload': self.do_large_payload, |
Yegappan Lakshmanan | 9247a22 | 2022-03-30 10:16:05 +0100 | [diff] [blame] | 230 | '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, |
dundargoc | c57b5bc | 2022-11-02 13:30:51 +0000 | [diff] [blame] | 234 | 'msg-specific-cb': self.do_msg_specific_cb, |
Yegappan Lakshmanan | 9247a22 | 2022-03-30 10:16:05 +0100 | [diff] [blame] | 235 | 'server-req': self.do_server_req, |
| 236 | 'extra-hdr-fields': self.do_extra_hdr_fields, |
Magnus Groß | 8fbd944 | 2023-08-27 00:49:51 +0200 | [diff] [blame] | 237 | 'delayed-payload': self.do_delayed_payload, |
Yegappan Lakshmanan | 9247a22 | 2022-03-30 10:16:05 +0100 | [diff] [blame] | 238 | '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 Lakshmanan | 1926ae4 | 2023-09-21 16:36:28 +0200 | [diff] [blame] | 242 | '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 Lakshmanan | 9247a22 | 2022-03-30 10:16:05 +0100 | [diff] [blame] | 245 | } |
| 246 | if decoded['method'] in test_map: |
| 247 | test_map[decoded['method']](decoded) |
| 248 | else: |
Yegappan Lakshmanan | bac9a9e | 2022-04-19 10:25:13 +0100 | [diff] [blame] | 249 | self.debuglog("Error: Unsupported method - " + decoded['method'] + "\n") |
Yegappan Lakshmanan | 9247a22 | 2022-03-30 10:16:05 +0100 | [diff] [blame] | 250 | else: |
Yegappan Lakshmanan | bac9a9e | 2022-04-19 10:25:13 +0100 | [diff] [blame] | 251 | self.debuglog("Error: 'method' field is not found\n") |
Yegappan Lakshmanan | 9247a22 | 2022-03-30 10:16:05 +0100 | [diff] [blame] | 252 | |
| 253 | except ValueError: |
Yegappan Lakshmanan | bac9a9e | 2022-04-19 10:25:13 +0100 | [diff] [blame] | 254 | self.debuglog("Error: json decoding failed\n") |
Yegappan Lakshmanan | 9247a22 | 2022-03-30 10:16:05 +0100 | [diff] [blame] | 255 | |
| 256 | def process_msgs(self, msgbuf): |
| 257 | while True: |
| 258 | sidx = msgbuf.find('Content-Length: ') |
| 259 | if sidx == -1: |
Yegappan Lakshmanan | bac9a9e | 2022-04-19 10:25:13 +0100 | [diff] [blame] | 260 | # partial message received |
Yegappan Lakshmanan | 9247a22 | 2022-03-30 10:16:05 +0100 | [diff] [blame] | 261 | return msgbuf |
| 262 | sidx += 16 |
| 263 | eidx = msgbuf.find('\r\n') |
| 264 | if eidx == -1: |
Yegappan Lakshmanan | bac9a9e | 2022-04-19 10:25:13 +0100 | [diff] [blame] | 265 | # partial message received |
Yegappan Lakshmanan | 9247a22 | 2022-03-30 10:16:05 +0100 | [diff] [blame] | 266 | return msgbuf |
| 267 | msglen = int(msgbuf[sidx:eidx]) |
| 268 | |
| 269 | hdrend = msgbuf.find('\r\n\r\n') |
| 270 | if hdrend == -1: |
Yegappan Lakshmanan | bac9a9e | 2022-04-19 10:25:13 +0100 | [diff] [blame] | 271 | # partial message received |
Yegappan Lakshmanan | 9247a22 | 2022-03-30 10:16:05 +0100 | [diff] [blame] | 272 | return msgbuf |
| 273 | |
Yegappan Lakshmanan | bac9a9e | 2022-04-19 10:25:13 +0100 | [diff] [blame] | 274 | 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 Lakshmanan | 9247a22 | 2022-03-30 10:16:05 +0100 | [diff] [blame] | 283 | # 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 Lakshmanan | 9247a22 | 2022-03-30 10:16:05 +0100 | [diff] [blame] | 293 | self.debug = False |
Yegappan Lakshmanan | bac9a9e | 2022-04-19 10:25:13 +0100 | [diff] [blame] | 294 | self.debuglog("=== socket opened ===\n") |
Yegappan Lakshmanan | 9247a22 | 2022-03-30 10:16:05 +0100 | [diff] [blame] | 295 | msgbuf = '' |
| 296 | while True: |
| 297 | try: |
| 298 | received = self.request.recv(4096).decode('utf-8') |
| 299 | except socket.error: |
Yegappan Lakshmanan | bac9a9e | 2022-04-19 10:25:13 +0100 | [diff] [blame] | 300 | self.debuglog("=== socket error ===\n") |
Yegappan Lakshmanan | 9247a22 | 2022-03-30 10:16:05 +0100 | [diff] [blame] | 301 | break |
| 302 | except IOError: |
Yegappan Lakshmanan | bac9a9e | 2022-04-19 10:25:13 +0100 | [diff] [blame] | 303 | self.debuglog("=== socket closed ===\n") |
Yegappan Lakshmanan | 9247a22 | 2022-03-30 10:16:05 +0100 | [diff] [blame] | 304 | break |
| 305 | if received == '': |
Yegappan Lakshmanan | bac9a9e | 2022-04-19 10:25:13 +0100 | [diff] [blame] | 306 | self.debuglog("=== socket closed ===\n") |
Yegappan Lakshmanan | 9247a22 | 2022-03-30 10:16:05 +0100 | [diff] [blame] | 307 | break |
Yegappan Lakshmanan | 9247a22 | 2022-03-30 10:16:05 +0100 | [diff] [blame] | 308 | |
| 309 | # Write the received lines into the file for debugging |
| 310 | if self.debug: |
Yegappan Lakshmanan | bac9a9e | 2022-04-19 10:25:13 +0100 | [diff] [blame] | 311 | self.debuglog("RECV: ({0} bytes) '{1}'\n".format(len(received), received)) |
Yegappan Lakshmanan | 9247a22 | 2022-03-30 10:16:05 +0100 | [diff] [blame] | 312 | |
| 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 | |
| 319 | class ThreadedTCPServer(socketserver.ThreadingMixIn, socketserver.TCPServer): |
| 320 | pass |
| 321 | |
| 322 | def 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 | |
| 328 | def 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 Lakshmanan | 9247a22 | 2022-03-30 10:16:05 +0100 | [diff] [blame] | 335 | time.sleep(0.5) |
| 336 | |
James McCoy | 765d82a | 2023-01-09 16:25:59 +0000 | [diff] [blame] | 337 | 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 Lakshmanan | 9247a22 | 2022-03-30 10:16:05 +0100 | [diff] [blame] | 343 | 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 Lakshmanan | 9247a22 | 2022-03-30 10:16:05 +0100 | [diff] [blame] | 352 | # 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 | |
| 360 | if __name__ == "__main__": |
| 361 | main("localhost", 0) |