io.quarkiverse.mybatis.deployment.MyBatisProcessor Maven / Gradle / Ivy
package io.quarkiverse.mybatis.deployment;
import java.util.ArrayList;
import java.util.List;
import java.util.Map;
import java.util.Optional;
import java.util.function.Consumer;
import java.util.function.Function;
import java.util.stream.Collectors;
import jakarta.inject.Named;
import jakarta.inject.Singleton;
import org.apache.commons.lang3.tuple.Pair;
import org.apache.ibatis.annotations.DeleteProvider;
import org.apache.ibatis.annotations.InsertProvider;
import org.apache.ibatis.annotations.Mapper;
import org.apache.ibatis.annotations.Result;
import org.apache.ibatis.annotations.ResultMap;
import org.apache.ibatis.annotations.ResultType;
import org.apache.ibatis.annotations.Results;
import org.apache.ibatis.annotations.SelectProvider;
import org.apache.ibatis.annotations.UpdateProvider;
import org.apache.ibatis.cache.decorators.LruCache;
import org.apache.ibatis.cache.impl.PerpetualCache;
import org.apache.ibatis.javassist.util.proxy.ProxyFactory;
import org.apache.ibatis.scripting.defaults.RawLanguageDriver;
import org.apache.ibatis.scripting.xmltags.XMLLanguageDriver;
import org.apache.ibatis.session.Configuration;
import org.apache.ibatis.session.SqlSessionFactory;
import org.apache.ibatis.session.SqlSessionFactoryBuilder;
import org.apache.ibatis.type.EnumTypeHandler;
import org.apache.ibatis.type.MappedJdbcTypes;
import org.apache.ibatis.type.MappedTypes;
import org.jboss.jandex.AnnotationInstance;
import org.jboss.jandex.AnnotationTarget;
import org.jboss.jandex.DotName;
import org.jboss.logging.Logger;
import io.quarkiverse.mybatis.runtime.MyBatisConfigurationFactory;
import io.quarkiverse.mybatis.runtime.MyBatisRecorder;
import io.quarkiverse.mybatis.runtime.MyBatisXMLConfigDelegateBuilder;
import io.quarkiverse.mybatis.runtime.config.MyBatisDataSourceRuntimeConfig;
import io.quarkiverse.mybatis.runtime.config.MyBatisRuntimeConfig;
import io.quarkiverse.mybatis.runtime.meta.MapperDataSource;
import io.quarkus.agroal.spi.JdbcDataSourceBuildItem;
import io.quarkus.arc.deployment.SyntheticBeanBuildItem;
import io.quarkus.arc.deployment.SyntheticBeansRuntimeInitBuildItem;
import io.quarkus.deployment.annotations.BuildProducer;
import io.quarkus.deployment.annotations.BuildStep;
import io.quarkus.deployment.annotations.Consume;
import io.quarkus.deployment.annotations.ExecutionTime;
import io.quarkus.deployment.annotations.Overridable;
import io.quarkus.deployment.annotations.Record;
import io.quarkus.deployment.builditem.CombinedIndexBuildItem;
import io.quarkus.deployment.builditem.FeatureBuildItem;
import io.quarkus.deployment.builditem.nativeimage.NativeImageProxyDefinitionBuildItem;
import io.quarkus.deployment.builditem.nativeimage.NativeImageResourceBuildItem;
import io.quarkus.deployment.builditem.nativeimage.ReflectiveClassBuildItem;
import io.quarkus.deployment.builditem.nativeimage.RuntimeInitializedClassBuildItem;
import io.quarkus.runtime.configuration.ConfigurationException;
@SuppressWarnings("unused")
public class MyBatisProcessor {
private static final Logger LOG = Logger.getLogger(MyBatisProcessor.class);
private static final String FEATURE = "mybatis";
private static final DotName MYBATIS_MAPPER = DotName.createSimple(Mapper.class.getName());
private static final DotName MYBATIS_TYPE_HANDLER = DotName.createSimple(MappedTypes.class.getName());
private static final DotName MYBATIS_JDBC_TYPE_HANDLER = DotName.createSimple(MappedJdbcTypes.class.getName());
private static final DotName MYBATIS_MAPPER_DATA_SOURCE = DotName.createSimple(MapperDataSource.class.getName());
@BuildStep
FeatureBuildItem feature() {
return new FeatureBuildItem(FEATURE);
}
@BuildStep
void runtimeInitialized(BuildProducer runtimeInit) {
runtimeInit.produce(new RuntimeInitializedClassBuildItem("org.apache.ibatis.logging.log4j.Log4jImpl"));
}
@BuildStep
void reflectiveClasses(BuildProducer reflectiveClass) {
reflectiveClass.produce(ReflectiveClassBuildItem.builder(ProxyFactory.class,
XMLLanguageDriver.class,
RawLanguageDriver.class,
SelectProvider.class,
UpdateProvider.class,
InsertProvider.class,
DeleteProvider.class,
Result.class,
Results.class,
ResultType.class,
ResultMap.class,
EnumTypeHandler.class).methods(false).fields(false).build());
reflectiveClass.produce(
ReflectiveClassBuildItem.builder(PerpetualCache.class, LruCache.class).methods(true).fields(true).build());
}
@BuildStep
@Overridable
void addMyBatisMappers(BuildProducer mappers,
BuildProducer reflective,
BuildProducer proxy,
CombinedIndexBuildItem indexBuildItem) {
for (AnnotationInstance i : indexBuildItem.getIndex().getAnnotations(MYBATIS_MAPPER)) {
if (i.target().kind() == AnnotationTarget.Kind.CLASS) {
DotName dotName = i.target().asClass().name();
reflective.produce(ReflectiveClassBuildItem.builder(dotName.toString()).methods(true).fields(false).build());
proxy.produce(new NativeImageProxyDefinitionBuildItem(dotName.toString()));
Optional mapperDatasource = i.target().asClass().annotationsMap().entrySet().stream()
.filter(entry -> entry.getKey().equals(MYBATIS_MAPPER_DATA_SOURCE))
.map(Map.Entry::getValue)
.map(annotationList -> annotationList.get(0))
.findFirst();
if (mapperDatasource.isPresent()) {
String dataSourceName = mapperDatasource.get().value().asString();
mappers.produce(new MyBatisMapperBuildItem(dotName, dataSourceName));
} else {
mappers.produce(new MyBatisMapperBuildItem(dotName, ""));
}
}
}
}
@BuildStep
void addMyBatisMappedTypes(BuildProducer mappedTypes,
BuildProducer mappedJdbcTypes,
CombinedIndexBuildItem indexBuildItem) {
List names = new ArrayList<>();
for (AnnotationInstance i : indexBuildItem.getIndex().getAnnotations(MYBATIS_TYPE_HANDLER)) {
if (i.target().kind() == AnnotationTarget.Kind.CLASS) {
DotName dotName = i.target().asClass().name();
mappedTypes.produce(new MyBatisMappedTypeBuildItem(dotName));
names.add(dotName);
}
}
for (AnnotationInstance i : indexBuildItem.getIndex().getAnnotations(MYBATIS_JDBC_TYPE_HANDLER)) {
if (i.target().kind() == AnnotationTarget.Kind.CLASS) {
DotName dotName = i.target().asClass().name();
if (!names.contains(dotName)) {
mappedJdbcTypes.produce(new MyBatisMappedJdbcTypeBuildItem(dotName));
}
}
}
}
@BuildStep
void initialSql(BuildProducer resource, MyBatisRuntimeConfig config) {
config.initialSql.ifPresent(initialSql -> resource.produce(new NativeImageResourceBuildItem(initialSql)));
config.dataSources.values().forEach(dataSource -> dataSource.initialSql
.ifPresent(initialSql -> resource.produce(new NativeImageResourceBuildItem(initialSql))));
}
@Record(ExecutionTime.STATIC_INIT)
@BuildStep
void generateSqlSessionFactory(MyBatisRuntimeConfig myBatisRuntimeConfig,
ConfigurationFactoryBuildItem configurationFactoryBuildItem,
SqlSessionFactoryBuilderBuildItem sqlSessionFactoryBuilderBuildItem,
List myBatisMapperBuildItems,
List myBatisMappedTypeBuildItems,
List myBatisMappedJdbcTypeBuildItems,
List configurationCustomizerBuildItems,
List jdbcDataSourcesBuildItem,
BuildProducer sqlSessionFactory,
MyBatisRecorder recorder) {
List mappedTypes = myBatisMappedTypeBuildItems
.stream().map(m -> m.getMappedTypeName().toString()).collect(Collectors.toList());
List mappedJdbcTypes = myBatisMappedJdbcTypeBuildItems
.stream().map(m -> m.getMappedJdbcTypeName().toString()).collect(Collectors.toList());
List> dataSources = new ArrayList<>();
if (myBatisRuntimeConfig.dataSource.isPresent()) {
String dataSourceName = myBatisRuntimeConfig.dataSource.get();
Optional jdbcDataSourceBuildItem = jdbcDataSourcesBuildItem.stream()
.filter(i -> i.getName().equals(dataSourceName))
.findFirst();
if (jdbcDataSourceBuildItem.isEmpty()) {
throw new ConfigurationException("Can not find datasource " + dataSourceName);
}
dataSources.add(Pair.of(dataSourceName, true));
} else {
dataSources = jdbcDataSourcesBuildItem.stream()
.map(dataSource -> Pair.of(dataSource.getName(), dataSource.isDefault()))
.collect(Collectors.toList());
if (dataSources.isEmpty()) {
throw new ConfigurationException("No datasource found");
}
}
List> configurationCustomizers = configurationCustomizerBuildItems.stream()
.map(ConfigurationCustomizerBuildItem::getCustomizer)
.collect(Collectors.toList());
dataSources.forEach(dataSource -> {
MyBatisDataSourceRuntimeConfig dataSourceConfig = myBatisRuntimeConfig.dataSources.get(dataSource.getKey());
List mappers = myBatisMapperBuildItems
.stream().filter(m -> m.getDataSourceName().equals(dataSource.getKey()))
.map(m -> m.getMapperName().toString()).collect(Collectors.toList());
sqlSessionFactory.produce(
new SqlSessionFactoryBuildItem(
recorder.createSqlSessionFactory(
configurationFactoryBuildItem.getFactory(),
configurationCustomizers,
sqlSessionFactoryBuilderBuildItem.getBuilder(),
myBatisRuntimeConfig,
dataSourceConfig,
dataSource.getKey(),
mappers,
mappedTypes,
mappedJdbcTypes),
dataSource.getKey(), dataSource.getValue(), false));
});
}
@BuildStep
@Overridable
ConfigurationFactoryBuildItem createConfigurationFactory() {
return new ConfigurationFactoryBuildItem(new MyBatisConfigurationFactory());
}
@BuildStep
@Overridable
SqlSessionFactoryBuilderBuildItem createSqlSessionFactoryBuilder() {
return new SqlSessionFactoryBuilderBuildItem(new SqlSessionFactoryBuilder());
}
@BuildStep
@Overridable
XMLConfigBuilderBuildItem createXMLConfigBuilder() {
return new XMLConfigBuilderBuildItem(new MyBatisXMLConfigDelegateBuilder());
}
@BuildStep
void xmlConfig(MyBatisRuntimeConfig config, BuildProducer xmlConfig) {
if (config.xmlconfig.enable) {
xmlConfig.produce(new MyBatisXmlConfigBuildItem("xmlconfig", true));
}
}
@Record(ExecutionTime.STATIC_INIT)
@BuildStep
void generateSqlSessionFactoryFromXmlConfig(
MyBatisRuntimeConfig config,
XMLConfigBuilderBuildItem xmlConfigBuilderBuildItem,
MyBatisXmlConfigBuildItem xmlConfigBuildItem,
BuildProducer sqlSessionFactory,
MyBatisRecorder recorder) {
if (xmlConfigBuildItem != null && xmlConfigBuildItem.isEnabled()) {
sqlSessionFactory.produce(
new SqlSessionFactoryBuildItem(
recorder.createSqlSessionFactory(config, xmlConfigBuilderBuildItem.getBuilder()),
xmlConfigBuildItem.getName(), false, true));
}
}
@Record(ExecutionTime.RUNTIME_INIT)
@BuildStep
void generateSqlSessionManager(List sqlSessionFactoryBuildItems,
BuildProducer sqlSessionManager,
MyBatisRecorder recorder) {
sqlSessionFactoryBuildItems.forEach(sessionFactory -> sqlSessionManager.produce(
new SqlSessionManagerBuildItem(
recorder.createSqlSessionManager(sessionFactory.getSqlSessionFactory()),
sessionFactory.getDataSourceName(),
sessionFactory.isDefaultDataSource())));
}
@Record(ExecutionTime.RUNTIME_INIT)
@BuildStep
void generateMapperBeans(MyBatisRecorder recorder,
List myBatisMapperBuildItems,
List myBatisMappedTypesBuildItems,
List myBatisMappedJdbcTypesBuildItems,
List sqlSessionManagerBuildItems,
BuildProducer syntheticBeanBuildItemBuildProducer) {
Map dataSourceToSessionManagerBuildItem = sqlSessionManagerBuildItems.stream()
.collect(Collectors.toMap(SqlSessionManagerBuildItem::getDataSourceName, Function.identity()));
SqlSessionManagerBuildItem defaultSqlSessionManagerBuildItem = getDefaultSessionManager(sqlSessionManagerBuildItems);
for (MyBatisMapperBuildItem i : myBatisMapperBuildItems) {
SqlSessionManagerBuildItem sessionManagerBuildItem;
if (i.getDataSourceName() == null) {
if (defaultSqlSessionManagerBuildItem.isDefaultDataSource() || sqlSessionManagerBuildItems.size() == 1) {
sessionManagerBuildItem = defaultSqlSessionManagerBuildItem;
} else {
throw new ConfigurationException("Could not choose data source for mapper: " + i.getMapperName() +
". Please use @MapperDataSource annotation for specified the mapper class");
}
} else {
sessionManagerBuildItem = dataSourceToSessionManagerBuildItem.get(i.getDataSourceName());
if (sessionManagerBuildItem == null) {
throw new ConfigurationException(String.format("Data source %s does not exist", i.getDataSourceName()));
}
}
SyntheticBeanBuildItem.ExtendedBeanConfigurator configurator = SyntheticBeanBuildItem
.configure(i.getMapperName())
.scope(Singleton.class)
.setRuntimeInit()
.unremovable()
.supplier(recorder.MyBatisMapperSupplier(i.getMapperName().toString(),
sessionManagerBuildItem.getSqlSessionManager()));
syntheticBeanBuildItemBuildProducer.produce(configurator.done());
}
for (MyBatisMappedTypeBuildItem i : myBatisMappedTypesBuildItems) {
SyntheticBeanBuildItem.ExtendedBeanConfigurator configurator = SyntheticBeanBuildItem
.configure(i.getMappedTypeName())
.scope(Singleton.class)
.setRuntimeInit()
.unremovable()
.supplier(recorder.MyBatisMappedTypeSupplier(i.getMappedTypeName().toString(),
defaultSqlSessionManagerBuildItem.getSqlSessionManager()));
syntheticBeanBuildItemBuildProducer.produce(configurator.done());
}
for (MyBatisMappedJdbcTypeBuildItem i : myBatisMappedJdbcTypesBuildItems) {
SyntheticBeanBuildItem.ExtendedBeanConfigurator configurator = SyntheticBeanBuildItem
.configure(i.getMappedJdbcTypeName())
.scope(Singleton.class)
.setRuntimeInit()
.unremovable()
.supplier(recorder.MyBatisMappedJdbcTypeSupplier(i.getMappedJdbcTypeName().toString(),
defaultSqlSessionManagerBuildItem.getSqlSessionManager()));
syntheticBeanBuildItemBuildProducer.produce(configurator.done());
}
}
@Record(ExecutionTime.STATIC_INIT)
@BuildStep
void register(List sqlSessionFactoryBuildItems,
BuildProducer syntheticBeanBuildItemBuildProducer,
MyBatisRecorder recorder) {
sqlSessionFactoryBuildItems.forEach(sqlSessionFactoryBuildItem -> {
SyntheticBeanBuildItem.ExtendedBeanConfigurator configurator = SyntheticBeanBuildItem
.configure(SqlSessionFactory.class)
.scope(Singleton.class)
.unremovable()
.supplier(recorder.MyBatisSqlSessionFactorySupplier(sqlSessionFactoryBuildItem.getSqlSessionFactory()));
String dataSourceName = sqlSessionFactoryBuildItem.getDataSourceName();
if (!sqlSessionFactoryBuildItem.isDefaultDataSource()) {
configurator.defaultBean();
configurator.addQualifier().annotation(Named.class).addValue("value", dataSourceName).done();
}
syntheticBeanBuildItemBuildProducer.produce(configurator.done());
});
}
@Record(ExecutionTime.RUNTIME_INIT)
@Consume(SyntheticBeansRuntimeInitBuildItem.class)
@BuildStep
void runInitialSql(List sqlSessionFactoryBuildItems,
MyBatisRuntimeConfig myBatisRuntimeConfig,
MyBatisRecorder recorder) {
sqlSessionFactoryBuildItems.forEach(sqlSessionFactoryBuildItem -> {
MyBatisDataSourceRuntimeConfig dataSourceConfig = myBatisRuntimeConfig.dataSources
.get(sqlSessionFactoryBuildItem.getDataSourceName());
Optional optionalInitialSql;
if (sqlSessionFactoryBuildItem.isDefaultDataSource() || sqlSessionFactoryBuildItems.size() == 1) {
optionalInitialSql = dataSourceConfig != null && dataSourceConfig.initialSql.isPresent()
? dataSourceConfig.initialSql
: myBatisRuntimeConfig.initialSql;
} else {
optionalInitialSql = dataSourceConfig != null ? dataSourceConfig.initialSql : Optional.empty();
}
optionalInitialSql.ifPresent(initialSql -> recorder.runInitialSql(
sqlSessionFactoryBuildItem.getSqlSessionFactory(), initialSql));
});
}
private SqlSessionManagerBuildItem getDefaultSessionManager(List sqlSessionManagerBuildItems) {
return sqlSessionManagerBuildItems.stream()
.filter(SqlSessionManagerBuildItem::isDefaultDataSource)
.findFirst()
.orElse(sqlSessionManagerBuildItems.get(0));
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy