-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathserenade_client.py
153 lines (113 loc) · 5.9 KB
/
serenade_client.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
# This file is based on a python example of the Serenade protocol written by
# Tommy MacWilliam, available here:
# https://github.com/serenadeai/protocol/blob/master/python-editor/app.py
import asyncio
import json
import random
import traceback
import websockets
import sys
import logging
from os.path import expanduser
icon = ''
id = str(random.random())
websocket = None
app_name = 'Vim'
match_re = 'term'
websocket_address = 'ws://localhost:17373'
event_loop = None
async def send(message, data):
if not websocket:
return
await send_raw(json.dumps({"message": message, "data": data}))
async def send_raw(msg):
if not websocket:
return
logging.info('Message: ' + msg)
await websocket.send(msg)
async def send_heartbeat():
# send a heartbeat every minute so that Serenade keeps the connection alive
while True:
if websocket:
await send("heartbeat", {"id": id})
await asyncio.sleep(60)
async def handle(message):
logging.info('Received: ' + str(message))
data = json.loads(message)["data"]
# if Serenade doesn't have anything for us to execute, then we're done
if "response" not in data or "execute" not in data["response"]:
return
output(message)
async def get_input():
global event_loop
reader = asyncio.StreamReader(limit=1024*1024)
protocol = asyncio.StreamReaderProtocol(reader)
await event_loop.connect_read_pipe(lambda: protocol, sys.stdin)
return reader
async def read_input():
reader = await get_input()
while True:
line = (await reader.readline()).decode().strip()
logging.info('INPUT: ' + line)
if not line.startswith('{'):
if line == 'active':
logging.info('Active')
await send_active()
else:
logging.error('Unknown input: ' + line)
continue
try:
await send_raw(line)
except Exception as e:
logging.error('Failed to send raw input: %s' % e)
def output(msg):
sys.stdout.write(str(msg) + '\n')
sys.stdout.flush()
async def send_active(send_icon=False):
msg = {
"id": id,
"app": app_name,
"match": match_re,
}
if send_icon:
msg["icon"] = icon
await send("active", msg)
async def handler():
global websocket
global id
asyncio.create_task(send_heartbeat())
asyncio.create_task(read_input())
while True:
try:
async with websockets.connect(websocket_address, max_size=10**9) as ws:
websocket = ws
output("Connected")
logging.info('Connected')
await send_active(True)
while True:
try:
message = await websocket.recv()
await handle(message)
except websockets.exceptions.ConnectionClosedError:
print("Disconnected")
logging.info('Disconnected')
websocket = None
break
except OSError:
websocket = None
await asyncio.sleep(1)
if __name__ == "__main__":
app_name = sys.argv[1]
match_re = sys.argv[2]
websocket_address = sys.argv[3]
should_log = sys.argv[4]
if should_log != '0':
logging.basicConfig(
filename=expanduser('~/.vim-serenade.log'),
level=logging.INFO,
format='%(asctime)s %(process)d %(levelname)-8s %(message)s',
datefmt='%Y-%m-%d %H:%M:%S',
)
output("Running")
event_loop = asyncio.new_event_loop()
event_loop.run_until_complete(handler())