From c3d7e5f0b52e012cc3f9e9f4f377e7943d2893b3 Mon Sep 17 00:00:00 2001 From: Steve Ebersole Date: Mon, 23 Sep 2024 13:53:27 -0500 Subject: [PATCH] HHH-18377 - Support for uuid v6 and v7 generated ids --- .../id/uuid/UuidVersion6Strategy.java | 34 ++++++------------- .../id/uuid/UuidVersion7Strategy.java | 28 +++++---------- 2 files changed, 20 insertions(+), 42 deletions(-) diff --git a/hibernate-core/src/main/java/org/hibernate/id/uuid/UuidVersion6Strategy.java b/hibernate-core/src/main/java/org/hibernate/id/uuid/UuidVersion6Strategy.java index 1e629d7a5b..847e5f42e8 100644 --- a/hibernate-core/src/main/java/org/hibernate/id/uuid/UuidVersion6Strategy.java +++ b/hibernate-core/src/main/java/org/hibernate/id/uuid/UuidVersion6Strategy.java @@ -14,6 +14,7 @@ import java.util.concurrent.atomic.AtomicLong; import java.util.concurrent.locks.Lock; import java.util.concurrent.locks.ReentrantLock; +import org.hibernate.Internal; import org.hibernate.engine.spi.SharedSessionContractImplementor; import org.hibernate.id.UUIDGenerationStrategy; @@ -30,64 +31,51 @@ import org.hibernate.id.UUIDGenerationStrategy; *
  • 48 bits - pseudorandom data to provide uniqueness.
  • * * + * @apiNote This strategy is field-compatible with Version 1, with the time bits reordered for improved DB locality. + * * @author Cedomir Igaly */ public class UuidVersion6Strategy implements UUIDGenerationStrategy, UuidValueGenerator { - private static final Instant EPOCH_1582; + public static final UuidVersion6Strategy INSTANCE = new UuidVersion6Strategy(); - static { - EPOCH_1582 = LocalDate.of( 1582, 10, 15 ) - .atStartOfDay( ZoneId.of( "UTC" ) ) - .toInstant(); - } + private static final Instant EPOCH_1582 = LocalDate.of( 1582, 10, 15 ) + .atStartOfDay( ZoneId.of( "UTC" ) ) + .toInstant(); private static class Holder { - static final SecureRandom numberGenerator = new SecureRandom(); } - public static final UuidVersion6Strategy INSTANCE = new UuidVersion6Strategy(); - private final Lock lock = new ReentrantLock( true ); - + private final AtomicLong clockSequence = new AtomicLong( 0 ); private long currentTimestamp; - private final AtomicLong clockSequence = new AtomicLong( 0 ); + @Internal public UuidVersion6Strategy() { this( getCurrentTimestamp(), 0 ); } + @Internal public UuidVersion6Strategy(final long currentTimestamp, final long clockSequence) { this.currentTimestamp = currentTimestamp; this.clockSequence.set( clockSequence ); } /** - * A variant 6 + * Version 6 */ @Override public int getGeneratedVersion() { - // UUIDv6 is a field-compatible version of UUIDv1, reordered for improved DB locality return 6; } - /** - * Delegates to {@link #generateUuid} - */ @Override public UUID generateUUID(SharedSessionContractImplementor session) { return generateUuid( session ); } - - /** - * @param session session - * - * @return UUID version 6 - * @see UuidValueGenerator#generateUuid(SharedSessionContractImplementor) - */ @Override public UUID generateUuid(SharedSessionContractImplementor session) { final long currentTimestamp = getCurrentTimestamp(); diff --git a/hibernate-core/src/main/java/org/hibernate/id/uuid/UuidVersion7Strategy.java b/hibernate-core/src/main/java/org/hibernate/id/uuid/UuidVersion7Strategy.java index 6f51c8f48f..59e6ff9041 100644 --- a/hibernate-core/src/main/java/org/hibernate/id/uuid/UuidVersion7Strategy.java +++ b/hibernate-core/src/main/java/org/hibernate/id/uuid/UuidVersion7Strategy.java @@ -12,6 +12,7 @@ import java.util.concurrent.atomic.AtomicLong; import java.util.concurrent.locks.Lock; import java.util.concurrent.locks.ReentrantLock; +import org.hibernate.Internal; import org.hibernate.engine.spi.SharedSessionContractImplementor; import org.hibernate.id.UUIDGenerationStrategy; @@ -33,6 +34,10 @@ import static java.time.temporal.ChronoUnit.MILLIS; *
  • 48 bits - pseudorandom data to provide uniqueness.
  • * * + * @apiNote Version 7 features a time-ordered value field derived from the widely implemented and + * well-known Unix Epoch timestamp source, the number of milliseconds since midnight 1 Jan 1970 UTC, + * leap seconds excluded. + * * @author Cedomir Igaly */ public class UuidVersion7Strategy implements UUIDGenerationStrategy, UuidValueGenerator { @@ -40,52 +45,37 @@ public class UuidVersion7Strategy implements UUIDGenerationStrategy, UuidValueGe public static final UuidVersion7Strategy INSTANCE = new UuidVersion7Strategy(); private static class Holder { - static final SecureRandom numberGenerator = new SecureRandom(); } private final Lock lock = new ReentrantLock( true ); - + private final AtomicLong clockSequence; private Duration currentTimestamp; - private final AtomicLong clockSequence; - + @Internal public UuidVersion7Strategy() { this( getCurrentTimestamp(), 0 ); } + @Internal public UuidVersion7Strategy(final Duration currentTimestamp, final long clockSequence) { this.currentTimestamp = currentTimestamp; this.clockSequence = new AtomicLong( clockSequence ); } /** - * A variant 7 + * Version 7 */ @Override public int getGeneratedVersion() { - /* - * UUIDv7 features a time-ordered value field derived from the widely implemented and well- - * known Unix Epoch timestamp source, the number of milliseconds since midnight 1 Jan 1970 UTC, - * leap seconds excluded. - */ return 7; } - /** - * Delegates to {@link #generateUuid} - */ @Override public UUID generateUUID(SharedSessionContractImplementor session) { return generateUuid( session ); } - /** - * @param session session - * - * @return UUID version 7 - * @see UuidValueGenerator#generateUuid(SharedSessionContractImplementor) - */ @Override public UUID generateUuid(SharedSessionContractImplementor session) { final Duration currentTimestamp = getCurrentTimestamp();