HHH-18377 - Support for uuid v6 and v7 generated ids

This commit is contained in:
Steve Ebersole 2024-09-23 13:44:48 -05:00
parent d1cb9c3679
commit 2f335cd786
2 changed files with 55 additions and 65 deletions

View File

@ -60,12 +60,14 @@ public @interface UuidGenerator {
* version 6.
* @see UuidVersion6Strategy
*/
@Incubating
VERSION_6,
/**
* Use a time-based generation strategy consistent with RFC 4122
* version 7.
* @see UuidVersion7Strategy
*/
@Incubating
VERSION_7
}

View File

@ -56,98 +56,22 @@ public class UuidGenerator implements BeforeExecutionGenerator {
public UuidGenerator(
org.hibernate.annotations.UuidGenerator config,
MemberDetails memberDetails) {
generator = determineValueGenerator( config, memberDetails );
generator = determineValueGenerator( config, memberDetails.getDeclaringType().getName(), memberDetails.getName() );
final Class<?> memberType = memberDetails.getType().determineRawClass().toJavaClass();
valueTransformer = determineProperTransformer( memberType );
}
private static UuidValueGenerator determineValueGenerator(
org.hibernate.annotations.UuidGenerator config,
MemberDetails memberDetails) {
if ( config != null ) {
if ( config.algorithm() != UuidValueGenerator.class ) {
if ( config.style() != AUTO ) {
throw new MappingException(
String.format(
Locale.ROOT,
"Style [%s] should not be specified with custom UUID value generator : %s.%s",
config.style().name(),
memberDetails.getDeclaringType().getName(),
memberDetails.getName()
)
);
}
return instantiateCustomGenerator( config.algorithm() );
}
if ( config.style() == TIME ) {
return new CustomVersionOneStrategy();
}
else if ( config.style() == VERSION_6 ) {
return UuidVersion6Strategy.INSTANCE;
}
else if ( config.style() == VERSION_7 ) {
return UuidVersion7Strategy.INSTANCE;
}
}
return StandardRandomStrategy.INSTANCE;
}
@Internal
public UuidGenerator(
org.hibernate.annotations.UuidGenerator config,
Member idMember) {
if ( config.algorithm() != UuidValueGenerator.class ) {
if ( config.style() != AUTO ) {
throw new MappingException(
String.format(
Locale.ROOT,
"Style [%s] should not be specified with custom UUID value generator : %s.%s",
config.style().name(),
idMember.getDeclaringClass().getName(),
idMember.getName()
)
);
}
generator = instantiateCustomGenerator( config.algorithm() );
}
else if ( config.style() == TIME ) {
generator = new CustomVersionOneStrategy();
}
else {
generator = StandardRandomStrategy.INSTANCE;
}
generator = determineValueGenerator( config, idMember.getDeclaringClass().getName(), idMember.getName() );
final Class<?> propertyType = getPropertyType( idMember );
this.valueTransformer = determineProperTransformer( propertyType );
}
private static UuidValueGenerator instantiateCustomGenerator(Class<? extends UuidValueGenerator> algorithmClass) {
try {
return algorithmClass.getDeclaredConstructor().newInstance();
}
catch (Exception e) {
throw new HibernateException( "Unable to instantiate " + algorithmClass.getName(), e );
}
}
private ValueTransformer determineProperTransformer(Class<?> propertyType) {
if ( UUID.class.isAssignableFrom( propertyType ) ) {
return UUIDJavaType.PassThroughTransformer.INSTANCE;
}
if ( String.class.isAssignableFrom( propertyType ) ) {
return UUIDJavaType.ToStringTransformer.INSTANCE;
}
if ( byte[].class.isAssignableFrom( propertyType ) ) {
return UUIDJavaType.ToBytesTransformer.INSTANCE;
}
throw new HibernateException( "Unanticipated return type [" + propertyType.getName() + "] for UUID conversion" );
}
public UuidGenerator(
org.hibernate.annotations.UuidGenerator config,
Member member,
@ -177,4 +101,68 @@ public class UuidGenerator implements BeforeExecutionGenerator {
public ValueTransformer getValueTransformer() {
return valueTransformer;
}
private static UuidValueGenerator determineValueGenerator(
org.hibernate.annotations.UuidGenerator config,
String memberDeclaringClassName,
String memberName) {
if ( config != null ) {
// there is an annotation
if ( config.algorithm() != UuidValueGenerator.class ) {
// the annotation specified a custom algorithm
if ( config.style() != AUTO ) {
throw new MappingException(
String.format(
Locale.ROOT,
"Style [%s] should not be specified with custom UUID value generator : %s.%s",
config.style().name(),
memberDeclaringClassName,
memberName
)
);
}
return instantiateCustomGenerator( config.algorithm() );
}
if ( config.style() == TIME ) {
return new CustomVersionOneStrategy();
}
if ( config.style() == VERSION_6 ) {
return UuidVersion6Strategy.INSTANCE;
}
if ( config.style() == VERSION_7 ) {
return UuidVersion7Strategy.INSTANCE;
}
// NOTE : AUTO falls through
}
// Either -
// 1. there is no annotation
// 2. the annotation specified AUTO (with no custom algorithm)
return StandardRandomStrategy.INSTANCE;
}
private static UuidValueGenerator instantiateCustomGenerator(Class<? extends UuidValueGenerator> algorithmClass) {
try {
return algorithmClass.getDeclaredConstructor().newInstance();
}
catch (Exception e) {
throw new HibernateException( "Unable to instantiate " + algorithmClass.getName(), e );
}
}
private ValueTransformer determineProperTransformer(Class<?> propertyType) {
if ( UUID.class.isAssignableFrom( propertyType ) ) {
return UUIDJavaType.PassThroughTransformer.INSTANCE;
}
if ( String.class.isAssignableFrom( propertyType ) ) {
return UUIDJavaType.ToStringTransformer.INSTANCE;
}
if ( byte[].class.isAssignableFrom( propertyType ) ) {
return UUIDJavaType.ToBytesTransformer.INSTANCE;
}
throw new HibernateException( "Unanticipated return type [" + propertyType.getName() + "] for UUID conversion" );
}
}