1
2
3
4
5
6
7
8
9
10
11
12
13
14 package org.eclipse.jetty.util.thread;
15
16 import java.util.Arrays;
17 import java.util.List;
18 import java.util.concurrent.CopyOnWriteArrayList;
19
20 import org.eclipse.jetty.util.component.LifeCycle;
21 import org.eclipse.jetty.util.log.Log;
22
23
24
25
26
27
28
29
30
31 public class ShutdownThread extends Thread
32 {
33 private static final ShutdownThread _thread = new ShutdownThread();
34
35 private boolean _hooked;
36 private final List<LifeCycle> _lifeCycles = new CopyOnWriteArrayList<LifeCycle>();
37
38
39
40
41
42
43
44 private ShutdownThread()
45 {
46 }
47
48
49 private synchronized void hook()
50 {
51 try
52 {
53 if (!_hooked)
54 Runtime.getRuntime().addShutdownHook(this);
55 _hooked=true;
56 }
57 catch(Exception e)
58 {
59 Log.ignore(e);
60 Log.info("shutdown already commenced");
61 }
62 }
63
64
65 private synchronized void unhook()
66 {
67 try
68 {
69 _hooked=false;
70 Runtime.getRuntime().removeShutdownHook(this);
71 }
72 catch(Exception e)
73 {
74 Log.ignore(e);
75 Log.info("shutdown already commenced");
76 }
77 }
78
79
80
81
82
83
84
85 public static ShutdownThread getInstance()
86 {
87 return _thread;
88 }
89
90
91 public static synchronized void register(LifeCycle... lifeCycles)
92 {
93 _thread._lifeCycles.addAll(Arrays.asList(lifeCycles));
94 if (_thread._lifeCycles.size()>0)
95 _thread.hook();
96 }
97
98
99 public static synchronized void register(int index, LifeCycle... lifeCycles)
100 {
101 _thread._lifeCycles.addAll(index,Arrays.asList(lifeCycles));
102 if (_thread._lifeCycles.size()>0)
103 _thread.hook();
104 }
105
106
107 public static synchronized void deregister(LifeCycle lifeCycle)
108 {
109 _thread._lifeCycles.remove(lifeCycle);
110 if (_thread._lifeCycles.size()==0)
111 _thread.unhook();
112 }
113
114
115 public void run()
116 {
117 for (LifeCycle lifeCycle : _thread._lifeCycles)
118 {
119 try
120 {
121 lifeCycle.stop();
122 Log.debug("Stopped " + lifeCycle);
123 }
124 catch (Exception ex)
125 {
126 Log.debug(ex);
127 }
128 }
129 }
130 }