Quarkus 的自定义配置源
Custom ConfigSource for Quarkus
我现在正在尝试在我的 Quarkus 应用程序中配置自定义 ConfigSource。与许多其他手册一样,我创建了自己的 DatabaseSourceConfig 并实现了 org.eclipse.microprofile.config.spi.ConfigSource interface.I 在以下位置注册了我的 ConfigSource:
/META-INF/services/org.eclipse.microprofile.config.spi.ConfigSource
这是我的配置源:
public class DatabaseConfigSource implements ConfigSource {
private DataSource dataSource;
public DatabaseConfigSource() {
try {
dataSource = (DataSource) new InitialContext().lookup("openejb:Resource/config-source-database");
} catch (final NamingException e) {
throw new IllegalStateException(e);
}
}
@Override
public Map<String, String> getProperties() {
// Implementing Method
}
@Override
public String getValue(final String propertyName) {
// Implementing Method
}
@Override
public String getName() {
return DatabaseConfigSource.class.getSimpleName();
}
}
但是由于 JNDI 名称,这不适用于 Quarkus。我需要使用 CDI。我试图使用这样的东西:
@Inject
@io.quarkus.agroal.DataSource("my_connection")
AgroalDataSource usersDataSource;
并在 application.properties 中声明此连接,但它对我没有帮助。我一直收到 NULL 异常。
也许有人有想法,我如何在不使用 JNDI 命名空间的情况下获得数据库连接?
您可以通过
获取数据源
AgroalDataSource dataSource = Arc.container()
.instance(AgroalDataSource.class, new DataSource.DataSourceLiteral("my_connection"))
.get();
尽管我认为您需要在构造函数之外的其他地方执行此操作,因为 ConfigSource 实例是在 CDI 完全启动之前创建的。可以缓存获取到的数据源实例,避免多次执行。
我自己找到了一些答案,也许对其他人也有用。
正如@Janmartiška 所说,CDI 比ConfigSource 启动得晚,这就是为什么我看不到任何通过CDI 注入连接的原因。
我创建了一些 HibernateUtil Class:
package org.myproject.config;
import java.util.Properties;
import org.hibernate.SessionFactory;
import org.hibernate.boot.registry.StandardServiceRegistryBuilder;
import org.hibernate.cfg.Configuration;
import org.hibernate.service.ServiceRegistry;
import org.myproject.entities.ConfigurationsEntity;
public class HibernateUtil {
private static SessionFactory sessionFactory;
private static SessionFactory buildSessionFactory() {
try {
Properties props = new Properties();
props.setProperty("hibernate.connection.url", "jdbc:mysql://[db-host]:[db-port]/db_name");
props.setProperty("hibernate.connection.driver_class", "com.mysql.cj.jdbc.Driver");
props.setProperty("hibernate.connection.username", "username");
props.setProperty("hibernate.connection.password", "password");
props.setProperty("hibernate.current_session_context_class", "thread");
props.setProperty("hibernate.dialect", "org.hibernate.dialect.MySQL8Dialect");
Configuration configuration = new Configuration();
configuration.addProperties(props);
configuration.addAnnotatedClass(ConfigurationsEntity.class);
System.out.println("Hibernate Configuration loaded");
ServiceRegistry serviceRegistry = new StandardServiceRegistryBuilder().applySettings(configuration.getProperties()).build();
System.out.println("Hibernate serviceRegistry created");
SessionFactory sessionFactory = configuration.buildSessionFactory(serviceRegistry);
return sessionFactory;
}
catch (Throwable ex) {
// Make sure you log the exception, as it might be swallowed
System.err.println("Initial SessionFactory creation failed." + ex);
throw new ExceptionInInitializerError(ex);
}
}
public static SessionFactory getSessionFactory() {
if(sessionFactory == null) sessionFactory = buildSessionFactory();
return sessionFactory;
}
}
比我在我的 SourceConfig 中使用的还要多:
package org.myproject.config;
import io.quarkus.runtime.annotations.RegisterForReflection;
import org.eclipse.microprofile.config.spi.ConfigSource;
import org.hibernate.Session;
import org.hibernate.SessionFactory;
import org.myproject.entities.ConfigurationsEntity;
import javax.persistence.NoResultException;
import javax.persistence.Query;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
@RegisterForReflection
public class DatabaseSourceConfig implements ConfigSource {
public SessionFactory sessionFactory;
public Session currentSession;
public DatabaseSourceConfig() {
sessionFactory = HibernateUtil.getSessionFactory();
this.checkFactoryConnection();
}
public void checkFactoryConnection() {
if (currentSession == null || (currentSession != null && !currentSession.isOpen())) {
try {
currentSession = sessionFactory.getCurrentSession();
} catch (NullPointerException e) {
currentSession = sessionFactory.openSession();
}
}
}
@Override
public Map<String, String> getProperties() {
// Implementing Method
}
@Override
public String getValue(String propertyName) {
this.checkFactoryConnection();
ConfigurationsEntity conf = new ConfigurationsEntity();
currentSession.beginTransaction();
try {
Query query = currentSession.createNamedQuery("Configuration.selectOne", ConfigurationsEntity.class);
query.setParameter("name", propertyName);
conf = (ConfigurationsEntity) query.getSingleResult();
currentSession.getTransaction().commit();
} catch (Exception ex) {
currentSession.getTransaction().rollback();
}
return conf.getValue();
}
@Override
public String getName() {
return DatabaseSourceConfig.class.getSimpleName();
}
}
现在我可以在其他 类 中使用我的 ConfigSource,例如:
@Inject
@ConfigProperty(name = "[property-name-like-in-db]")
public String someProperty;
经过我的进一步研究,发现 ConfigSource 无法访问 CDi 和 application.properties。这就是为什么除了以上述方式建立与数据库的连接之外别无他法。
但是,我对该示例进行了一些编辑。我缓存了数据库中的属性并创建了一个 @ApplicationScoped Bean,它每 5 分钟查看一次数据库,以查看其中一个属性 "updated_at" 的时间戳是否晚于 Bean 加载属性中的其他属性。
但是,我不得不说,根据 Quarkus 和 Apache 开发人员的说法——这违反了“不可变部署”,并且不打算在运行时更改应用程序设置。所以写不写在app里就看你了
我现在正在尝试在我的 Quarkus 应用程序中配置自定义 ConfigSource。与许多其他手册一样,我创建了自己的 DatabaseSourceConfig 并实现了 org.eclipse.microprofile.config.spi.ConfigSource interface.I 在以下位置注册了我的 ConfigSource:
/META-INF/services/org.eclipse.microprofile.config.spi.ConfigSource
这是我的配置源:
public class DatabaseConfigSource implements ConfigSource {
private DataSource dataSource;
public DatabaseConfigSource() {
try {
dataSource = (DataSource) new InitialContext().lookup("openejb:Resource/config-source-database");
} catch (final NamingException e) {
throw new IllegalStateException(e);
}
}
@Override
public Map<String, String> getProperties() {
// Implementing Method
}
@Override
public String getValue(final String propertyName) {
// Implementing Method
}
@Override
public String getName() {
return DatabaseConfigSource.class.getSimpleName();
}
}
但是由于 JNDI 名称,这不适用于 Quarkus。我需要使用 CDI。我试图使用这样的东西:
@Inject
@io.quarkus.agroal.DataSource("my_connection")
AgroalDataSource usersDataSource;
并在 application.properties 中声明此连接,但它对我没有帮助。我一直收到 NULL 异常。 也许有人有想法,我如何在不使用 JNDI 命名空间的情况下获得数据库连接?
您可以通过
获取数据源AgroalDataSource dataSource = Arc.container()
.instance(AgroalDataSource.class, new DataSource.DataSourceLiteral("my_connection"))
.get();
尽管我认为您需要在构造函数之外的其他地方执行此操作,因为 ConfigSource 实例是在 CDI 完全启动之前创建的。可以缓存获取到的数据源实例,避免多次执行。
我自己找到了一些答案,也许对其他人也有用。 正如@Janmartiška 所说,CDI 比ConfigSource 启动得晚,这就是为什么我看不到任何通过CDI 注入连接的原因。 我创建了一些 HibernateUtil Class:
package org.myproject.config;
import java.util.Properties;
import org.hibernate.SessionFactory;
import org.hibernate.boot.registry.StandardServiceRegistryBuilder;
import org.hibernate.cfg.Configuration;
import org.hibernate.service.ServiceRegistry;
import org.myproject.entities.ConfigurationsEntity;
public class HibernateUtil {
private static SessionFactory sessionFactory;
private static SessionFactory buildSessionFactory() {
try {
Properties props = new Properties();
props.setProperty("hibernate.connection.url", "jdbc:mysql://[db-host]:[db-port]/db_name");
props.setProperty("hibernate.connection.driver_class", "com.mysql.cj.jdbc.Driver");
props.setProperty("hibernate.connection.username", "username");
props.setProperty("hibernate.connection.password", "password");
props.setProperty("hibernate.current_session_context_class", "thread");
props.setProperty("hibernate.dialect", "org.hibernate.dialect.MySQL8Dialect");
Configuration configuration = new Configuration();
configuration.addProperties(props);
configuration.addAnnotatedClass(ConfigurationsEntity.class);
System.out.println("Hibernate Configuration loaded");
ServiceRegistry serviceRegistry = new StandardServiceRegistryBuilder().applySettings(configuration.getProperties()).build();
System.out.println("Hibernate serviceRegistry created");
SessionFactory sessionFactory = configuration.buildSessionFactory(serviceRegistry);
return sessionFactory;
}
catch (Throwable ex) {
// Make sure you log the exception, as it might be swallowed
System.err.println("Initial SessionFactory creation failed." + ex);
throw new ExceptionInInitializerError(ex);
}
}
public static SessionFactory getSessionFactory() {
if(sessionFactory == null) sessionFactory = buildSessionFactory();
return sessionFactory;
}
}
比我在我的 SourceConfig 中使用的还要多:
package org.myproject.config;
import io.quarkus.runtime.annotations.RegisterForReflection;
import org.eclipse.microprofile.config.spi.ConfigSource;
import org.hibernate.Session;
import org.hibernate.SessionFactory;
import org.myproject.entities.ConfigurationsEntity;
import javax.persistence.NoResultException;
import javax.persistence.Query;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
@RegisterForReflection
public class DatabaseSourceConfig implements ConfigSource {
public SessionFactory sessionFactory;
public Session currentSession;
public DatabaseSourceConfig() {
sessionFactory = HibernateUtil.getSessionFactory();
this.checkFactoryConnection();
}
public void checkFactoryConnection() {
if (currentSession == null || (currentSession != null && !currentSession.isOpen())) {
try {
currentSession = sessionFactory.getCurrentSession();
} catch (NullPointerException e) {
currentSession = sessionFactory.openSession();
}
}
}
@Override
public Map<String, String> getProperties() {
// Implementing Method
}
@Override
public String getValue(String propertyName) {
this.checkFactoryConnection();
ConfigurationsEntity conf = new ConfigurationsEntity();
currentSession.beginTransaction();
try {
Query query = currentSession.createNamedQuery("Configuration.selectOne", ConfigurationsEntity.class);
query.setParameter("name", propertyName);
conf = (ConfigurationsEntity) query.getSingleResult();
currentSession.getTransaction().commit();
} catch (Exception ex) {
currentSession.getTransaction().rollback();
}
return conf.getValue();
}
@Override
public String getName() {
return DatabaseSourceConfig.class.getSimpleName();
}
}
现在我可以在其他 类 中使用我的 ConfigSource,例如:
@Inject
@ConfigProperty(name = "[property-name-like-in-db]")
public String someProperty;
经过我的进一步研究,发现 ConfigSource 无法访问 CDi 和 application.properties。这就是为什么除了以上述方式建立与数据库的连接之外别无他法。 但是,我对该示例进行了一些编辑。我缓存了数据库中的属性并创建了一个 @ApplicationScoped Bean,它每 5 分钟查看一次数据库,以查看其中一个属性 "updated_at" 的时间戳是否晚于 Bean 加载属性中的其他属性。
但是,我不得不说,根据 Quarkus 和 Apache 开发人员的说法——这违反了“不可变部署”,并且不打算在运行时更改应用程序设置。所以写不写在app里就看你了