#java #maven #javafx #maven-shade-plugin
#java #maven #javafx #maven-shade-plugin
Вопрос:
Я только что закончил сборку приложения с использованием JavaFX и Hibernate, которое отлично собирается и запускается на IntelliJ, поэтому я решил сгенерировать «fat-jar» с помощью плагина maven shade, поскольку я использую некоторые библиотеки с автоматическими именами модулей, поэтому я не могу использовать jlink.
Jar создан нормально, он правильно загружает начальный экран, однако, когда он пытается установить соединение с моей базой данных, в частности, эта строка является проблемной (JPAUtil.java:19);
factory = Persistence.createEntityManagerFactory(persistenceUnit, properties);
Я получаю следующие исключения;
java.lang.IllegalStateException: Required class information is missing
at org.jboss.jandex.Indexer.rebuildNestedType(Indexer.java:926)
at org.jboss.jandex.Indexer.resolveTypePath(Indexer.java:786)
at org.jboss.jandex.Indexer.resolveTypeAnnotation(Indexer.java:705)
at org.jboss.jandex.Indexer.resolveTypeAnnotations(Indexer.java:613)
at org.jboss.jandex.Indexer.index(Indexer.java:1602)
at org.hibernate.boot.archive.scan.spi.ClassFileArchiveEntryHandler.toClassDescriptor(ClassFileArchiveEntryHandler.java:64)
at org.hibernate.boot.archive.scan.spi.ClassFileArchiveEntryHandler.handleEntry(ClassFileArchiveEntryHandler.java:52)
at org.hibernate.boot.archive.internal.JarFileBasedArchiveDescriptor.visitArchive(JarFileBasedArchiveDescriptor.java:147)
at org.hibernate.boot.archive.scan.spi.AbstractScannerImpl.scan(AbstractScannerImpl.java:48)
at org.hibernate.boot.model.process.internal.ScanningCoordinator.coordinateScan(ScanningCoordinator.java:76)
at org.hibernate.boot.model.process.spi.MetadataBuildingProcess.prepare(MetadataBuildingProcess.java:98)
at org.hibernate.jpa.boot.internal.EntityManagerFactoryBuilderImpl.<init>(EntityManagerFactoryBuilderImpl.java:254)
at org.hibernate.jpa.boot.internal.EntityManagerFactoryBuilderImpl.<init>(EntityManagerFactoryBuilderImpl.java:175)
at org.hibernate.jpa.boot.spi.Bootstrap.getEntityManagerFactoryBuilder(Bootstrap.java:76)
at org.hibernate.jpa.HibernatePersistenceProvider.getEntityManagerFactoryBuilder(HibernatePersistenceProvider.java:171)
at org.hibernate.jpa.HibernatePersistenceProvider.getEntityManagerFactoryBuilderOrNull(HibernatePersistenceProvider.java:119)
at org.hibernate.jpa.HibernatePersistenceProvider.getEntityManagerFactoryBuilderOrNull(HibernatePersistenceProvider.java:61)
at org.hibernate.jpa.HibernatePersistenceProvider.createEntityManagerFactory(HibernatePersistenceProvider.java:50)
at javax.persistence.Persistence.createEntityManagerFactory(Persistence.java:79)
at com.it.util.JPAUtil.getEntityManagerFactory(JPAUtil.java:19)
at com.it.controller.HomeController.initializeSettings(HomeController.java:1571)
at com.it.controller.HomeController.initialize(HomeController.java:1407)
at javafx.fxml.FXMLLoader.loadImpl(FXMLLoader.java:2573)
at javafx.fxml.FXMLLoader.loadImpl(FXMLLoader.java:2466)
at javafx.fxml.FXMLLoader.loadImpl(FXMLLoader.java:3237)
at javafx.fxml.FXMLLoader.loadImpl(FXMLLoader.java:3194)
at javafx.fxml.FXMLLoader.loadImpl(FXMLLoader.java:3163)
at javafx.fxml.FXMLLoader.loadImpl(FXMLLoader.java:3136)
at javafx.fxml.FXMLLoader.loadImpl(FXMLLoader.java:3113)
at javafx.fxml.FXMLLoader.load(FXMLLoader.java:3106)
at com.it.Home.start(Home.java:23)
at com.sun.javafx.application.LauncherImpl.lambda$launchApplication1$9(LauncherImpl.java:846)
at com.sun.javafx.application.PlatformImpl.lambda$runAndWait$12(PlatformImpl.java:455)
at com.sun.javafx.application.PlatformImpl.lambda$runLater$10(PlatformImpl.java:428)
at java.base/java.security.AccessController.doPrivileged(Native Method)
at com.sun.javafx.application.PlatformImpl.lambda$runLater$11(PlatformImpl.java:427)
at com.sun.glass.ui.InvokeLaterDispatcher$Future.run(InvokeLaterDispatcher.java:96)
at com.sun.glass.ui.win.WinApplication._runLoop(Native Method)
at com.sun.glass.ui.win.WinApplication.lambda$runLoop$3(WinApplication.java:174)
at java.base/java.lang.Thread.run(Thread.java:834)
Я знаю, что синтаксис правильный, как я упоминал ранее, он корректно работает в IDE, поэтому я подозреваю, что чего-то не хватает в файле jar, сгенерированном shade, однако я уже потратил на это почти 3 дня и просто не могу найти причину. Информация, показанная в stacktrace, не очень помогает, по крайней мере, я ничего не вижу — я даже просмотрел все классы, упомянутые в stacktrace, и все они, похоже, находятся в сгенерированном Jar.
Надеюсь, у кого-нибудь здесь было что-то подобное или, возможно, он лучше меня знаком с процессом maven shade. Будем признательны за любую помощь или указание на то, на что можно обратить внимание.
Дополнительные файлы, которые помогли бы определить проблему, следующие;
persistence.xml — другие свойства загружаются в код из XML-файла.
<?xml version="1.0" encoding="UTF-8"?>
<persistence version="2.2" xmlns="http://xmlns.jcp.org/xml/ns/persistence"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/persistence http://xmlns.jcp.org/xml/ns/persistence/persistence_2_2.xsd">
<persistence-unit name="Project-Postgre">
<description>Hibernate JPA Configuration</description>
<provider>org.hibernate.jpa.HibernatePersistenceProvider</provider>
<properties>
<property name="javax.persistence.jdbc.driver" value="org.postgresql.Driver" />
<property name="hibernate.dialect" value="org.hibernate.dialect.PostgreSQLDialect"/>
<property name="hibernate.show_sql" value="false" />
<property name="hibernate.format_sql" value = "false" />
</properties>
</persistence-unit>
</persistence>
pom.xml
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
<modelVersion>4.0.0</modelVersion>
<groupId>com.it</groupId>
<artifactId>Project</artifactId>
<version>1.0</version>
<packaging>jar</packaging>
<properties>
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
<maven.compiler.source>11</maven.compiler.source>
<maven.compiler.target>11</maven.compiler.target>
<maven.compiler.release>11</maven.compiler.release>
<javafx.version>14</javafx.version>
</properties>
<name>Project</name>
<dependencies>
<dependency>
<groupId>org.openjfx</groupId>
<artifactId>javafx-controls</artifactId>
<version>${javafx.version}</version>
</dependency>
<dependency>
<groupId>org.openjfx</groupId>
<artifactId>javafx-fxml</artifactId>
<version>${javafx.version}</version>
</dependency>
<dependency>
<groupId>javax.persistence</groupId>
<artifactId>javax.persistence-api</artifactId>
<version>2.2</version>
</dependency>
<!-- https://mvnrepository.com/artifact/org.apache.poi/poi -->
<dependency>
<groupId>org.apache.poi</groupId>
<artifactId>poi</artifactId>
<version>4.1.2</version>
</dependency>
<!-- https://mvnrepository.com/artifact/org.apache.poi/poi-ooxml -->
<dependency>
<groupId>org.apache.poi</groupId>
<artifactId>poi-ooxml</artifactId>
<version>4.1.2</version>
</dependency>
<!-- https://mvnrepository.com/artifact/de.jensd/fontawesomefx -->
<dependency>
<groupId>de.jensd</groupId>
<artifactId>fontawesomefx</artifactId>
<version>8.2</version>
</dependency>
<!-- https://mvnrepository.com/artifact/org.hibernate/hibernate-core -->
<dependency>
<groupId>org.hibernate</groupId>
<artifactId>hibernate-core</artifactId>
<version>5.4.20.Final</version>
</dependency>
<!-- https://mvnrepository.com/artifact/org.postgresql/postgresql -->
<dependency>
<groupId>org.postgresql</groupId>
<artifactId>postgresql</artifactId>
<version>42.2.15</version>
</dependency>
<!-- https://mvnrepository.com/artifact/com.oracle.database.jdbc/ojdbc10 -->
<dependency>
<groupId>com.oracle.database.jdbc</groupId>
<artifactId>ojdbc10</artifactId>
<version>19.7.0.0</version>
</dependency>
</dependencies>
<build>
<plugins>
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-compiler-plugin</artifactId>
<version>3.8.1</version>
<configuration>
<release>${maven.compiler.release}</release>
</configuration>
</plugin>
<plugin>
<groupId>org.openjfx</groupId>
<artifactId>javafx-maven-plugin</artifactId>
<version>0.0.3</version>
<configuration>
<mainClass>com.it.HomeFX</mainClass>
</configuration>
</plugin>
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-shade-plugin</artifactId>
<version>3.2.0</version>
<executions>
<execution>
<phase>package</phase>
<goals>
<goal>shade</goal>
</goals>
<configuration>
<shadedArtifactAttached>true</shadedArtifactAttached>
<shadedClassifierName>project-classifier</shadedClassifierName>
<outputFile>targetshade${project.artifactId}.jar</outputFile>
<transformers>
<transformer implementation=
"org.apache.maven.plugins.shade.resource.ManifestResourceTransformer">
<mainClass>com.it.HomeFX</mainClass>
</transformer>
</transformers>
</configuration>
</execution>
</executions>
</plugin>
</plugins>
</build>
</project>
JPAUtil
public static EntityManagerFactory getEntityManagerFactory(String persistenceUnit, String connectionURL, String user, String password) {
if (factory == null) {
Map<String, String> properties = new HashMap<>();
properties.put("javax.persistence.jdbc.url", connectionURL);
properties.put("javax.persistence.jdbc.user", user);
properties.put("javax.persistence.jdbc.password", password);
factory = Persistence.createEntityManagerFactory(persistenceUnit, properties);
}
return factory;
}
Комментарии:
1. Что насчет
persistence.xml
файла? вы уверены, что у вас там есть вся информация? обратите внимание, что согласно вашим журналам, ваш @Controller не может работать, и это из-заEntityManagerFactory
не работает.2. Я уверен, что все есть, как я упоминал ранее, программа корректно работает на IntelliJ. В любом случае, я также добавил свой класс JPAUtil в исходное сообщение, так что вы можете увидеть все дополнительные сведения из persistence.xml который загружается для emf.
Ответ №1:
У меня была та же ошибка при использовании: spring.версия: 5.2.7.RELEASE spring.boot.version: 2.3.1.RELEASE postgresql-версия: 42.2.16
Это происходит со всеми версиями postgres начиная с 42.2.15. Версия 42.2.14 (или более ранняя) для меня работает нормально.
Ответ №2:
У меня была такая же проблема. Я попытался понизить версию драйвера PostgreSQL JDBC до 42.0.0. Исключение исчезло, и все работает нормально.
Ответ №3:
Проблема вызвана недопустимым байт-кодом, созданным javac
1.8 (например, известно, что AdoptOpenJDK 1.8u222 подвержен влиянию).
Вот проблема (с reproducer):https://github.com/wildfly/jandex/issues/92
Исправление заключается в обновлении до org.jboss:jandex:2.2.3.Final
, которое включает обходной путь (и пару других исправлений для аннотаций типов), или использовании компилятора Java 11 (== javac
11).
На всякий случай, org.postgresql:postgresql:42.2.15
вызывает проблему, поскольку он использует Checker Framework для проверки на нулевость и включает в себя несколько @Nullable
и @NonNull
аннотаций, чтобы проверки проходили успешно.
Готовящаяся к выпуску org.postgresql:42.2.18
и org.postgresql:42.3.0
будет иметь соответствующие обходные пути, поэтому она будет работать и со старыми jandex
версиями.
Если вы читаете здесь, я бы рекомендовал добавить в ваш конвейер сборки анализаторы байт-кода forbidden-apis и jandex (например, плагины de.thetaphi.forbiddenapis
и com.github.vlsi.jandex
Gradle) для раннего обнаружения недопустимого байт-кода (который в противном случае мог бы остаться незамеченным), особенно если вы все еще используете javac
1.8.
Ответ №4:
Мне удалось преодолеть эту проблему, переключившись с гибернации на OpenJPA. Hibernate не предоставил достаточно информации для отладки проблемы, и при переключении на OpenJPA мне просто пришлось добавить это в свой pom.xml;
<dependency>
<groupId>org.apache.openjpa</groupId>
<artifactId>openjpa-all</artifactId>
<version>3.1.2</version>
</dependency>
Shade создал JAR со всеми необходимыми библиотеками, и подключение к базе данных было успешно.
Ответ №5:
В конечном итоге исправлено в драйвере PostgreSQL JDBC версии 42.2.19, см. https://jdbc.postgresql.org/documentation/changelog.html#version_42.2.19 .
Обновите свою версию до 42.2.19, и это должно решить проблему.