LANGUAGE/!$%!% ERROR NOTE
[SpringBoot + ElasticSearch] NoClassDefFoundError: org/elasticsearch/common/CheckedConsumer
forgiveall
2021. 1. 5. 20:24
SpringBoot 1.5.4 + ElasticSearch 7.3.2
SpringBoot에 물려서 높은 버전의 ElasticSearch로 설정할 경우 오류가 발생하는 경우가 있지만 해결방법은 있다.
1. Error
Error
Caused by: org.springframework.beans.factory.BeanCreationException: Error creating bean with name 'elasticSearchConfig': Invocation of init method failed; nested exception is java.lang.NoClassDefFoundError: org/elasticsearch/common/CheckedConsumer ... Caused by: java.lang.NoClassDefFoundError: org/elasticsearch/common/CheckedConsumer ... Caused by: java.lang.ClassNotFoundException: org.elasticsearch.common.CheckedConsumer at java.net.URLClassLoader.findClass(URLClassLoader.java:382) ~[na:1.8.0_211] at java.lang.ClassLoader.loadClass(ClassLoader.java:424) ~[na:1.8.0_211] at sun.misc.Launcher$AppClassLoader.loadClass(Launcher.java:349) ~[na:1.8.0_211] at java.lang.ClassLoader.loadClass(ClassLoader.java:357) ~[na:1.8.0_211] ... 87 common frames omitted
2. Problem
SpringBoot
에서 제공해주는 ElasticSearch API에서 벗어나서 더 높은 버전의 독립된 API로 업그레이드
하는 상황에서 발생하였다.
- ※ 환경
- JDK: 1.8
- Gradle: 4.10.3
SpringBoot
: 1.5.4ElasticSearch
: 7.3.2
BEFORE
build.gradle
compile 'org.elasticsearch:elasticsearch:1.5.2' compile 'org.springframework.data:spring-data-commons:1.11.4.RELEASE' compile 'org.springframework.data:spring-data-elasticsearch:1.3.4.RELEASE'
AFTER
build.gradle
compile "org.elasticsearch:elasticsearch:7.3.2" compile "org.elasticsearch.client:elasticsearch-rest-client:7.3.2" compile "org.elasticsearch.client:elasticsearch-rest-high-level-client:7.3.2"
7.3.2로 변경하는 과정에서 어떻게하더라도 2.4.5버전의 elasticsearch에 의존성이 설정되어 오류가 발생하는 것으로 보인다.
gradle로 의존성을 tree구조로 확인해보면 명시한 버전(7.3.2
)이 아닌 자동으로 옛버전(2.4.5
)으로 강제설정되는 것을 알 수 있다.
Command Line에서 Gradle을 사용하여 tree구조로 의존성 보기
gradle dependencies
+--- org.elasticsearch:elasticsearch:7.3.2 -> 2.4.5
'spring-boot' plugin이 SpringBoot버전에 가장 적합한 ElasticSearch로 전환시켜버리는 모양
이다.
3. Solved
시도 1. (실패)
내가 아는 의존성모듈의 버전 강제하는 방법을 시도하였으나 적용되지 않았다.
build.gradle
configurations.all { resolutionStrategy.eachDependency { DependencyResolveDetails details -> def req = details.requested if (req.group == 'org.elasticsearch' && req.name == 'elasticsearch'){ details.useVersion '7.3.2' details.because "$req.group:$req.name:$req.version ==> Correct to official IRUDA Session Version ==> 7.3.2" } } }
시도 2. (성공)
'spring-boot' plugin이 의존성을 특정할 수 있도록 ext 변수를 적용하여 설정하는 옵션을 제공
한다.
build.gradle
ext { set('elasticsearch.version', elasticsearch_version) }
시도 3. (성공)
또는 'spring-boot' plugin의 dependencyManagement 기능을 이용하여 특정
할 수 있다.
build.gradle
dependencyManagement { dependencies { dependency group:'org.elasticsearch', name:'elasticsearch', version:'7.3.2' } }
4. Reference
- Dependency Management DSL: https://docs.spring.io/dependency-management-plugin/docs/current/reference/html/#dependency-management-configuration-dsl
- Spring Boot, Elasticsearch 6.2.4, Gradle dependency issues: https://stackoverflow.com/questions/52447986/spring-boot-elasticsearch-6-2-4-gradle-dependency-issues
- NoClassDefFoundError error creating RestHighLevelClient bean: https://stackoverflow.com/questions/46854919/noclassdeffounderror-error-creating-resthighlevelclient-bean
- Overriding Dependency Versions with Spring Boot: https://spring.io/blog/2016/04/13/overriding-dependency-versions-with-spring-boot