comparison src/ch/ethz/ssh2/util/TimeoutService.java @ 308:42b15aaa7ac7 ganymed

merge
author Carl Byington <carl@five-ten-sg.com>
date Wed, 30 Jul 2014 14:21:50 -0700
parents 071eccdff8ea
children
comparison
equal deleted inserted replaced
306:90e47d99ea54 308:42b15aaa7ac7
18 * <p> 18 * <p>
19 * Implemented having large scale programs in mind: if you open many concurrent SSH connections 19 * Implemented having large scale programs in mind: if you open many concurrent SSH connections
20 * that rely on timeouts, then there will be only one timeout thread. Once all timeouts 20 * that rely on timeouts, then there will be only one timeout thread. Once all timeouts
21 * have expired/are cancelled, the thread will (sooner or later) exit. 21 * have expired/are cancelled, the thread will (sooner or later) exit.
22 * Only after new timeouts arrive a new thread (singleton) will be instantiated. 22 * Only after new timeouts arrive a new thread (singleton) will be instantiated.
23 * 23 *
24 * @author Christian Plattner 24 * @author Christian Plattner
25 * @version $Id: TimeoutService.java 89 2014-04-07 14:36:24Z dkocher@sudo.ch $ 25 * @version $Id: TimeoutService.java 89 2014-04-07 14:36:24Z dkocher@sudo.ch $
26 */ 26 */
27 public class TimeoutService 27 public class TimeoutService {
28 { 28 private static final Logger log = Logger.getLogger(TimeoutService.class);
29 private static final Logger log = Logger.getLogger(TimeoutService.class);
30 29
31 public static class TimeoutToken 30 public static class TimeoutToken {
32 { 31 private long runTime;
33 private long runTime; 32 private Runnable handler;
34 private Runnable handler;
35 33
36 private TimeoutToken(long runTime, Runnable handler) 34 private TimeoutToken(long runTime, Runnable handler) {
37 { 35 this.runTime = runTime;
38 this.runTime = runTime; 36 this.handler = handler;
39 this.handler = handler; 37 }
40 } 38 }
41 }
42 39
43 private static class TimeoutThread extends Thread 40 private static class TimeoutThread extends Thread {
44 { 41 @Override
45 @Override 42 public void run() {
46 public void run() 43 synchronized (todolist) {
47 { 44 while (true) {
48 synchronized (todolist) 45 if (todolist.size() == 0) {
49 { 46 timeoutThread = null;
50 while (true) 47 return;
51 { 48 }
52 if (todolist.size() == 0)
53 {
54 timeoutThread = null;
55 return;
56 }
57 49
58 long now = System.currentTimeMillis(); 50 long now = System.currentTimeMillis();
51 TimeoutToken tt = todolist.getFirst();
59 52
60 TimeoutToken tt = todolist.getFirst(); 53 if (tt.runTime > now) {
54 /* Not ready yet, sleep a little bit */
55 try {
56 todolist.wait(tt.runTime - now);
57 }
58 catch (InterruptedException ignored) {
59 }
61 60
62 if (tt.runTime > now) 61 /* We cannot simply go on, since it could be that the token
63 { 62 * was removed (cancelled) or another one has been inserted in
64 /* Not ready yet, sleep a little bit */ 63 * the meantime.
64 */
65 continue;
66 }
65 67
66 try 68 todolist.removeFirst();
67 {
68 todolist.wait(tt.runTime - now);
69 }
70 catch (InterruptedException ignored)
71 {
72 }
73 69
74 /* We cannot simply go on, since it could be that the token 70 try {
75 * was removed (cancelled) or another one has been inserted in 71 tt.handler.run();
76 * the meantime. 72 }
77 */ 73 catch (Exception e) {
74 StringWriter sw = new StringWriter();
75 e.printStackTrace(new PrintWriter(sw));
76 log.warning("Exeception in Timeout handler:" + e.getMessage() + "(" + sw.toString() + ")");
77 }
78 }
79 }
80 }
81 }
78 82
79 continue; 83 /* The list object is also used for locking purposes */
80 } 84 private static final LinkedList<TimeoutToken> todolist = new LinkedList<TimeoutService.TimeoutToken>();
81 85
82 todolist.removeFirst(); 86 private static Thread timeoutThread = null;
83 87
84 try 88 /**
85 { 89 * It is assumed that the passed handler will not execute for a long time.
86 tt.handler.run(); 90 *
87 } 91 * @param runTime
88 catch (Exception e) 92 * @param handler
89 { 93 * @return a TimeoutToken that can be used to cancel the timeout.
90 StringWriter sw = new StringWriter(); 94 */
91 e.printStackTrace(new PrintWriter(sw)); 95 public static TimeoutToken addTimeoutHandler(long runTime, Runnable handler) {
92 log.warning("Exeception in Timeout handler:" + e.getMessage() + "(" + sw.toString() + ")"); 96 TimeoutToken token = new TimeoutToken(runTime, handler);
93 }
94 }
95 }
96 }
97 }
98 97
99 /* The list object is also used for locking purposes */ 98 synchronized (todolist) {
100 private static final LinkedList<TimeoutToken> todolist = new LinkedList<TimeoutService.TimeoutToken>(); 99 todolist.add(token);
100 Collections.sort(todolist, new Comparator<TimeoutToken>() {
101 public int compare(TimeoutToken o1, TimeoutToken o2) {
102 if (o1.runTime > o2.runTime)
103 return 1;
101 104
102 private static Thread timeoutThread = null; 105 if (o1.runTime == o2.runTime)
106 return 0;
103 107
104 /** 108 return -1;
105 * It is assumed that the passed handler will not execute for a long time. 109 }
106 * 110 });
107 * @param runTime
108 * @param handler
109 * @return a TimeoutToken that can be used to cancel the timeout.
110 */
111 public static TimeoutToken addTimeoutHandler(long runTime, Runnable handler)
112 {
113 TimeoutToken token = new TimeoutToken(runTime, handler);
114 111
115 synchronized (todolist) 112 if (timeoutThread != null)
116 { 113 timeoutThread.interrupt();
117 todolist.add(token); 114 else {
115 timeoutThread = new TimeoutThread();
116 timeoutThread.setDaemon(true);
117 timeoutThread.start();
118 }
119 }
118 120
119 Collections.sort(todolist, new Comparator<TimeoutToken>() 121 return token;
120 { 122 }
121 public int compare(TimeoutToken o1, TimeoutToken o2)
122 {
123 if (o1.runTime > o2.runTime)
124 return 1;
125 if (o1.runTime == o2.runTime)
126 return 0;
127 return -1;
128 }
129 });
130 123
131 if (timeoutThread != null) 124 public static void cancelTimeoutHandler(TimeoutToken token) {
132 timeoutThread.interrupt(); 125 synchronized (todolist) {
133 else 126 todolist.remove(token);
134 {
135 timeoutThread = new TimeoutThread();
136 timeoutThread.setDaemon(true);
137 timeoutThread.start();
138 }
139 }
140 127
141 return token; 128 if (timeoutThread != null)
142 } 129 timeoutThread.interrupt();
143 130 }
144 public static void cancelTimeoutHandler(TimeoutToken token) 131 }
145 {
146 synchronized (todolist)
147 {
148 todolist.remove(token);
149
150 if (timeoutThread != null)
151 timeoutThread.interrupt();
152 }
153 }
154 132
155 } 133 }