-
Notifications
You must be signed in to change notification settings - Fork 191
/
server.py
229 lines (198 loc) · 7.58 KB
/
server.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
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
from flask import Flask, request, redirect
from flask_socketio import SocketIO, send, emit, join_room, leave_room, close_room
import json, random, datetime, argparse
# for socketio
import eventlet
eventlet.monkey_patch()
app = Flask(__name__)
app.config['SECRET_KEY'] = 'secret!'
socketio = SocketIO(app, async_mode='eventlet', cors_allowed_origins="*", max_http_buffer_size=1e8)
sessions = {} # indexed by session name
@socketio.on('message')
def handleMessage(msg):
print(datetime.datetime.now(),'Message: ' + msg)
send(msg, broadcast=True)
@socketio.on('check session name event')
def handleCheckName(json):
name = json['name']
#print("checking ",name,name in sessions)
if name in sessions:
emit('check session name response', "exists")
else:
emit('check session name response', "free")
@socketio.on('create session event')
def handleCreateSession(json):
name = json['name']
if name in sessions:
print("Invalid name",name)
emit('create session response', 0)
else:
secret = random.randint(1,4e9)
emit('create session response', secret)
print("Session Created",name,request.sid)
sessions[name] = {'cnt': 0,
'sid': request.sid, # id of the controller
'state': json['state'],
'view': json['view'],
'query_updates': {}, # each clients individual selections
'query_active' : False,
'secret' : secret
}
join_room(name)
emit('session count', 0, room=request.sid)
@socketio.on('reconnection')
def handleReconnection(json):
'''Reconnections change the socket id'''
name = json['name']
secret = json['secret']
oldsid = json['sid']
# print("Reconnect",name,secret,oldsid,sessions[name]['secret'])
if name not in sessions:
return
#reconnecting clients need to re-join the room as well
join_room(name)
if secret == sessions[name]['secret']: #controller
sessions[name]['sid'] = request.sid
else : #client
updates = sessions[name]['query_updates']
if oldsid in updates: #change updates to match new sid
updates[request.sid] = updates[oldsid]
del updates[oldsid]
#make sure clients are up to date
if 'view' in sessions[name]:
emit('viewer view change response', sessions[name]['view'], room=request.sid)
if 'state' in sessions[name]:
emit('viewer state change response', sessions[name]['state'], room=request.sid)
@socketio.on('join session event')
def handleJoinSession(json):
name = json['name']
#print("Join",name)
if name not in sessions:
emit('join session response', "0")
else:
emit('join session response', request.sid)
join_room(name)
try:
sessions[name]['cnt'] += 1
except:
emit('error: restart connection')
emit('session count', sessions[name]['cnt'], room=sessions[name]['sid'])
#send session state to connecting client
if 'view' in sessions[name]:
emit('viewer view change response', sessions[name]['view'], room=request.sid)
if 'state' in sessions[name]:
emit('viewer state change response', sessions[name]['state'], room=request.sid)
#print("Joined Session: " + str(json))
if sessions[name]['query_active']:
emit('query start response', sessions[name]['query_active'], room=request.sid)
@socketio.on('delete session event')
def handleDeleteSession(json):
name = json['name']
if name not in sessions:
return
if request.sid != sessions[name]['sid']:
return #only the controller can send updates
emit('leave session response', room=name)
close_room(name)
print("Session Deleted:" + str(json))
del sessions[name]
emit('delete session response')
@socketio.on('leave session event')
def handleLeaveSession(json):
name = json['name']
leave_room(name)
if name not in sessions:
return
sessions[name]['cnt'] -= 1
updates = sessions[name]['query_updates']
if request.sid in updates:
del updates[request.sid]
emit('query update response', len(updates), room=name)
emit('session count', sessions[name]['cnt'], room=sessions[name]['sid'])
print("Session Left: " + str(json))
emit('leave session response')
@socketio.on('viewer view change event')
def handleViewerChange(json):
name = json['name']
viewstate = json['view']
if name not in sessions:
return
if request.sid != sessions[name]['sid']:
#print(datetime.datetime.now(),"Incorrect sid for viewer change %s vs %s"%(request.sid,sessions[name]['sid']))
return #only the controller can send updates
sessions[name]['view'] = viewstate
emit('viewer view change response', viewstate, room=name)
@socketio.on('viewer state change event')
def handleStateChange(json):
name = json['name']
state = json['state']
#print("State change",name)
if name not in sessions:
return
if request.sid != sessions[name]['sid']:
#print("Incorrect sid for state change")
return #only the controller can send updates
sessions[name]['state'] = state
emit('viewer state change response', state, room=name)
#state change might make labels meaningless, so clear
sessions[name]['query_updates'] = {}
sessions[name]['query_active'] = False
emit('query end response', '', room=name)
@socketio.on('query start')
def handleQueryStart(json):
name = json['name']
#print("query start",name)
if name not in sessions:
return
if request.sid != sessions[name]['sid']:
return #only the controller can send updates
sessions[name]['query_updates'] = {}
sessions[name]['query_active'] = json['kind']
emit('query start response', json['kind'], room=name)
@socketio.on('query end')
def handleQueryEnd(json):
name = json['name']
if name not in sessions:
return
if request.sid != sessions[name]['sid']:
return #only the controller can send updates
sessions[name]['query_updates'] = {}
sessions[name]['query_active'] = False
emit('query end response', '', room=name)
@socketio.on('query update')
def handleQueryUpdate(json):
'''A single update from a client, respond with number unique responses'''
name = json['name']
sel = json['selected']
if name not in sessions:
return
# process update
updates = sessions[name]['query_updates']
updates[request.sid] = sel
emit('query update response', len(updates), room=sessions[name]['sid'])
@socketio.on('query fetch')
def handleQueryFetch(json):
'''Return the merged results'''
name = json['name']
if name not in sessions:
return
# process update
updates = sessions[name]['query_updates']
cnts = {}
for positions in updates.values():
for pos in positions:
pos = tuple(pos)
if pos in cnts:
cnts[pos] += 1
else:
cnts[pos] = 1
cnts = list(cnts.items())
emit('query fetch response', cnts, room=request.sid)
@app.route('/')
def home():
return redirect('static/viewer.html')
if __name__ == '__main__':
parser = argparse.ArgumentParser(description='Run 3Dmol.js learning environment server.')
parser.add_argument('-p','--port', type=int, default=5000, help='Port to run on (default 5000)')
args = parser.parse_args()
socketio.run(app,port=args.port)