Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

HHH-19247 Add ordinal() to TypeContributor #9865

Closed
wants to merge 1 commit into from
Closed
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -41,4 +41,21 @@ public interface TypeContributor {
* @param serviceRegistry The service registry
*/
void contribute(TypeContributions typeContributions, ServiceRegistry serviceRegistry);

/**
* Determines order in which the contributions will be applied
* (lowest ordinal first).
* <p>
* The range 0-500 is reserved for Hibernate, range 500-1000 for libraries and
* 1000-Integer.MAX_VALUE for user-defined TypeContributors.
* <p>
* Contributions from higher precedence contributors (higher numbers) effectively override
* contributions from lower precedence. E.g. if a contributor with precedence 2000 contributes
* some type, that will override Hibernate's standard type of that name.
*
* @return the ordinal for this TypeContributor
*/
default int ordinal(){
return 1000;
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -87,6 +87,7 @@

import jakarta.persistence.AttributeConverter;

import static java.util.Comparator.comparingInt;
import static org.hibernate.internal.util.config.ConfigurationHelper.getPreferredSqlTypeCodeForArray;
import static org.hibernate.internal.util.config.ConfigurationHelper.getPreferredSqlTypeCodeForDuration;
import static org.hibernate.internal.util.config.ConfigurationHelper.getPreferredSqlTypeCodeForInstant;
Expand Down Expand Up @@ -642,7 +643,7 @@ public void contributeType(CompositeUserType<?> type) {
final JdbcType dialectArrayDescriptor = jdbcTypeRegistry.findDescriptor( SqlTypes.ARRAY );

// add TypeContributor contributed types.
for ( TypeContributor contributor : classLoaderService.loadJavaServices( TypeContributor.class ) ) {
for ( TypeContributor contributor : sortedTypeContributors( classLoaderService ) ) {
contributor.contribute( typeContributions, options.getServiceRegistry() );
}

Expand Down Expand Up @@ -749,6 +750,17 @@ public void contributeType(CompositeUserType<?> type) {
}
}

private static List<TypeContributor> sortedTypeContributors(
ClassLoaderService classLoaderService) {
Collection<TypeContributor> typeContributors = classLoaderService.loadJavaServices( TypeContributor.class );
List<TypeContributor> contributors = new ArrayList<>( typeContributors );
contributors.sort(
comparingInt( TypeContributor::ordinal )
.thenComparing( a -> a.getClass().getCanonicalName() )
);
return contributors;
}

private static void adaptToPreferredSqlTypeCode(
JdbcTypeRegistry jdbcTypeRegistry,
JdbcType dialectUuidDescriptor,
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,77 @@
package org.hibernate.orm.test.type.contributor.usertype;

import org.hibernate.boot.model.TypeContributions;
import org.hibernate.boot.model.TypeContributor;
import org.hibernate.service.ServiceRegistry;

import org.hibernate.testing.orm.junit.BootstrapServiceRegistry;
import org.hibernate.testing.orm.junit.DomainModel;
import org.hibernate.testing.orm.junit.Jira;
import org.hibernate.testing.orm.junit.SessionFactory;
import org.hibernate.testing.orm.junit.SessionFactoryScope;

import org.hibernate.type.descriptor.jdbc.VarcharJdbcType;
import org.hibernate.type.spi.TypeConfiguration;

import org.junit.jupiter.api.Assertions;
import org.junit.jupiter.api.Test;

/**
* Test to assert that service loaded types are processed in ascending ordinal order
*
* @author Steven Barendregt
*/
@DomainModel
@SessionFactory
@BootstrapServiceRegistry(
javaServices = {
@BootstrapServiceRegistry.JavaService(role = TypeContributor.class, impl = TypeContributionOrdinalTest.HigherOrdinalServiceLoadedVarcharTypeContributor.class),
@BootstrapServiceRegistry.JavaService(role = TypeContributor.class, impl = TypeContributionOrdinalTest.ServiceLoadedVarcharTypeContributor.class)
}
)
public class TypeContributionOrdinalTest {

@Test
@Jira(value = "https://hibernate.atlassian.net/issues/HHH-19247")
public void testHigherOrdinalServiceLoadedCustomUserTypeTakesPrecedence(SessionFactoryScope scope) {
final TypeConfiguration typeConfigurations = scope.getSessionFactory()
.getMappingMetamodel()
.getTypeConfiguration();
Assertions.assertInstanceOf(
HigherOrdinalServiceLoadedVarcharTypeContributor.HigherOrdinalExtendedVarcharJdbcType.class,
typeConfigurations.getJdbcTypeRegistry().findDescriptor( 12 )
);
}

public static class ServiceLoadedVarcharTypeContributor implements TypeContributor {

@Override
public void contribute(TypeContributions typeContributions, ServiceRegistry serviceRegistry) {
typeContributions.contributeJdbcType( ExtendedVarcharJdbcType.INSTANCE );
}

public static class ExtendedVarcharJdbcType extends VarcharJdbcType {

public static final ExtendedVarcharJdbcType INSTANCE = new ExtendedVarcharJdbcType();
}

}

public static class HigherOrdinalServiceLoadedVarcharTypeContributor implements TypeContributor {

@Override
public void contribute(TypeContributions typeContributions, ServiceRegistry serviceRegistry) {
typeContributions.contributeJdbcType( HigherOrdinalExtendedVarcharJdbcType.INSTANCE );
}

@Override
public int ordinal() {
return 2000;
}

public static class HigherOrdinalExtendedVarcharJdbcType extends VarcharJdbcType {

public static final HigherOrdinalExtendedVarcharJdbcType INSTANCE = new HigherOrdinalExtendedVarcharJdbcType();
}
}
}
Loading