1
1
/*
2
- * Copyright 2002-2022 the original author or authors.
2
+ * Copyright 2002-2024 the original author or authors.
3
3
*
4
4
* Licensed under the Apache License, Version 2.0 (the "License");
5
5
* you may not use this file except in compliance with the License.
37
37
import java .util .Date ;
38
38
import java .util .Map ;
39
39
import java .util .UUID ;
40
+ import java .util .concurrent .locks .Lock ;
41
+ import java .util .concurrent .locks .ReentrantLock ;
40
42
41
43
import javax .xml .XMLConstants ;
42
44
import javax .xml .datatype .Duration ;
@@ -192,7 +194,7 @@ public class Jaxb2Marshaller implements MimeMarshaller, MimeUnmarshaller, Generi
192
194
@ Nullable
193
195
private ClassLoader beanClassLoader ;
194
196
195
- private final Object jaxbContextMonitor = new Object ();
197
+ private final Lock jaxbContextLock = new ReentrantLock ();
196
198
197
199
@ Nullable
198
200
private volatile JAXBContext jaxbContext ;
@@ -204,6 +206,12 @@ public class Jaxb2Marshaller implements MimeMarshaller, MimeUnmarshaller, Generi
204
206
205
207
private boolean processExternalEntities = false ;
206
208
209
+ @ Nullable
210
+ private volatile SAXParserFactory schemaParserFactory ;
211
+
212
+ @ Nullable
213
+ private volatile SAXParserFactory sourceParserFactory ;
214
+
207
215
208
216
/**
209
217
* Set multiple JAXB context paths. The given array of context paths gets
@@ -426,6 +434,7 @@ public void setMappedClass(Class<?> mappedClass) {
426
434
*/
427
435
public void setSupportDtd (boolean supportDtd ) {
428
436
this .supportDtd = supportDtd ;
437
+ this .sourceParserFactory = null ;
429
438
}
430
439
431
440
/**
@@ -450,6 +459,7 @@ public void setProcessExternalEntities(boolean processExternalEntities) {
450
459
if (processExternalEntities ) {
451
460
this .supportDtd = true ;
452
461
}
462
+ this .sourceParserFactory = null ;
453
463
}
454
464
455
465
/**
@@ -497,7 +507,9 @@ public JAXBContext getJaxbContext() {
497
507
if (context != null ) {
498
508
return context ;
499
509
}
500
- synchronized (this .jaxbContextMonitor ) {
510
+
511
+ this .jaxbContextLock .lock ();
512
+ try {
501
513
context = this .jaxbContext ;
502
514
if (context == null ) {
503
515
try {
@@ -521,6 +533,9 @@ else if (!ObjectUtils.isEmpty(this.packagesToScan)) {
521
533
}
522
534
return context ;
523
535
}
536
+ finally {
537
+ this .jaxbContextLock .unlock ();
538
+ }
524
539
}
525
540
526
541
private JAXBContext createJaxbContextFromContextPath (String contextPath ) throws JAXBException {
@@ -587,17 +602,24 @@ private Schema loadSchema(Resource[] resources, String schemaLanguage) throws IO
587
602
Assert .notEmpty (resources , "No resources given" );
588
603
Assert .hasLength (schemaLanguage , "No schema language provided" );
589
604
Source [] schemaSources = new Source [resources .length ];
590
- SAXParserFactory saxParserFactory = SAXParserFactory .newInstance ();
591
- saxParserFactory .setNamespaceAware (true );
592
- saxParserFactory .setFeature ("http://xml.org/sax/features/namespace-prefixes" , true );
605
+
606
+ SAXParserFactory saxParserFactory = this .schemaParserFactory ;
607
+ if (saxParserFactory == null ) {
608
+ saxParserFactory = SAXParserFactory .newInstance ();
609
+ saxParserFactory .setNamespaceAware (true );
610
+ saxParserFactory .setFeature ("http://xml.org/sax/features/namespace-prefixes" , true );
611
+ this .schemaParserFactory = saxParserFactory ;
612
+ }
593
613
SAXParser saxParser = saxParserFactory .newSAXParser ();
594
614
XMLReader xmlReader = saxParser .getXMLReader ();
615
+
595
616
for (int i = 0 ; i < resources .length ; i ++) {
596
617
Resource resource = resources [i ];
597
618
Assert .isTrue (resource != null && resource .exists (), () -> "Resource does not exist: " + resource );
598
619
InputSource inputSource = SaxResourceUtils .createInputSource (resource );
599
620
schemaSources [i ] = new SAXSource (xmlReader , inputSource );
600
621
}
622
+
601
623
SchemaFactory schemaFactory = SchemaFactory .newInstance (schemaLanguage );
602
624
if (this .schemaResourceResolver != null ) {
603
625
schemaFactory .setResourceResolver (this .schemaResourceResolver );
@@ -886,11 +908,16 @@ else if (streamSource.getReader() != null) {
886
908
887
909
try {
888
910
if (xmlReader == null ) {
889
- SAXParserFactory saxParserFactory = SAXParserFactory .newInstance ();
890
- saxParserFactory .setNamespaceAware (true );
891
- saxParserFactory .setFeature ("http://apache.org/xml/features/disallow-doctype-decl" , !isSupportDtd ());
892
- String name = "http://xml.org/sax/features/external-general-entities" ;
893
- saxParserFactory .setFeature (name , isProcessExternalEntities ());
911
+ SAXParserFactory saxParserFactory = this .sourceParserFactory ;
912
+ if (saxParserFactory == null ) {
913
+ saxParserFactory = SAXParserFactory .newInstance ();
914
+ saxParserFactory .setNamespaceAware (true );
915
+ saxParserFactory .setFeature (
916
+ "http://apache.org/xml/features/disallow-doctype-decl" , !isSupportDtd ());
917
+ saxParserFactory .setFeature (
918
+ "http://xml.org/sax/features/external-general-entities" , isProcessExternalEntities ());
919
+ this .sourceParserFactory = saxParserFactory ;
920
+ }
894
921
SAXParser saxParser = saxParserFactory .newSAXParser ();
895
922
xmlReader = saxParser .getXMLReader ();
896
923
}
0 commit comments