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