This is an automated email from the ASF dual-hosted git repository.

etudenhoefner pushed a commit to branch main
in repository https://gitbox.apache.org/repos/asf/iceberg.git


The following commit(s) were added to refs/heads/main by this push:
     new 16e84356da Core: Add UUIDv7 generator (#14700)
16e84356da is described below

commit 16e84356dae1975fa04d8c3ecce30a90df18ca9f
Author: Huaxin Gao <[email protected]>
AuthorDate: Tue Dec 2 00:46:37 2025 -0800

    Core: Add UUIDv7 generator (#14700)
---
 .../java/org/apache/iceberg/util/UUIDUtil.java     | 31 ++++++++++++++++++++
 .../java/org/apache/iceberg/util/TestUUIDUtil.java | 34 ++++++++++++++++++++++
 2 files changed, 65 insertions(+)

diff --git a/api/src/main/java/org/apache/iceberg/util/UUIDUtil.java 
b/api/src/main/java/org/apache/iceberg/util/UUIDUtil.java
index b72feec00b..3146a3763c 100644
--- a/api/src/main/java/org/apache/iceberg/util/UUIDUtil.java
+++ b/api/src/main/java/org/apache/iceberg/util/UUIDUtil.java
@@ -20,10 +20,13 @@ package org.apache.iceberg.util;
 
 import java.nio.ByteBuffer;
 import java.nio.ByteOrder;
+import java.security.SecureRandom;
 import java.util.UUID;
 import org.apache.iceberg.relocated.com.google.common.base.Preconditions;
 
 public class UUIDUtil {
+  private static final SecureRandom SECURE_RANDOM = new SecureRandom();
+
   private UUIDUtil() {}
 
   public static UUID convert(byte[] buf) {
@@ -78,4 +81,32 @@ public class UUIDUtil {
     buffer.putLong(8, value.getLeastSignificantBits());
     return buffer;
   }
+
+  /**
+   * Generate a RFC 9562 UUIDv7.
+   *
+   * <p>Layout: - 48-bit Unix epoch milliseconds - 4-bit version (0b0111) - 
12-bit random (rand_a) -
+   * 2-bit variant (RFC 4122, 0b10) - 62-bit random (rand_b)
+   */
+  public static UUID generateUuidV7() {
+    long epochMs = System.currentTimeMillis();
+    Preconditions.checkState(
+        (epochMs >>> 48) == 0, "Invalid timestamp: does not fit within 48 
bits: %s", epochMs);
+
+    // Draw 10 random bytes once: 2 bytes for rand_a (12 bits) and 8 bytes for 
rand_b (62 bits)
+    byte[] randomBytes = new byte[10];
+    SECURE_RANDOM.nextBytes(randomBytes);
+    ByteBuffer rb = ByteBuffer.wrap(randomBytes).order(ByteOrder.BIG_ENDIAN);
+    long randMSB = ((long) rb.getShort()) & 0x0FFFL; // 12 bits
+    long randLSB = rb.getLong() & 0x3FFFFFFFFFFFFFFFL; // 62 bits
+
+    long msb = (epochMs << 16); // place timestamp in the top 48 bits
+    msb |= 0x7000L; // version 7 (UUID bits 48..51)
+    msb |= randMSB; // low 12 bits of MSB
+
+    long lsb = 0x8000000000000000L; // RFC 4122 variant '10'
+    lsb |= randLSB;
+
+    return new UUID(msb, lsb);
+  }
 }
diff --git a/api/src/test/java/org/apache/iceberg/util/TestUUIDUtil.java 
b/api/src/test/java/org/apache/iceberg/util/TestUUIDUtil.java
new file mode 100644
index 0000000000..c5f85c2f20
--- /dev/null
+++ b/api/src/test/java/org/apache/iceberg/util/TestUUIDUtil.java
@@ -0,0 +1,34 @@
+/*
+ * Licensed to the Apache Software Foundation (ASF) under one
+ * or more contributor license agreements.  See the NOTICE file
+ * distributed with this work for additional information
+ * regarding copyright ownership.  The ASF licenses this file
+ * to you under the Apache License, Version 2.0 (the
+ * "License"); you may not use this file except in compliance
+ * with the License.  You may obtain a copy of the License at
+ *
+ *   http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing,
+ * software distributed under the License is distributed on an
+ * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
+ * KIND, either express or implied.  See the License for the
+ * specific language governing permissions and limitations
+ * under the License.
+ */
+package org.apache.iceberg.util;
+
+import static org.assertj.core.api.Assertions.assertThat;
+
+import java.util.UUID;
+import org.junit.jupiter.api.Test;
+
+public class TestUUIDUtil {
+
+  @Test
+  public void uuidV7HasVersionAndVariant() {
+    UUID uuid = UUIDUtil.generateUuidV7();
+    assertThat(uuid.version()).isEqualTo(7);
+    assertThat(uuid.variant()).isEqualTo(2);
+  }
+}

Reply via email to