1 | |
package org.apache.maven.doxia.parser; |
2 | |
|
3 | |
|
4 | |
|
5 | |
|
6 | |
|
7 | |
|
8 | |
|
9 | |
|
10 | |
|
11 | |
|
12 | |
|
13 | |
|
14 | |
|
15 | |
|
16 | |
|
17 | |
|
18 | |
|
19 | |
|
20 | |
|
21 | |
|
22 | |
import java.io.BufferedReader; |
23 | |
import java.io.ByteArrayInputStream; |
24 | |
import java.io.File; |
25 | |
import java.io.FileOutputStream; |
26 | |
import java.io.IOException; |
27 | |
import java.io.InputStream; |
28 | |
import java.io.OutputStream; |
29 | |
import java.io.Reader; |
30 | |
import java.io.StringReader; |
31 | |
import java.net.URL; |
32 | |
import java.util.Hashtable; |
33 | |
import java.util.LinkedHashMap; |
34 | |
import java.util.Locale; |
35 | |
import java.util.Map; |
36 | |
import java.util.regex.Matcher; |
37 | |
import java.util.regex.Pattern; |
38 | |
|
39 | |
import org.apache.http.HttpEntity; |
40 | |
import org.apache.http.HttpResponse; |
41 | |
import org.apache.http.HttpStatus; |
42 | |
import org.apache.http.client.ClientProtocolException; |
43 | |
import org.apache.http.client.HttpRequestRetryHandler; |
44 | |
import org.apache.http.client.methods.HttpGet; |
45 | |
import org.apache.http.impl.client.DefaultHttpClient; |
46 | |
import org.apache.http.impl.client.DefaultHttpRequestRetryHandler; |
47 | |
import org.apache.http.util.EntityUtils; |
48 | |
|
49 | |
import org.apache.maven.doxia.macro.MacroExecutionException; |
50 | |
import org.apache.maven.doxia.markup.XmlMarkup; |
51 | |
import org.apache.maven.doxia.sink.Sink; |
52 | |
import org.apache.maven.doxia.sink.SinkEventAttributeSet; |
53 | |
import org.apache.maven.doxia.util.HtmlTools; |
54 | |
import org.apache.maven.doxia.util.XmlValidator; |
55 | |
|
56 | |
import org.codehaus.plexus.util.FileUtils; |
57 | |
import org.codehaus.plexus.util.IOUtil; |
58 | |
import org.codehaus.plexus.util.StringUtils; |
59 | |
import org.codehaus.plexus.util.xml.pull.MXParser; |
60 | |
import org.codehaus.plexus.util.xml.pull.XmlPullParser; |
61 | |
import org.codehaus.plexus.util.xml.pull.XmlPullParserException; |
62 | |
|
63 | |
import org.xml.sax.EntityResolver; |
64 | |
import org.xml.sax.InputSource; |
65 | |
import org.xml.sax.SAXException; |
66 | |
|
67 | |
|
68 | |
|
69 | |
|
70 | |
|
71 | |
|
72 | |
|
73 | |
|
74 | 44 | public abstract class AbstractXmlParser |
75 | |
extends AbstractParser |
76 | |
implements XmlMarkup |
77 | |
{ |
78 | |
|
79 | |
|
80 | |
|
81 | |
|
82 | |
|
83 | |
|
84 | 2 | private static final Pattern PATTERN_ENTITY_1 = |
85 | |
Pattern.compile( ENTITY_START + "(\\s)+([^>|^\\s]+)(\\s)+\"(\\s)*(&[a-zA-Z]{2,6};)(\\s)*\"(\\s)*>" ); |
86 | |
|
87 | |
|
88 | |
|
89 | |
|
90 | |
|
91 | |
|
92 | |
|
93 | 2 | private static final Pattern PATTERN_ENTITY_2 = |
94 | |
Pattern.compile( ENTITY_START + "(\\s)+([^>|^\\s]+)(\\s)+\"(\\s)*(&(#x?[0-9a-fA-F]{1,5};)*)(\\s)*\"(\\s)*>" ); |
95 | |
|
96 | |
private boolean ignorableWhitespace; |
97 | |
|
98 | |
private boolean collapsibleWhitespace; |
99 | |
|
100 | |
private boolean trimmableWhitespace; |
101 | |
|
102 | |
private Map<String, String> entities; |
103 | |
|
104 | 44 | private boolean validate = false; |
105 | |
|
106 | |
|
107 | |
public void parse( Reader source, Sink sink ) |
108 | |
throws ParseException |
109 | |
{ |
110 | 52 | init(); |
111 | |
|
112 | 52 | Reader src = source; |
113 | |
|
114 | |
|
115 | 52 | if ( isValidate() ) |
116 | |
{ |
117 | |
String content; |
118 | |
try |
119 | |
{ |
120 | 0 | content = IOUtil.toString( new BufferedReader( src ) ); |
121 | |
} |
122 | 0 | catch ( IOException e ) |
123 | |
{ |
124 | 0 | throw new ParseException( "Error reading the model: " + e.getMessage(), e ); |
125 | 0 | } |
126 | |
|
127 | 0 | new XmlValidator( getLog() ).validate( content ); |
128 | |
|
129 | 0 | src = new StringReader( content ); |
130 | |
} |
131 | |
|
132 | |
|
133 | |
try |
134 | |
{ |
135 | 52 | XmlPullParser parser = new MXParser(); |
136 | |
|
137 | 52 | parser.setInput( src ); |
138 | |
|
139 | 52 | sink.enableLogging( getLog() ); |
140 | |
|
141 | 52 | parseXml( parser, sink ); |
142 | |
} |
143 | 0 | catch ( XmlPullParserException ex ) |
144 | |
{ |
145 | 0 | throw new ParseException( "Error parsing the model: " + ex.getMessage(), ex, ex.getLineNumber(), |
146 | |
ex.getColumnNumber() ); |
147 | |
} |
148 | 0 | catch ( MacroExecutionException ex ) |
149 | |
{ |
150 | 0 | throw new ParseException( "Macro execution failed: " + ex.getMessage(), ex ); |
151 | 52 | } |
152 | |
|
153 | 52 | setSecondParsing( false ); |
154 | 52 | init(); |
155 | 52 | } |
156 | |
|
157 | |
|
158 | |
|
159 | |
|
160 | |
|
161 | |
|
162 | |
@Override |
163 | |
public void parse( String string, Sink sink ) |
164 | |
throws ParseException |
165 | |
{ |
166 | 46 | super.parse( string, sink ); |
167 | 46 | } |
168 | |
|
169 | |
|
170 | |
@Override |
171 | |
public final int getType() |
172 | |
{ |
173 | 0 | return XML_TYPE; |
174 | |
} |
175 | |
|
176 | |
|
177 | |
|
178 | |
|
179 | |
|
180 | |
|
181 | |
|
182 | |
|
183 | |
protected SinkEventAttributeSet getAttributesFromParser( XmlPullParser parser ) |
184 | |
{ |
185 | 218 | int count = parser.getAttributeCount(); |
186 | |
|
187 | 218 | if ( count < 0 ) |
188 | |
{ |
189 | 2 | return null; |
190 | |
} |
191 | |
|
192 | 216 | SinkEventAttributeSet atts = new SinkEventAttributeSet( count ); |
193 | |
|
194 | 260 | for ( int i = 0; i < count; i++ ) |
195 | |
{ |
196 | 44 | atts.addAttribute( parser.getAttributeName( i ), parser.getAttributeValue( i ) ); |
197 | |
} |
198 | |
|
199 | 216 | return atts; |
200 | |
} |
201 | |
|
202 | |
|
203 | |
|
204 | |
|
205 | |
|
206 | |
|
207 | |
|
208 | |
|
209 | |
|
210 | |
private void parseXml( XmlPullParser parser, Sink sink ) |
211 | |
throws XmlPullParserException, MacroExecutionException |
212 | |
{ |
213 | 52 | int eventType = parser.getEventType(); |
214 | |
|
215 | 658 | while ( eventType != XmlPullParser.END_DOCUMENT ) |
216 | |
{ |
217 | 606 | if ( eventType == XmlPullParser.START_TAG ) |
218 | |
{ |
219 | 214 | handleStartTag( parser, sink ); |
220 | |
} |
221 | 392 | else if ( eventType == XmlPullParser.END_TAG ) |
222 | |
{ |
223 | 214 | handleEndTag( parser, sink ); |
224 | |
} |
225 | 178 | else if ( eventType == XmlPullParser.TEXT ) |
226 | |
{ |
227 | 78 | String text = getText( parser ); |
228 | |
|
229 | 78 | if ( isIgnorableWhitespace() ) |
230 | |
{ |
231 | 0 | if ( text.trim().length() != 0 ) |
232 | |
{ |
233 | 0 | handleText( parser, sink ); |
234 | |
} |
235 | |
} |
236 | |
else |
237 | |
{ |
238 | 78 | handleText( parser, sink ); |
239 | |
} |
240 | 78 | } |
241 | 100 | else if ( eventType == XmlPullParser.CDSECT ) |
242 | |
{ |
243 | 4 | handleCdsect( parser, sink ); |
244 | |
} |
245 | 96 | else if ( eventType == XmlPullParser.COMMENT ) |
246 | |
{ |
247 | 4 | handleComment( parser, sink ); |
248 | |
} |
249 | 92 | else if ( eventType == XmlPullParser.ENTITY_REF ) |
250 | |
{ |
251 | 34 | handleEntity( parser, sink ); |
252 | |
} |
253 | 58 | else if ( eventType == XmlPullParser.IGNORABLE_WHITESPACE ) |
254 | |
{ |
255 | |
|
256 | |
} |
257 | 56 | else if ( eventType == XmlPullParser.PROCESSING_INSTRUCTION ) |
258 | |
{ |
259 | |
|
260 | |
} |
261 | 56 | else if ( eventType == XmlPullParser.DOCDECL ) |
262 | |
{ |
263 | 4 | addLocalEntities( parser, parser.getText() ); |
264 | |
|
265 | 4 | for ( byte[] res : CachedFileEntityResolver.ENTITY_CACHE.values() ) |
266 | |
{ |
267 | 0 | addDTDEntities( parser, new String( res ) ); |
268 | |
} |
269 | |
} |
270 | |
|
271 | |
try |
272 | |
{ |
273 | 606 | eventType = parser.nextToken(); |
274 | |
} |
275 | 0 | catch ( IOException io ) |
276 | |
{ |
277 | 0 | throw new XmlPullParserException( "IOException: " + io.getMessage(), parser, io ); |
278 | 606 | } |
279 | |
} |
280 | 52 | } |
281 | |
|
282 | |
|
283 | |
|
284 | |
|
285 | |
|
286 | |
|
287 | |
|
288 | |
|
289 | |
|
290 | |
protected abstract void handleStartTag( XmlPullParser parser, Sink sink ) |
291 | |
throws XmlPullParserException, MacroExecutionException; |
292 | |
|
293 | |
|
294 | |
|
295 | |
|
296 | |
|
297 | |
|
298 | |
|
299 | |
|
300 | |
|
301 | |
protected abstract void handleEndTag( XmlPullParser parser, Sink sink ) |
302 | |
throws XmlPullParserException, MacroExecutionException; |
303 | |
|
304 | |
|
305 | |
|
306 | |
|
307 | |
|
308 | |
|
309 | |
|
310 | |
|
311 | |
|
312 | |
|
313 | |
|
314 | |
protected void handleText( XmlPullParser parser, Sink sink ) |
315 | |
throws XmlPullParserException |
316 | |
{ |
317 | 0 | String text = getText( parser ); |
318 | |
|
319 | |
|
320 | |
|
321 | |
|
322 | |
|
323 | 0 | if ( StringUtils.isNotEmpty( text ) ) |
324 | |
{ |
325 | 0 | sink.text( text ); |
326 | |
} |
327 | 0 | } |
328 | |
|
329 | |
|
330 | |
|
331 | |
|
332 | |
|
333 | |
|
334 | |
|
335 | |
|
336 | |
|
337 | |
|
338 | |
|
339 | |
protected void handleCdsect( XmlPullParser parser, Sink sink ) |
340 | |
throws XmlPullParserException |
341 | |
{ |
342 | 0 | sink.text( getText( parser ) ); |
343 | 0 | } |
344 | |
|
345 | |
|
346 | |
|
347 | |
|
348 | |
|
349 | |
|
350 | |
|
351 | |
|
352 | |
|
353 | |
|
354 | |
|
355 | |
protected void handleComment( XmlPullParser parser, Sink sink ) |
356 | |
throws XmlPullParserException |
357 | |
{ |
358 | 0 | sink.comment( getText( parser ).trim() ); |
359 | 0 | } |
360 | |
|
361 | |
|
362 | |
|
363 | |
|
364 | |
|
365 | |
|
366 | |
|
367 | |
|
368 | |
|
369 | |
|
370 | |
|
371 | |
|
372 | |
|
373 | |
|
374 | |
|
375 | |
protected void handleEntity( XmlPullParser parser, Sink sink ) |
376 | |
throws XmlPullParserException |
377 | |
{ |
378 | 34 | String text = getText( parser ); |
379 | |
|
380 | 34 | String name = parser.getName(); |
381 | |
|
382 | 34 | if ( "#160".equals( name ) || "nbsp".equals( name ) || "#x00A0".equals( name ) ) |
383 | |
{ |
384 | 4 | sink.nonBreakingSpace(); |
385 | |
} |
386 | |
else |
387 | |
{ |
388 | 30 | String unescaped = HtmlTools.unescapeHTML( text ); |
389 | |
|
390 | 30 | sink.text( unescaped ); |
391 | |
} |
392 | 34 | } |
393 | |
|
394 | |
|
395 | |
|
396 | |
|
397 | |
|
398 | |
|
399 | |
|
400 | |
|
401 | |
|
402 | |
|
403 | |
|
404 | |
|
405 | |
|
406 | |
|
407 | |
|
408 | |
protected void handleUnknown( XmlPullParser parser, Sink sink, int type ) |
409 | |
{ |
410 | 4 | Object[] required = new Object[] { new Integer( type ) }; |
411 | |
|
412 | 4 | SinkEventAttributeSet attribs = getAttributesFromParser( parser ); |
413 | |
|
414 | 4 | sink.unknown( parser.getName(), required, attribs ); |
415 | 4 | } |
416 | |
|
417 | |
|
418 | |
|
419 | |
|
420 | |
|
421 | |
|
422 | |
|
423 | |
|
424 | |
protected boolean isIgnorableWhitespace() |
425 | |
{ |
426 | 78 | return ignorableWhitespace; |
427 | |
} |
428 | |
|
429 | |
|
430 | |
|
431 | |
|
432 | |
|
433 | |
|
434 | |
|
435 | |
|
436 | |
|
437 | |
|
438 | |
protected void setIgnorableWhitespace( boolean ignorable ) |
439 | |
{ |
440 | 0 | this.ignorableWhitespace = ignorable; |
441 | 0 | } |
442 | |
|
443 | |
|
444 | |
|
445 | |
|
446 | |
|
447 | |
|
448 | |
|
449 | |
|
450 | |
protected boolean isCollapsibleWhitespace() |
451 | |
{ |
452 | 198 | return collapsibleWhitespace; |
453 | |
} |
454 | |
|
455 | |
|
456 | |
|
457 | |
|
458 | |
|
459 | |
|
460 | |
|
461 | |
|
462 | |
|
463 | |
|
464 | |
protected void setCollapsibleWhitespace( boolean collapsible ) |
465 | |
{ |
466 | 0 | this.collapsibleWhitespace = collapsible; |
467 | 0 | } |
468 | |
|
469 | |
|
470 | |
|
471 | |
|
472 | |
|
473 | |
|
474 | |
|
475 | |
|
476 | |
protected boolean isTrimmableWhitespace() |
477 | |
{ |
478 | 198 | return trimmableWhitespace; |
479 | |
} |
480 | |
|
481 | |
|
482 | |
|
483 | |
|
484 | |
|
485 | |
|
486 | |
|
487 | |
|
488 | |
|
489 | |
|
490 | |
protected void setTrimmableWhitespace( boolean trimmable ) |
491 | |
{ |
492 | 0 | this.trimmableWhitespace = trimmable; |
493 | 0 | } |
494 | |
|
495 | |
|
496 | |
|
497 | |
|
498 | |
|
499 | |
|
500 | |
|
501 | |
|
502 | |
|
503 | |
|
504 | |
|
505 | |
protected String getText( XmlPullParser parser ) |
506 | |
{ |
507 | 198 | String text = parser.getText(); |
508 | |
|
509 | 198 | if ( isTrimmableWhitespace() ) |
510 | |
{ |
511 | 0 | text = text.trim(); |
512 | |
} |
513 | |
|
514 | 198 | if ( isCollapsibleWhitespace() ) |
515 | |
{ |
516 | 0 | StringBuilder newText = new StringBuilder(); |
517 | 0 | String[] elts = StringUtils.split( text, " \r\n" ); |
518 | 0 | for ( int i = 0; i < elts.length; i++ ) |
519 | |
{ |
520 | 0 | newText.append( elts[i] ); |
521 | 0 | if ( ( i + 1 ) < elts.length ) |
522 | |
{ |
523 | 0 | newText.append( " " ); |
524 | |
} |
525 | |
} |
526 | 0 | text = newText.toString(); |
527 | |
} |
528 | |
|
529 | 198 | return text; |
530 | |
} |
531 | |
|
532 | |
|
533 | |
|
534 | |
|
535 | |
|
536 | |
|
537 | |
|
538 | |
|
539 | |
|
540 | |
|
541 | |
|
542 | |
|
543 | |
|
544 | |
protected Map<String, String> getLocalEntities() |
545 | |
{ |
546 | 16 | if ( entities == null ) |
547 | |
{ |
548 | 4 | entities = new LinkedHashMap<String, String>(); |
549 | |
} |
550 | |
|
551 | 16 | return entities; |
552 | |
} |
553 | |
|
554 | |
|
555 | |
|
556 | |
|
557 | |
|
558 | |
|
559 | |
|
560 | |
public boolean isValidate() |
561 | |
{ |
562 | 52 | return validate; |
563 | |
} |
564 | |
|
565 | |
|
566 | |
|
567 | |
|
568 | |
|
569 | |
|
570 | |
|
571 | |
|
572 | |
public void setValidate( boolean validate ) |
573 | |
{ |
574 | 4 | this.validate = validate; |
575 | 4 | } |
576 | |
|
577 | |
|
578 | |
|
579 | |
|
580 | |
|
581 | |
|
582 | |
|
583 | |
|
584 | |
|
585 | |
|
586 | |
|
587 | |
|
588 | |
|
589 | |
|
590 | |
|
591 | |
|
592 | |
private void addEntity( XmlPullParser parser, String entityName, String entityValue ) |
593 | |
throws XmlPullParserException |
594 | |
{ |
595 | 16 | if ( entityName.endsWith( "amp" ) || entityName.endsWith( "lt" ) || entityName.endsWith( "gt" ) |
596 | |
|| entityName.endsWith( "quot" ) || entityName.endsWith( "apos" ) ) |
597 | |
{ |
598 | 0 | return; |
599 | |
} |
600 | |
|
601 | 16 | parser.defineEntityReplacementText( entityName, entityValue ); |
602 | 16 | getLocalEntities().put( entityName, entityValue ); |
603 | 16 | } |
604 | |
|
605 | |
|
606 | |
|
607 | |
|
608 | |
|
609 | |
|
610 | |
|
611 | |
|
612 | |
|
613 | |
|
614 | |
|
615 | |
|
616 | |
|
617 | |
|
618 | |
private void addLocalEntities( XmlPullParser parser, String text ) |
619 | |
throws XmlPullParserException |
620 | |
{ |
621 | 4 | int entitiesCount = StringUtils.countMatches( text, ENTITY_START ); |
622 | 4 | if ( entitiesCount > 0 ) |
623 | |
{ |
624 | |
|
625 | 4 | int start = text.indexOf( '['); |
626 | 4 | int end = text.lastIndexOf( ']'); |
627 | 4 | if ( start != -1 && end != -1 ) |
628 | |
{ |
629 | 4 | addDTDEntities( parser, text.substring( start + 1, end ) ); |
630 | |
} |
631 | |
} |
632 | 4 | } |
633 | |
|
634 | |
|
635 | |
|
636 | |
|
637 | |
|
638 | |
|
639 | |
|
640 | |
|
641 | |
|
642 | |
|
643 | |
|
644 | |
|
645 | |
|
646 | |
|
647 | |
|
648 | |
|
649 | |
private void addDTDEntities( XmlPullParser parser, String text ) |
650 | |
throws XmlPullParserException |
651 | |
{ |
652 | 4 | int entitiesCount = StringUtils.countMatches( text, ENTITY_START ); |
653 | 4 | if ( entitiesCount > 0 ) |
654 | |
{ |
655 | 4 | final String txt = StringUtils.replace( text, ENTITY_START, "\n" + ENTITY_START ); |
656 | 4 | BufferedReader reader = new BufferedReader( new StringReader( txt ) ); |
657 | |
String line; |
658 | 4 | String tmpLine = ""; |
659 | |
try |
660 | |
{ |
661 | |
Matcher matcher; |
662 | 32 | while ( ( line = reader.readLine() ) != null ) |
663 | |
{ |
664 | 28 | tmpLine += "\n" + line; |
665 | 28 | matcher = PATTERN_ENTITY_1.matcher( tmpLine ); |
666 | 28 | if ( matcher.find() && matcher.groupCount() == 7 ) |
667 | |
{ |
668 | 4 | String entityName = matcher.group( 2 ); |
669 | 4 | String entityValue = matcher.group( 5 ); |
670 | |
|
671 | 4 | addEntity( parser, entityName, entityValue ); |
672 | 4 | tmpLine = ""; |
673 | 4 | } |
674 | |
else |
675 | |
{ |
676 | 24 | matcher = PATTERN_ENTITY_2.matcher( tmpLine ); |
677 | 24 | if ( matcher.find() && matcher.groupCount() == 8 ) |
678 | |
{ |
679 | 12 | String entityName = matcher.group( 2 ); |
680 | 12 | String entityValue = matcher.group( 5 ); |
681 | |
|
682 | 12 | addEntity( parser, entityName, entityValue ); |
683 | 12 | tmpLine = ""; |
684 | 12 | } |
685 | |
} |
686 | |
} |
687 | |
} |
688 | 0 | catch ( IOException e ) |
689 | |
{ |
690 | |
|
691 | |
} |
692 | |
finally |
693 | |
{ |
694 | 4 | IOUtil.close( reader ); |
695 | 4 | } |
696 | |
} |
697 | 4 | } |
698 | |
|
699 | |
|
700 | |
|
701 | |
|
702 | |
|
703 | 0 | public static class CachedFileEntityResolver |
704 | |
implements EntityResolver |
705 | |
{ |
706 | |
|
707 | 2 | protected static final Map<String, byte[]> ENTITY_CACHE = new Hashtable<String, byte[]>(); |
708 | |
|
709 | |
|
710 | |
public InputSource resolveEntity( String publicId, String systemId ) |
711 | |
throws SAXException, IOException |
712 | |
{ |
713 | 0 | byte[] res = ENTITY_CACHE.get( systemId ); |
714 | |
|
715 | 0 | if ( res == null ) |
716 | |
{ |
717 | 0 | String systemName = FileUtils.getFile( systemId ).getName(); |
718 | 0 | File temp = new File( System.getProperty( "java.io.tmpdir" ), systemName ); |
719 | |
|
720 | 0 | if ( !temp.exists() ) |
721 | |
{ |
722 | |
|
723 | 0 | if ( systemId.toLowerCase( Locale.ENGLISH ).startsWith( "file" ) ) |
724 | |
{ |
725 | |
|
726 | |
|
727 | 0 | String resource = "/" + systemName; |
728 | 0 | URL url = getClass().getResource( resource ); |
729 | 0 | if ( url != null ) |
730 | |
{ |
731 | 0 | res = toByteArray( url ); |
732 | |
} |
733 | |
else |
734 | |
{ |
735 | 0 | throw new SAXException( "Could not find the SYSTEM entity: " + systemId |
736 | |
+ " because '" + resource + "' is not available of the classpath." ); |
737 | |
} |
738 | 0 | } |
739 | |
else |
740 | |
{ |
741 | 0 | res = toByteArray( new URL( systemId ) ); |
742 | |
} |
743 | |
|
744 | |
|
745 | 0 | copy( res, temp ); |
746 | |
} |
747 | |
else |
748 | |
{ |
749 | |
|
750 | 0 | res = toByteArray( temp.toURI().toURL() ); |
751 | |
} |
752 | |
|
753 | 0 | ENTITY_CACHE.put( systemId, res ); |
754 | |
} |
755 | |
|
756 | 0 | InputSource is = new InputSource( new ByteArrayInputStream( res ) ); |
757 | 0 | is.setPublicId( publicId ); |
758 | 0 | is.setSystemId( systemId ); |
759 | |
|
760 | 0 | return is; |
761 | |
} |
762 | |
|
763 | |
|
764 | |
|
765 | |
|
766 | |
|
767 | |
|
768 | |
|
769 | |
|
770 | |
|
771 | |
|
772 | |
|
773 | |
private static byte[] toByteArray( URL url ) |
774 | |
throws SAXException |
775 | |
{ |
776 | 0 | if ( !( url.getProtocol().equalsIgnoreCase( "http" ) || url.getProtocol().equalsIgnoreCase( "https" ) ) ) |
777 | |
{ |
778 | 0 | InputStream is = null; |
779 | |
try |
780 | |
{ |
781 | 0 | is = url.openStream(); |
782 | 0 | if ( is == null ) |
783 | |
{ |
784 | 0 | throw new SAXException( "Cannot open stream from the url: " + url.toString() ); |
785 | |
} |
786 | 0 | return IOUtil.toByteArray( is ); |
787 | |
} |
788 | 0 | catch ( IOException e ) |
789 | |
{ |
790 | 0 | throw new SAXException( "IOException: " + e.getMessage(), e ); |
791 | |
} |
792 | |
finally |
793 | |
{ |
794 | 0 | IOUtil.close( is ); |
795 | |
} |
796 | |
} |
797 | |
|
798 | |
|
799 | 0 | DefaultHttpClient client = new DefaultHttpClient(); |
800 | 0 | HttpGet method = new HttpGet( url.toString() ); |
801 | |
|
802 | |
|
803 | 0 | method.setHeader( "user-agent", "Apache-Doxia/" + doxiaVersion() ); |
804 | |
|
805 | 0 | HttpRequestRetryHandler retryHandler = new DefaultHttpRequestRetryHandler( 3, false ); |
806 | 0 | client.setHttpRequestRetryHandler( retryHandler ); |
807 | |
|
808 | 0 | HttpEntity entity = null; |
809 | |
try |
810 | |
{ |
811 | 0 | HttpResponse response = client.execute( method ); |
812 | 0 | int statusCode = response.getStatusLine().getStatusCode(); |
813 | 0 | if ( statusCode != HttpStatus.SC_OK ) |
814 | |
{ |
815 | 0 | throw new IOException( "The status code when accessing the URL '" + url.toString() + "' was " |
816 | |
+ statusCode + ", which is not allowed. The server gave this reason for the failure '" |
817 | |
+ response.getStatusLine().getReasonPhrase() + "'." ); |
818 | |
} |
819 | |
|
820 | 0 | entity = response.getEntity(); |
821 | 0 | return EntityUtils.toByteArray( entity ); |
822 | |
} |
823 | 0 | catch ( ClientProtocolException e ) |
824 | |
{ |
825 | 0 | throw new SAXException( "ClientProtocolException: Fatal protocol violation: " + e.getMessage(), e ); |
826 | |
} |
827 | 0 | catch ( IOException e ) |
828 | |
{ |
829 | 0 | throw new SAXException( "IOException: Fatal transport error: " + e.getMessage(), e ); |
830 | |
} |
831 | |
finally |
832 | |
{ |
833 | 0 | if ( entity != null ) |
834 | |
{ |
835 | |
try |
836 | |
{ |
837 | 0 | entity.consumeContent(); |
838 | |
} |
839 | 0 | catch ( IOException e ) |
840 | |
{ |
841 | |
|
842 | 0 | } |
843 | |
} |
844 | |
} |
845 | |
} |
846 | |
|
847 | |
|
848 | |
|
849 | |
|
850 | |
|
851 | |
|
852 | |
|
853 | |
|
854 | |
|
855 | |
private void copy( byte[] res, File f ) |
856 | |
throws SAXException |
857 | |
{ |
858 | 0 | if ( f.isDirectory() ) |
859 | |
{ |
860 | 0 | throw new SAXException( "'" + f.getAbsolutePath() + "' is a directory, can not write it." ); |
861 | |
} |
862 | |
|
863 | 0 | OutputStream os = null; |
864 | |
try |
865 | |
{ |
866 | 0 | os = new FileOutputStream( f ); |
867 | 0 | IOUtil.copy( res, os ); |
868 | |
} |
869 | 0 | catch ( IOException e ) |
870 | |
{ |
871 | 0 | throw new SAXException( "IOException: " + e.getMessage(), e ); |
872 | |
} |
873 | |
finally |
874 | |
{ |
875 | 0 | IOUtil.close( os ); |
876 | 0 | } |
877 | 0 | } |
878 | |
} |
879 | |
} |