1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16 package org.apache.ibatis.datasource.pooled;
17
18 import java.lang.reflect.InvocationHandler;
19 import java.lang.reflect.Method;
20 import java.lang.reflect.Proxy;
21 import java.sql.Connection;
22 import java.sql.SQLException;
23
24 import org.apache.ibatis.reflection.ExceptionUtil;
25
26
27
28
29 class PooledConnection implements InvocationHandler {
30
31 private static final String CLOSE = "close";
32 private static final Class<?>[] IFACES = new Class<?>[] { Connection.class };
33
34 private final int hashCode;
35 private final PooledDataSource dataSource;
36 private final Connection realConnection;
37 private final Connection proxyConnection;
38 private long checkoutTimestamp;
39 private long createdTimestamp;
40 private long lastUsedTimestamp;
41 private int connectionTypeCode;
42 private boolean valid;
43
44
45
46
47
48
49
50 public PooledConnection(Connection connection, PooledDataSource dataSource) {
51 this.hashCode = connection.hashCode();
52 this.realConnection = connection;
53 this.dataSource = dataSource;
54 this.createdTimestamp = System.currentTimeMillis();
55 this.lastUsedTimestamp = System.currentTimeMillis();
56 this.valid = true;
57 this.proxyConnection = (Connection) Proxy.newProxyInstance(Connection.class.getClassLoader(), IFACES, this);
58 }
59
60
61
62
63 public void invalidate() {
64 valid = false;
65 }
66
67
68
69
70
71
72 public boolean isValid() {
73 return valid && realConnection != null && dataSource.pingConnection(this);
74 }
75
76
77
78
79
80
81 public Connection getRealConnection() {
82 return realConnection;
83 }
84
85
86
87
88
89
90 public Connection getProxyConnection() {
91 return proxyConnection;
92 }
93
94
95
96
97
98
99 public int getRealHashCode() {
100 return realConnection == null ? 0 : realConnection.hashCode();
101 }
102
103
104
105
106
107
108 public int getConnectionTypeCode() {
109 return connectionTypeCode;
110 }
111
112
113
114
115
116
117 public void setConnectionTypeCode(int connectionTypeCode) {
118 this.connectionTypeCode = connectionTypeCode;
119 }
120
121
122
123
124
125
126 public long getCreatedTimestamp() {
127 return createdTimestamp;
128 }
129
130
131
132
133
134
135 public void setCreatedTimestamp(long createdTimestamp) {
136 this.createdTimestamp = createdTimestamp;
137 }
138
139
140
141
142
143
144 public long getLastUsedTimestamp() {
145 return lastUsedTimestamp;
146 }
147
148
149
150
151
152
153 public void setLastUsedTimestamp(long lastUsedTimestamp) {
154 this.lastUsedTimestamp = lastUsedTimestamp;
155 }
156
157
158
159
160
161
162 public long getTimeElapsedSinceLastUse() {
163 return System.currentTimeMillis() - lastUsedTimestamp;
164 }
165
166
167
168
169
170
171 public long getAge() {
172 return System.currentTimeMillis() - createdTimestamp;
173 }
174
175
176
177
178
179
180 public long getCheckoutTimestamp() {
181 return checkoutTimestamp;
182 }
183
184
185
186
187
188
189 public void setCheckoutTimestamp(long timestamp) {
190 this.checkoutTimestamp = timestamp;
191 }
192
193
194
195
196
197
198 public long getCheckoutTime() {
199 return System.currentTimeMillis() - checkoutTimestamp;
200 }
201
202 @Override
203 public int hashCode() {
204 return hashCode;
205 }
206
207
208
209
210
211
212
213 @Override
214 public boolean equals(Object obj) {
215 if (obj instanceof PooledConnection) {
216 return realConnection.hashCode() == ((PooledConnection) obj).realConnection.hashCode();
217 } else if (obj instanceof Connection) {
218 return hashCode == obj.hashCode();
219 } else {
220 return false;
221 }
222 }
223
224
225
226
227
228
229
230
231
232 @Override
233 public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
234 String methodName = method.getName();
235 if (CLOSE.equals(methodName)) {
236 dataSource.pushConnection(this);
237 return null;
238 }
239 try {
240 if (!Object.class.equals(method.getDeclaringClass())) {
241
242
243 checkConnection();
244 }
245 return method.invoke(realConnection, args);
246 } catch (Throwable t) {
247 throw ExceptionUtil.unwrapThrowable(t);
248 }
249
250 }
251
252 private void checkConnection() throws SQLException {
253 if (!valid) {
254 throw new SQLException("Error accessing PooledConnection. Connection is invalid.");
255 }
256 }
257
258 }