0
|
1 package rep;
|
|
2
|
|
3 import java.io.IOException;
|
|
4 import java.net.InetSocketAddress;
|
11
|
5 import java.net.SocketAddress;
|
2
|
6 import java.nio.channels.SelectableChannel;
|
0
|
7 import java.nio.channels.SelectionKey;
|
|
8 import java.nio.channels.Selector;
|
|
9 import java.nio.channels.ServerSocketChannel;
|
|
10 import java.nio.channels.SocketChannel;
|
83
|
11 import java.util.LinkedList;
|
15
|
12 import java.util.StringTokenizer;
|
0
|
13
|
56
|
14 import rep.xml.SessionXMLDecoder;
|
45
|
15 import rep.xml.SessionXMLEncoder;
|
|
16
|
1
|
17 //+-------+--------+--------+-------+--------+---------+------+
|
|
18 //| cmd | session| editor | seqid | lineno | textsiz | text |
|
|
19 //| | id | id | | | | |
|
|
20 //+-------+--------+--------+-------+--------+---------+------+
|
|
21 //o-------header section (network order)-------------o
|
|
22 /*int cmd; // command
|
|
23 int sid; // session ID
|
|
24 int eid; // editor ID
|
|
25 int seqno; // Sequence number
|
|
26 int lineno; // line number
|
|
27 int textsize; // textsize
|
|
28 byte[] text;*/
|
|
29
|
8
|
30 public class SessionManager implements ConnectionListener, REPActionListener{
|
0
|
31
|
|
32
|
|
33 private SessionList sessionlist;
|
5
|
34 //SocketChannel sessionchannel;
|
83
|
35 private SessionManagerGUI gui;
|
2
|
36 private Selector selector;
|
7
|
37 private SessionManagerList smList;
|
17
|
38 private String myHost;
|
21
|
39 private boolean isMaster = true;
|
77
|
40 private EditorList allEditorList;
|
78
|
41 private String maxHost;
|
76
|
42 //private boolean addressIsGlobal;
|
6
|
43 //private SocketChannel sessionchannel;
|
7
|
44 //private boolean co;
|
2
|
45 public SessionManager(int port) {
|
83
|
46 gui = new SessionManagerGUI();
|
2
|
47 }
|
|
48
|
|
49 public void openSelector() throws IOException{
|
|
50 selector = Selector.open();
|
|
51 }
|
0
|
52
|
|
53 public void sessionManagerNet(int port) throws InterruptedException, IOException {
|
2
|
54
|
0
|
55 ServerSocketChannel ssc = ServerSocketChannel.open();
|
38
|
56 ssc.configureBlocking(false);
|
0
|
57 ssc.socket().bind(new InetSocketAddress(port));
|
|
58 ssc.register(selector, SelectionKey.OP_ACCEPT);
|
6
|
59
|
|
60
|
0
|
61 sessionlist = new SessionList();
|
7
|
62 smList = new SessionManagerList();
|
77
|
63 allEditorList = new EditorList();
|
0
|
64
|
|
65 while(true){
|
|
66 selector.select();
|
|
67 for(SelectionKey key : selector.selectedKeys()){
|
|
68 if(key.isAcceptable()){
|
28
|
69 /*** serverChannelはenableになったSelectionKeyのchannel ***/
|
|
70 ServerSocketChannel serverChannel = (ServerSocketChannel)key.channel();
|
9
|
71 /*** EditorChannel を用いない記述 ***/
|
28
|
72 SocketChannel channel = serverChannel.accept(); //keyからchannelを取って、accept
|
2
|
73 registerChannel (selector, channel, SelectionKey.OP_READ);
|
0
|
74 channel = null;
|
9
|
75
|
|
76 /*** EditorChannel を用いた記述 ****/
|
|
77 //EditorChannel echannel = (EditorChannel) ssc.accept();
|
|
78 //echannel.setIO();
|
|
79 //registerChannel(selector, echannel, SelectionKey.OP_READ);
|
|
80 //echannel = null;
|
|
81
|
|
82 /*** SelectableEditorChannel ***/
|
|
83 //SocketChannel channel = ssc.accept();
|
|
84 //SelectableEditorChannel echannel2 = new SelectableEditorChannel(channel);
|
|
85 //registerChannel(selector, echannel2, SelectionKey.OP_READ);
|
|
86 //channel = null;
|
|
87 //echannel2 = null;
|
|
88
|
6
|
89 }else if(key.isReadable()){
|
9
|
90
|
|
91 /*** EditorChannel を用いない記述 ***/
|
0
|
92 SocketChannel channel = (SocketChannel)key.channel();
|
75
|
93 REPPacketReceive receive = new REPPacketReceive(channel); //getPacket(), putPacket() にする。
|
|
94 receive.setkey(key);
|
53
|
95 //REPCommand repCom = repRec.unpackUConv();
|
75
|
96 REPCommand receivedCommand = receive.unpack();
|
|
97 manager(channel, receivedCommand);
|
9
|
98
|
|
99 /*** EditorChannel を用いた記述 ****/
|
|
100 //EditorChannel echannel = (EditorChannel) key.channel();
|
|
101 //REPCommand command = echannel.getPacket();
|
|
102 //manager(echannel, command);
|
|
103
|
6
|
104 }else if(key.isConnectable()){
|
|
105 System.out.println("Connectable");
|
21
|
106 }
|
0
|
107 }
|
|
108 }
|
|
109 }
|
1
|
110
|
2
|
111 private synchronized void registerChannel(Selector selector, SelectableChannel channel, int ops) throws IOException {
|
|
112 if(channel == null) {
|
|
113 return;
|
|
114 }
|
9
|
115 //System.out.println("registerChannel()");
|
2
|
116 channel.configureBlocking(false);
|
6
|
117 selector.wakeup();
|
2
|
118 channel.register(selector, ops);
|
|
119 }
|
|
120
|
75
|
121 private void manager(SocketChannel channel, REPCommand receivedCommand) {
|
|
122 if(receivedCommand == null) return;
|
69
|
123 Editor editor;
|
|
124 Session session;
|
75
|
125 REPCommand sendCommand = receivedCommand.clone();
|
78
|
126 REPPacketSend send = new REPPacketSend(channel);
|
|
127 //SessionXMLEncoder encoder = new SessionXMLEncoder();
|
75
|
128
|
|
129 switch(receivedCommand.cmd){
|
38
|
130
|
0
|
131 case REP.SMCMD_JOIN:
|
83
|
132
|
|
133 int tempeid = allEditorList.addEditor(channel, receivedCommand);
|
|
134 //sendCommand.setCMD(REP.SMCMD_JOIN_ACK);
|
|
135 //sendCommand.setEID(eid);
|
|
136 //REPPacketSend packetSend = new REPPacketSend(channel);
|
|
137 //packetSend.send(sendCommand);
|
|
138 gui.setComboEditor(tempeid, channel);
|
|
139
|
38
|
140 break;
|
31
|
141
|
1
|
142 case REP.SMCMD_JOIN_ACK:
|
31
|
143 // editorList.setEID(repCmd);
|
|
144 // editorList.sendJoinAck(repCmd);
|
|
145 // sessionmanagerGUI.setComboEditor(repCmd.eid, channel);
|
1
|
146 break;
|
31
|
147
|
0
|
148 case REP.SMCMD_PUT:
|
69
|
149 editor = new Editor(channel);
|
56
|
150 editor.setEID(1);
|
75
|
151 editor.setName(receivedCommand.string);
|
69
|
152 session = new Session(editor);
|
67
|
153 session.setOwner(true);
|
63
|
154 session.addEditor(editor);
|
56
|
155 sessionlist.addSession(session);
|
83
|
156 gui.setComboSession(session.getSID(), session.getName());
|
|
157 gui.setComboEditor(editor.getEID(), editor.getChannel());
|
56
|
158 session.addToRoutingTable(editor);
|
84
|
159 sendCommand.setCMD(REP.SMCMD_PUT_ACK);
|
|
160 sendCommand.setEID(1);
|
|
161 sendCommand.setSID(session.getSID());
|
|
162 editor.send(sendCommand);
|
56
|
163
|
|
164 //if(isMaster){
|
78
|
165 SessionXMLEncoder sessionEncoder = new SessionXMLEncoder(session);
|
56
|
166 REPCommand command = new REPCommand();
|
|
167 command.setSID(session.getSID());
|
78
|
168 command.setString(sessionEncoder.sessionListToXML());
|
56
|
169 if(isMaster){
|
|
170 command.setCMD(REP.SMCMD_UPDATE_ACK);
|
|
171 smList.sendToSlave(command);
|
|
172 }else{
|
|
173 command.setCMD(REP.SMCMD_UPDATE);
|
|
174 smList.sendToMaster(command);
|
31
|
175 }
|
38
|
176 break;
|
31
|
177
|
9
|
178 // case REP.SMCMD_PUT_ACK:
|
|
179 // break;
|
31
|
180
|
0
|
181 case REP.SMCMD_SELECT:
|
66
|
182 // sessionlist.addEditor(channel, repCmd.sid, repCmd); //sessionlistへ追加
|
70
|
183 editor = new Editor(channel);
|
75
|
184 session = sessionlist.getSession(receivedCommand.sid);
|
70
|
185 if(session.isOwner()){
|
83
|
186 int eid2 = session.addEditor(editor);
|
|
187 editor.setEID(eid2);
|
78
|
188 //REPPacketSend send = new REPPacketSend(channel);
|
84
|
189 //receivedCommand.setCMD(REP.SMCMD_SELECT_ACK);
|
|
190 //receivedCommand.setEID(eid2);
|
|
191 sendCommand.setCMD(REP.SMCMD_SELECT_ACK);
|
|
192 sendCommand.setEID(eid2);
|
|
193 send.send(sendCommand);
|
66
|
194 }else {
|
85
|
195 Editor master = session.getMaster();
|
|
196 master.send(receivedCommand);
|
66
|
197 }
|
|
198
|
8
|
199 break;
|
38
|
200
|
8
|
201 case REP.SMCMD_SELECT_ACK:
|
85
|
202
|
|
203 String hostport = receivedCommand.string;
|
|
204 Editor editor2 = allEditorList.getEditor(hostport);
|
|
205 if(editor2 != null) {
|
|
206 REPCommand command2 = new REPCommand();
|
|
207 command2.setCMD(REP.SMCMD_JOIN_ACK);
|
|
208 command2.setEID(receivedCommand.eid);
|
|
209 editor2.send(command2);
|
|
210 }else{
|
|
211 smList.sendExcept(channel, receivedCommand);
|
|
212 }
|
|
213
|
|
214 //receivedCommand.setCMD(REP.SMCMD_JOIN_ACK);
|
|
215 //receivedCommand.setEID(receivedCommand.eid);
|
|
216 //session = sessionlist.getSession(receivedCommand.sid);
|
|
217 //session.sendToEditor(receivedCommand);
|
69
|
218 //Editor editor3 = session3.getEditorList().get(0);
|
|
219 //REPPacketSend send = new REPPacketSend(editor3.getChannel());
|
|
220 //send.send(repCmd);
|
1
|
221 break;
|
38
|
222
|
8
|
223 case REP.SMCMD_SM_JOIN:
|
78
|
224
|
83
|
225 //SessionManagerのリストへ追加
|
|
226 smList.add(channel);
|
|
227
|
78
|
228 //XMLからSessionListオブジェクトを生成する。
|
77
|
229 SessionXMLDecoder decoder = new SessionXMLDecoder();
|
79
|
230 SessionList receivedSessionList = decoder.decode(receivedCommand.string);
|
78
|
231
|
83
|
232 //SessionListへ追加し変換テーブルを生成する。
|
|
233 sessionlist.update(channel, receivedSessionList);
|
|
234
|
78
|
235 //myHost を設定。
|
76
|
236 if(myHost == null) setMyHostName(getLocalHostName(channel));
|
78
|
237
|
79
|
238 //maxHost を設定。
|
82
|
239 setMaxHost(channel, receivedSessionList.getMaxHost());
|
77
|
240
|
78
|
241 //SessionListからXMLを生成。
|
|
242 //joinしてきたSessionManagerに対してACKを送信。
|
|
243 SessionXMLEncoder sessionlistEncoder = new SessionXMLEncoder(sessionlist);
|
|
244 sendCommand = new REPCommand();
|
|
245 sendCommand.setCMD(REP.SMCMD_SM_JOIN_ACK);
|
|
246 sendCommand.setString(sessionlistEncoder.sessionListToXML());
|
|
247 send.send(sendCommand);
|
|
248
|
83
|
249 //その他の SessionManager に対して SMCMD_UPDATEを 送信。
|
78
|
250 sendCommand = new REPCommand();
|
83
|
251 sendCommand.setCMD(REP.SMCMD_UPDATE);
|
78
|
252 sendCommand.setString(receivedCommand.string);
|
|
253 smList.sendExcept(channel, sendCommand);
|
|
254
|
83
|
255 //その他のSessionManagerに対してSMCMD_SM_JOINを送信。
|
|
256 //sendCommand = new REPCommand();
|
|
257 //sendCommand.setCMD(REP.SMCMD_SM_JOIN);
|
|
258 //sendCommand.setString(receivedCommand.string);
|
|
259 //smList.sendExcept(channel, sendCommand);
|
|
260
|
31
|
261 if(isMaster){
|
45
|
262 }else {
|
31
|
263 }
|
78
|
264
|
8
|
265 break;
|
38
|
266
|
8
|
267 case REP.SMCMD_SM_JOIN_ACK:
|
82
|
268
|
|
269 //XMLからSessionListオブジェクトを生成。
|
|
270 SessionXMLDecoder decoder2 = new SessionXMLDecoder();
|
|
271 SessionList receivedSessionList2 = decoder2.decode(receivedCommand.string);
|
|
272
|
|
273 //maxHostを決定。
|
|
274 setMaxHost(channel, receivedSessionList2.getMaxHost());
|
|
275
|
38
|
276 if(isMaster){
|
39
|
277 }else{
|
38
|
278 }
|
|
279
|
6
|
280 break;
|
38
|
281
|
8
|
282 case REP.SMCMD_UPDATE:
|
70
|
283
|
83
|
284 //SessionのownerのEditor
|
69
|
285 editor = new Editor(channel);
|
75
|
286 editor.setName(receivedCommand.string);
|
70
|
287
|
69
|
288 session = new Session(editor);
|
70
|
289 session.addEditor(editor);
|
|
290
|
69
|
291 sessionlist.addSession(session);
|
70
|
292
|
83
|
293 gui.setComboSession(session.getSID(), session.getName());
|
70
|
294
|
83
|
295 //if(isMaster){
|
|
296 // receivedCommand.setCMD(REP.SMCMD_UPDATE_ACK);
|
|
297 // smList.sendToSlave(receivedCommand);
|
|
298 //}else{
|
|
299 // receivedCommand.setCMD(REP.SMCMD_UPDATE);
|
|
300 // smList.sendToMaster(receivedCommand);
|
|
301 //}
|
9
|
302 break;
|
38
|
303
|
9
|
304 case REP.SMCMD_UPDATE_ACK:
|
75
|
305 if(receivedCommand.sid > sessionlist.getList().size()){
|
73
|
306 editor = new Editor(channel);
|
75
|
307 editor.setName(receivedCommand.string);
|
73
|
308
|
|
309 session = new Session(editor);
|
|
310 session.addEditor(editor);
|
|
311
|
|
312 sessionlist.addSession(session);
|
|
313
|
83
|
314 gui.setComboSession(session.getSID(), session.getName());
|
73
|
315 }
|
75
|
316 smList.sendToSlave(receivedCommand);
|
1
|
317 break;
|
38
|
318
|
3
|
319 case REP.REPCMD_READ:
|
17
|
320 //sessionlist.sendCmd(channel, repCmd);
|
3
|
321 break;
|
38
|
322
|
0
|
323 default:
|
9
|
324 //sessionlist.sendCmd(channel, repCmd);
|
75
|
325 sessionlist.sendToNextEditor(channel, receivedCommand);
|
0
|
326 break;
|
|
327 }
|
|
328 }
|
83
|
329
|
79
|
330 private boolean setMaxHost(SocketChannel channel, String host) {
|
81
|
331 if(maxHost == null) {
|
|
332 maxHost = myHost;
|
|
333 sessionlist.setMaxHost(maxHost);
|
|
334 }
|
82
|
335 if(host.compareTo(maxHost) > 0){
|
79
|
336 //host > MaxHost なら maxHost = host
|
|
337 //masterを設定する。
|
|
338 maxHost = host;
|
|
339 sessionlist.setMaxHost(maxHost);
|
|
340 setMaster(false, channel);
|
|
341 return true;
|
|
342 }else{
|
|
343 return false;
|
|
344 }
|
78
|
345 }
|
|
346
|
76
|
347 private void setMyHostName(String localHostName) {
|
|
348 myHost = localHostName;
|
81
|
349 if(maxHost == null) {
|
|
350 maxHost = myHost;
|
|
351 sessionlist.setMaxHost(maxHost);
|
|
352 }
|
77
|
353 allEditorList.setHost(myHost);
|
76
|
354 }
|
0
|
355
|
77
|
356 private void setMaster(boolean b, SocketChannel channel) {
|
75
|
357 isMaster = b;
|
|
358 System.out.println("isMaster = " + b);
|
77
|
359 smList.setMaster(channel);
|
75
|
360 }
|
|
361
|
0
|
362 public static void main(String[] args) throws InterruptedException, IOException {
|
6
|
363 int port = 8766;
|
1
|
364
|
82
|
365 if(args.length > 0){
|
39
|
366 port = Integer.parseInt(args[0]);
|
0
|
367 }
|
|
368 SessionManager sm = new SessionManager(port);
|
2
|
369 sm.openSelector();
|
|
370 sm.openWindow();
|
0
|
371 sm.sessionManagerNet(port);
|
|
372 }
|
|
373
|
2
|
374 private void openWindow() {
|
83
|
375 Thread th = new Thread( gui );
|
2
|
376 th.start();
|
75
|
377 //System.out.println(sessionmanagerGUI.toString());
|
83
|
378 gui.addConnectionListener(this);
|
|
379 gui.addREPActionListener(this);
|
2
|
380 }
|
|
381
|
|
382 private void connectSession(String host) {
|
6
|
383 int port = 8766;
|
1
|
384 InetSocketAddress addr = new InetSocketAddress(host, port);
|
|
385 try {
|
6
|
386 SocketChannel sessionchannel = SocketChannel.open();
|
1
|
387 sessionchannel.configureBlocking(true);
|
|
388 sessionchannel.connect(addr);
|
6
|
389 while(!sessionchannel.finishConnect()){
|
77
|
390 System.out.print("test afro");
|
6
|
391 }
|
|
392 System.out.println("");
|
2
|
393 registerChannel(selector, sessionchannel, SelectionKey.OP_READ);
|
45
|
394
|
77
|
395 sm_join(sessionchannel);
|
45
|
396
|
1
|
397 }catch (IOException e) {
|
|
398 e.printStackTrace();
|
|
399 }
|
|
400 }
|
77
|
401
|
|
402 private void sm_join(SocketChannel channel){
|
79
|
403
|
|
404 //SM_JOINコマンドを生成。
|
77
|
405 REPCommand command = new REPCommand();
|
|
406 command.setCMD(REP.SMCMD_SM_JOIN);
|
79
|
407
|
|
408 //hostnameをセット。
|
82
|
409 setMyHostName(getLocalHostName(channel));
|
|
410
|
|
411 //XMLを生成。送信コマンドにセット。
|
77
|
412 SessionXMLEncoder encoder = new SessionXMLEncoder(sessionlist);
|
|
413 String string = encoder.sessionListToXML();
|
|
414 command.setString(string);
|
|
415
|
79
|
416 //SM_JOINコマンドを送信。
|
77
|
417 REPPacketSend send = new REPPacketSend(channel);
|
|
418 send.send(command);
|
|
419
|
79
|
420 //SessionManagerのListに追加。
|
77
|
421 smList.add(channel);
|
|
422 }
|
2
|
423
|
75
|
424 private String getLocalHostName(SocketChannel channel) {
|
74
|
425 String host = null;
|
|
426 host = channel.socket().getLocalAddress().getHostName();
|
|
427 return host;
|
|
428 }
|
|
429
|
77
|
430 // private String getSocketString(SocketChannel sessionchannel) {
|
|
431 // SocketAddress socket = sessionchannel.socket().getRemoteSocketAddress();
|
|
432 // //String inetAddressString = sessionchannel.socket().getInetAddress().toString();
|
|
433 // StringTokenizer stn = new StringTokenizer(socket.toString(), "/");
|
|
434 // String socketString = null;
|
|
435 // while(stn.hasMoreTokens()){
|
|
436 // socketString = stn.nextToken();
|
|
437 // //System.out.println(socketString);
|
|
438 // }
|
|
439 // return socketString;
|
|
440 // }
|
14
|
441
|
2
|
442 public void connectionOccured(ConnectionEvent event) {
|
|
443 connectSession(event.getHost());
|
|
444 }
|
8
|
445
|
|
446 public void ActionOccured(REPActionEvent event) {
|
|
447 System.out.println("Action!");
|
|
448 SocketChannel editorChannel = event.getEditorChannel();
|
|
449 int sid = event.getSID();
|
84
|
450 //int eid = 0;
|
70
|
451 //int eid = event.getEID();
|
|
452 //sessionlist.addEditor(editorChannel, sid, eid);
|
|
453 Editor editor = new Editor(editorChannel);
|
|
454 Session session = sessionlist.getSession(sid);
|
|
455 session.addEditor(editor);
|
|
456
|
72
|
457 Editor master = session.getMaster();
|
|
458
|
|
459 REPCommand command = new REPCommand();
|
|
460 command.setCMD(REP.SMCMD_SELECT);
|
|
461 command.setSID(sid);
|
84
|
462 command.setString(editor.getHost() + ":" + editor.getPort());
|
72
|
463
|
|
464 master.send(command);
|
|
465
|
84
|
466 //REPPacketSend send = new REPPacketSend(editorChannel);
|
|
467 //send.send(new REPCommand(REP.SMCMD_SELECT_ACK, sid, eid, 0,0,0,""));
|
72
|
468
|
|
469
|
|
470
|
71
|
471 //sessionlist.sendSelect(sid);
|
8
|
472 }
|
0
|
473 }
|