Bram Moolenaar | 3b5f929 | 2016-01-28 22:37:01 +0100 | [diff] [blame] | 1 | #!/usr/bin/python |
| 2 | # Server that will accept connections from a Vim channel. |
| 3 | # Run this server and then in Vim you can open the channel: |
| 4 | # :let handle = connect('localhost:8765', 'json') |
| 5 | # |
| 6 | # Then Vim can send requests to the server: |
| 7 | # :let response = sendexpr(handle, 'hello!') |
| 8 | # |
| 9 | # And you can control Vim by typing a JSON message here, e.g.: |
| 10 | # ["ex","echo 'hi there'"] |
| 11 | # |
| 12 | # See ":help channel-demo" in Vim. |
| 13 | |
Bram Moolenaar | 488a130 | 2016-02-01 22:01:10 +0100 | [diff] [blame] | 14 | from __future__ import print_function |
Bram Moolenaar | 3b5f929 | 2016-01-28 22:37:01 +0100 | [diff] [blame] | 15 | import json |
| 16 | import socket |
| 17 | import sys |
| 18 | import threading |
| 19 | |
Bram Moolenaar | 488a130 | 2016-02-01 22:01:10 +0100 | [diff] [blame] | 20 | try: |
| 21 | # Python 3 |
| 22 | import socketserver |
| 23 | except ImportError: |
| 24 | # Python 2 |
| 25 | import SocketServer as socketserver |
| 26 | |
Bram Moolenaar | 3b5f929 | 2016-01-28 22:37:01 +0100 | [diff] [blame] | 27 | thesocket = None |
| 28 | |
Bram Moolenaar | 488a130 | 2016-02-01 22:01:10 +0100 | [diff] [blame] | 29 | class ThreadedTCPRequestHandler(socketserver.BaseRequestHandler): |
Bram Moolenaar | 3b5f929 | 2016-01-28 22:37:01 +0100 | [diff] [blame] | 30 | |
| 31 | def handle(self): |
Bram Moolenaar | 488a130 | 2016-02-01 22:01:10 +0100 | [diff] [blame] | 32 | print("=== socket opened ===") |
Bram Moolenaar | 3b5f929 | 2016-01-28 22:37:01 +0100 | [diff] [blame] | 33 | global thesocket |
| 34 | thesocket = self.request |
| 35 | while True: |
| 36 | try: |
Bram Moolenaar | 488a130 | 2016-02-01 22:01:10 +0100 | [diff] [blame] | 37 | data = self.request.recv(4096).decode('utf-8') |
Bram Moolenaar | 3b5f929 | 2016-01-28 22:37:01 +0100 | [diff] [blame] | 38 | except socket.error: |
Bram Moolenaar | 488a130 | 2016-02-01 22:01:10 +0100 | [diff] [blame] | 39 | print("=== socket error ===") |
Bram Moolenaar | 3b5f929 | 2016-01-28 22:37:01 +0100 | [diff] [blame] | 40 | break |
| 41 | except IOError: |
Bram Moolenaar | 488a130 | 2016-02-01 22:01:10 +0100 | [diff] [blame] | 42 | print("=== socket closed ===") |
Bram Moolenaar | 3b5f929 | 2016-01-28 22:37:01 +0100 | [diff] [blame] | 43 | break |
| 44 | if data == '': |
Bram Moolenaar | 488a130 | 2016-02-01 22:01:10 +0100 | [diff] [blame] | 45 | print("=== socket closed ===") |
Bram Moolenaar | 3b5f929 | 2016-01-28 22:37:01 +0100 | [diff] [blame] | 46 | break |
Bram Moolenaar | 488a130 | 2016-02-01 22:01:10 +0100 | [diff] [blame] | 47 | print("received: {}".format(data)) |
Bram Moolenaar | 3b5f929 | 2016-01-28 22:37:01 +0100 | [diff] [blame] | 48 | try: |
| 49 | decoded = json.loads(data) |
| 50 | except ValueError: |
Bram Moolenaar | 488a130 | 2016-02-01 22:01:10 +0100 | [diff] [blame] | 51 | print("json decoding failed") |
| 52 | decoded = [-1, ''] |
Bram Moolenaar | 3b5f929 | 2016-01-28 22:37:01 +0100 | [diff] [blame] | 53 | |
Bram Moolenaar | 488a130 | 2016-02-01 22:01:10 +0100 | [diff] [blame] | 54 | # Send a response if the sequence number is positive. |
| 55 | # Negative numbers are used for "eval" responses. |
| 56 | if decoded[0] >= 0: |
| 57 | if decoded[1] == 'hello!': |
| 58 | response = "got it" |
| 59 | else: |
| 60 | response = "what?" |
| 61 | encoded = json.dumps([decoded[0], response]) |
| 62 | print("sending {}".format(encoded)) |
| 63 | self.request.sendall(encoded.encode('utf-8')) |
Bram Moolenaar | 3b5f929 | 2016-01-28 22:37:01 +0100 | [diff] [blame] | 64 | thesocket = None |
| 65 | |
Bram Moolenaar | 488a130 | 2016-02-01 22:01:10 +0100 | [diff] [blame] | 66 | class ThreadedTCPServer(socketserver.ThreadingMixIn, socketserver.TCPServer): |
Bram Moolenaar | 3b5f929 | 2016-01-28 22:37:01 +0100 | [diff] [blame] | 67 | pass |
| 68 | |
| 69 | if __name__ == "__main__": |
| 70 | HOST, PORT = "localhost", 8765 |
| 71 | |
| 72 | server = ThreadedTCPServer((HOST, PORT), ThreadedTCPRequestHandler) |
| 73 | ip, port = server.server_address |
| 74 | |
| 75 | # Start a thread with the server -- that thread will then start one |
| 76 | # more thread for each request |
| 77 | server_thread = threading.Thread(target=server.serve_forever) |
| 78 | |
| 79 | # Exit the server thread when the main thread terminates |
| 80 | server_thread.daemon = True |
| 81 | server_thread.start() |
Bram Moolenaar | 488a130 | 2016-02-01 22:01:10 +0100 | [diff] [blame] | 82 | print("Server loop running in thread: ", server_thread.name) |
Bram Moolenaar | 3b5f929 | 2016-01-28 22:37:01 +0100 | [diff] [blame] | 83 | |
Bram Moolenaar | 488a130 | 2016-02-01 22:01:10 +0100 | [diff] [blame] | 84 | print("Listening on port {}".format(PORT)) |
Bram Moolenaar | 3b5f929 | 2016-01-28 22:37:01 +0100 | [diff] [blame] | 85 | while True: |
| 86 | typed = sys.stdin.readline() |
| 87 | if "quit" in typed: |
Bram Moolenaar | 488a130 | 2016-02-01 22:01:10 +0100 | [diff] [blame] | 88 | print("Goodbye!") |
Bram Moolenaar | 3b5f929 | 2016-01-28 22:37:01 +0100 | [diff] [blame] | 89 | break |
| 90 | if thesocket is None: |
Bram Moolenaar | 488a130 | 2016-02-01 22:01:10 +0100 | [diff] [blame] | 91 | print("No socket yet") |
Bram Moolenaar | 3b5f929 | 2016-01-28 22:37:01 +0100 | [diff] [blame] | 92 | else: |
Bram Moolenaar | 488a130 | 2016-02-01 22:01:10 +0100 | [diff] [blame] | 93 | print("sending {}".format(typed)) |
| 94 | thesocket.sendall(typed.encode('utf-8')) |
Bram Moolenaar | 3b5f929 | 2016-01-28 22:37:01 +0100 | [diff] [blame] | 95 | |
| 96 | server.shutdown() |
| 97 | server.server_close() |