35
35
* @author Phillip Webb
36
36
* @author Andy Wilkinson
37
37
*/
38
- class JarURLConnection extends java .net .JarURLConnection {
38
+ final class JarURLConnection extends java .net .JarURLConnection {
39
39
40
- private static final FileNotFoundException FILE_NOT_FOUND_EXCEPTION = new FileNotFoundException ();
40
+ private static ThreadLocal <Boolean > useFastExceptions = new ThreadLocal <Boolean >();
41
+
42
+ private static final FileNotFoundException FILE_NOT_FOUND_EXCEPTION = new FileNotFoundException (
43
+ "Jar file or entry not found" );
44
+
45
+ private static final IllegalStateException NOT_FOUND_CONNECTION_EXCEPTION = new IllegalStateException (
46
+ FILE_NOT_FOUND_EXCEPTION );
41
47
42
48
private static final String SEPARATOR = "!/" ;
43
49
@@ -63,7 +69,8 @@ protected URLConnection openConnection(URL u) throws IOException {
63
69
64
70
private static final String READ_ACTION = "read" ;
65
71
66
- private static ThreadLocal <Boolean > useFastExceptions = new ThreadLocal <Boolean >();
72
+ private static final JarURLConnection NOT_FOUND_CONNECTION = JarURLConnection
73
+ .notFound ();
67
74
68
75
private final JarFile jarFile ;
69
76
@@ -75,48 +82,20 @@ protected URLConnection openConnection(URL u) throws IOException {
75
82
76
83
private JarEntry jarEntry ;
77
84
78
- protected JarURLConnection (URL url , JarFile jarFile ) throws IOException {
85
+ private JarURLConnection (URL url , JarFile jarFile , JarEntryName jarEntryName )
86
+ throws IOException {
79
87
// What we pass to super is ultimately ignored
80
88
super (EMPTY_JAR_URL );
81
89
this .url = url ;
82
- String spec = extractFullSpec (url , jarFile .getPathFromRoot ());
83
- int separator ;
84
- int index = 0 ;
85
- while ((separator = spec .indexOf (SEPARATOR , index )) > 0 ) {
86
- jarFile = getNestedJarFile (jarFile , spec .substring (index , separator ));
87
- index += separator + SEPARATOR .length ();
88
- }
89
90
this .jarFile = jarFile ;
90
- this .jarEntryName = getJarEntryName (spec .substring (index ));
91
- }
92
-
93
- private String extractFullSpec (URL url , String pathFromRoot ) {
94
- String file = url .getFile ();
95
- int separatorIndex = file .indexOf (SEPARATOR );
96
- if (separatorIndex < 0 ) {
97
- return "" ;
98
- }
99
- int specIndex = separatorIndex + SEPARATOR .length () + pathFromRoot .length ();
100
- return file .substring (specIndex );
101
- }
102
-
103
- private JarFile getNestedJarFile (JarFile jarFile , String name ) throws IOException {
104
- JarEntry jarEntry = jarFile .getJarEntry (name );
105
- if (jarEntry == null ) {
106
- throwFileNotFound (jarEntry , jarFile );
107
- }
108
- return jarFile .getNestedJarFile (jarEntry );
109
- }
110
-
111
- private JarEntryName getJarEntryName (String spec ) {
112
- if (spec .length () == 0 ) {
113
- return EMPTY_JAR_ENTRY_NAME ;
114
- }
115
- return new JarEntryName (spec );
91
+ this .jarEntryName = jarEntryName ;
116
92
}
117
93
118
94
@ Override
119
95
public void connect () throws IOException {
96
+ if (this .jarFile == null ) {
97
+ throw FILE_NOT_FOUND_EXCEPTION ;
98
+ }
120
99
if (!this .jarEntryName .isEmpty () && this .jarEntry == null ) {
121
100
this .jarEntry = this .jarFile .getJarEntry (getEntryName ());
122
101
if (this .jarEntry == null ) {
@@ -126,15 +105,6 @@ public void connect() throws IOException {
126
105
this .connected = true ;
127
106
}
128
107
129
- private void throwFileNotFound (Object entry , JarFile jarFile )
130
- throws FileNotFoundException {
131
- if (Boolean .TRUE .equals (useFastExceptions .get ())) {
132
- throw FILE_NOT_FOUND_EXCEPTION ;
133
- }
134
- throw new FileNotFoundException (
135
- "JAR entry " + entry + " not found in " + jarFile .getName ());
136
- }
137
-
138
108
@ Override
139
109
public JarFile getJarFile () throws IOException {
140
110
connect ();
@@ -143,6 +113,9 @@ public JarFile getJarFile() throws IOException {
143
113
144
114
@ Override
145
115
public URL getJarFileURL () {
116
+ if (this .jarFile == null ) {
117
+ throw NOT_FOUND_CONNECTION_EXCEPTION ;
118
+ }
146
119
if (this .jarFileUrl == null ) {
147
120
this .jarFileUrl = buildJarFileUrl ();
148
121
}
@@ -167,7 +140,7 @@ private URL buildJarFileUrl() {
167
140
168
141
@ Override
169
142
public JarEntry getJarEntry () throws IOException {
170
- if (this .jarEntryName .isEmpty ()) {
143
+ if (this .jarEntryName == null || this . jarEntryName .isEmpty ()) {
171
144
return null ;
172
145
}
173
146
connect ();
@@ -176,11 +149,17 @@ public JarEntry getJarEntry() throws IOException {
176
149
177
150
@ Override
178
151
public String getEntryName () {
152
+ if (this .jarFile == null ) {
153
+ throw NOT_FOUND_CONNECTION_EXCEPTION ;
154
+ }
179
155
return this .jarEntryName .toString ();
180
156
}
181
157
182
158
@ Override
183
159
public InputStream getInputStream () throws IOException {
160
+ if (this .jarFile == null ) {
161
+ throw FILE_NOT_FOUND_EXCEPTION ;
162
+ }
184
163
if (this .jarEntryName .isEmpty ()) {
185
164
throw new IOException ("no entry name specified" );
186
165
}
@@ -192,8 +171,20 @@ public InputStream getInputStream() throws IOException {
192
171
return inputStream ;
193
172
}
194
173
174
+ private void throwFileNotFound (Object entry , JarFile jarFile )
175
+ throws FileNotFoundException {
176
+ if (Boolean .TRUE .equals (useFastExceptions .get ())) {
177
+ throw FILE_NOT_FOUND_EXCEPTION ;
178
+ }
179
+ throw new FileNotFoundException (
180
+ "JAR entry " + entry + " not found in " + jarFile .getName ());
181
+ }
182
+
195
183
@ Override
196
184
public int getContentLength () {
185
+ if (this .jarFile == null ) {
186
+ return -1 ;
187
+ }
197
188
try {
198
189
if (this .jarEntryName .isEmpty ()) {
199
190
return this .jarFile .size ();
@@ -214,11 +205,14 @@ public Object getContent() throws IOException {
214
205
215
206
@ Override
216
207
public String getContentType () {
217
- return this .jarEntryName . getContentType ();
208
+ return ( this .jarEntryName == null ? null : this . jarEntryName . getContentType () );
218
209
}
219
210
220
211
@ Override
221
212
public Permission getPermission () throws IOException {
213
+ if (this .jarFile == null ) {
214
+ throw FILE_NOT_FOUND_EXCEPTION ;
215
+ }
222
216
if (this .permission == null ) {
223
217
this .permission = new FilePermission (
224
218
this .jarFile .getRootJarFile ().getFile ().getPath (), READ_ACTION );
@@ -230,6 +224,56 @@ static void setUseFastExceptions(boolean useFastExceptions) {
230
224
JarURLConnection .useFastExceptions .set (useFastExceptions );
231
225
}
232
226
227
+ static JarURLConnection get (URL url , JarFile jarFile ) throws IOException {
228
+ String spec = extractFullSpec (url , jarFile .getPathFromRoot ());
229
+ int separator ;
230
+ int index = 0 ;
231
+ while ((separator = spec .indexOf (SEPARATOR , index )) > 0 ) {
232
+ String entryName = spec .substring (index , separator );
233
+ JarEntry jarEntry = jarFile .getJarEntry (entryName );
234
+ if (jarEntry == null ) {
235
+ return JarURLConnection .notFound (jarFile , JarEntryName .get (entryName ));
236
+ }
237
+ jarFile = jarFile .getNestedJarFile (jarEntry );
238
+ index += separator + SEPARATOR .length ();
239
+ }
240
+ JarEntryName jarEntryName = JarEntryName .get (spec , index );
241
+ if (Boolean .TRUE .equals (useFastExceptions .get ())) {
242
+ if (!jarEntryName .isEmpty ()
243
+ && !jarFile .containsEntry (jarEntryName .toString ())) {
244
+ return NOT_FOUND_CONNECTION ;
245
+ }
246
+ }
247
+ return new JarURLConnection (url , jarFile , jarEntryName );
248
+ }
249
+
250
+ private static String extractFullSpec (URL url , String pathFromRoot ) {
251
+ String file = url .getFile ();
252
+ int separatorIndex = file .indexOf (SEPARATOR );
253
+ if (separatorIndex < 0 ) {
254
+ return "" ;
255
+ }
256
+ int specIndex = separatorIndex + SEPARATOR .length () + pathFromRoot .length ();
257
+ return file .substring (specIndex );
258
+ }
259
+
260
+ private static JarURLConnection notFound () {
261
+ try {
262
+ return notFound (null , null );
263
+ }
264
+ catch (IOException ex ) {
265
+ throw new IllegalStateException (ex );
266
+ }
267
+ }
268
+
269
+ private static JarURLConnection notFound (JarFile jarFile , JarEntryName jarEntryName )
270
+ throws IOException {
271
+ if (Boolean .TRUE .equals (useFastExceptions .get ())) {
272
+ return NOT_FOUND_CONNECTION ;
273
+ }
274
+ return new JarURLConnection (null , jarFile , jarEntryName );
275
+ }
276
+
233
277
/**
234
278
* A JarEntryName parsed from a URL String.
235
279
*/
@@ -316,6 +360,17 @@ private String deduceContentType() {
316
360
return type ;
317
361
}
318
362
363
+ public static JarEntryName get (String spec ) {
364
+ return get (spec , 0 );
365
+ }
366
+
367
+ public static JarEntryName get (String spec , int beginIndex ) {
368
+ if (spec .length () <= beginIndex ) {
369
+ return EMPTY_JAR_ENTRY_NAME ;
370
+ }
371
+ return new JarEntryName (spec .substring (beginIndex ));
372
+ }
373
+
319
374
}
320
375
321
376
}
0 commit comments