This library provides base classes and tooling to easily and safely create immutable typed Value Object IDs (VO-IDs) internally backed by a UUIDv7 or bigint.
Contributions for serialization libraries, other ORMs, alternative JPA implementations or anything else are welcome.
Blog: Specialized Value Objects for entity identifiers
For seamless type support in Hibernate ORM, you should pick one of the following variants:
| Hibernate Version | Artifact | 
|---|---|
| 6.6, 6.5, 6.4, and 6.3 | org.framefork:typed-ids-hibernate-63 | 
| 6.2 | org.framefork:typed-ids-hibernate-62 | 
| 6.1 | org.framefork:typed-ids-hibernate-61 | 
Find the latest version in this project's GitHub releases or on Maven Central.
If you want just the plain base classes without ORM support, you can install just the org.framefork:typed-ids.
Minimum supported Java is 17.
The library tries to make sure your data is stored with the types best fit for the job.
In some cases, that means changing the default DDL that Hibernate uses for the SqlTypes.UUID jdbc type.
| Database | Database Type for UUID | 
|---|---|
| PostgreSQL | uuid | 
| MySQL | binary(16) | 
| MariaDB 10.7+ | uuid | 
| MariaDB before 10.7 | binary(16) | 
| other | whatever Hibernate uses as the dialect default | 
The library only sets the type if there is no JDBC type for SqlTypes.UUID already set,
which means that if you want to use something different you should be able to do so using a custom org.hibernate.boot.model.TypeContributor.
The library explicitly supports AUTO, IDENTITY and SEQUENCE strategies for generating bigint identifiers via the @GeneratedValue annotation for the VO-IDs.
This pattern is supported only for ObjectBigIntId.
One of the primary goals of this library is to enable generated typed IDs in application code - specifically in entity constructors.
Being able to generate identifiers in app code solves many problems around application design and architecture by getting rid of the dependency of entity on the database. The classic approach is to let the database generate the identifiers, which is perfectly fine if you prefer that, but it breaks entity state because until you persist them they're invalid and incomplete. But when you generate the ID at construction time, the entity is valid from the first moment. The only way to do this reliably is to generate random identifiers so that you don't get conflicts when persisting the entities, but using perfectly random values has its problems - see UUID as a primary key for a solution.
This pattern is supported for both ObjectUuid and ObjectBigIntId.
This library supports several libraries for generating the IDs in the JVM but does not pull them in, instead it expects you to pick one and add it yourself.
- UUIDs with com.fasterxml.uuid:java-uuid-generator(see project homepage)
- BigInts/longs with io.hypersistence:hypersistence-tsid(see project homepage)
If you want to use a different library, the $Generators.setFactory() extension point should hopefully be self-explanatory.
The ObjectUuid.randomUUID() generates UUIDv7 instead of Java's default UUIDv4.
The UUIDv4 is not well suited to be used in indexes and primary keys due to performance reasons.
The UUIDv7, while still larger than plain long, does not suffer from the performance problems that UUIDv4 has, and can be safely used for primary keys.
In case you don't like the default generator, you can opt to replace it using ObjectUuid.Generators.setFactory(UuidGenerator.Factory).
It might come in handy in tests where you might want to use a deterministic generator.
Some additional resources:
- Illustrating Primary Key models in InnoDB and their impact on disk usage - Percona Blog
- GUID/UUID Performance Breakthrough - Rick James
- MySQL InnoDB Primary Key Choice: GUID/UUID vs Integer Insert Performance - KCCoder
- Unreasonable Defaults: Primary Key as Clustering Key - Use The Index, Luke
- SQL server full-text index and its stop words - Jiangong Sun
- MySQL UUIDs – Bad For Performance - Percona Blog
- Choose the right primary key to save a large amount of disk I/O - Too Many Afterthoughts
- UUID vs Bigint Battle! - Scaling Postgres 302
This base type is designed to wrap a native UUID, and allows you to expose any utility functions you may need. The following snippet is the standard boilerplate, but you may opt to skip some of the methods, or add a few custom ones.
public record User(Id id)
{
    public static final class Id extends ObjectUuid<Id>
    {
        private Id(final UUID inner)
        {
            super(inner);
        }
        public static Id random()
        {
            return ObjectUuid.randomUUID(Id::new);
        }
        public static Id from(final String value)
        {
            return ObjectUuid.fromString(Id::new, value);
        }
        public static Id from(final UUID value)
        {
            return ObjectUuid.fromUuid(Id::new, value);
        }
    }
}
// ...
var user = new User(Id.random());With Kotlin, the standard boilerplate should look like the following snippet
data class User(id: Id) {
    class Id private constructor(id: UUID) : ObjectUuid<Id>(id) {
        companion object {
            fun random() = randomUUID(::Id)
            fun from(value: String) = fromString(::Id, value)
            fun from(value: UUID) = fromUuid(::Id, value)
        }
    }
}This base type is designed to wrap a native long, and allows you to expose any utility functions you may need.
The following snippet is the standard boilerplate, but you may opt to skip some of the methods, or add a few custom ones.
public record User(Id id)
{
    public static final class Id extends ObjectBigIntId<Id>
    {
        private Id(final long inner)
        {
            super(inner);
        }
        public static Id random()
        {
            return ObjectBigIntId.randomBigInt(Id::new);
        }
        public static Id from(final String value)
        {
            return ObjectBigIntId.fromString(Id::new, value);
        }
        public static Id from(final long value)
        {
            return ObjectBigIntId.fromLong(Id::new, value);
        }
    }
}
// ...
var user = new User(Id.random());With Kotlin, the standard boilerplate should look like the following snippet
data class User(id: Id) {
    class Id private constructor(id: UUID) : ObjectBigIntId<Id>(id) {
        companion object {
            fun random() = randomBigInt(::Id)
            fun from(value: String) = fromString(::Id, value)
            fun from(value: Long) = fromLong(::Id, value)
        }
    }
}This library provides a mechanism to index your ID classes at compile time, which is useful for pleasant integrations with various frameworks.
Set up the org.framefork:typed-ids-index-java-classes-processor as an annotation processor.
It's based on org.atteo.classindex:classindex, and when executed,
it writes (somewhere in your build output directory) to META-INF/services/org.framefork.typedIds.uuid.ObjectUuid or META-INF/services/org.framefork.typedIds.bigint.ObjectBigIntId,
which can be later read by the standard java.util.ServiceLoader mechanism.
With Gradle, register the processor like this:
dependencies {
    annotationProcessor("org.framefork:typed-ids-index-java-classes-processor")
    testAnnotationProcessor("org.framefork:typed-ids-index-java-classes-processor")
}With Maven, you can register it as an optional dependency if annotation processors discovery works in your project
<dependencies>
    <dependency>
        <groupId>org.framefork</groupId>
        <artifactId>typed-ids-index-java-classes-processor</artifactId>
        <optional>true</optional>
    </dependency>
</dependencies>Or if you want to be safe, you can register the processor explicitly
<plugin>
    <groupId>org.apache.maven.plugins</groupId>
    <artifactId>maven-compiler-plugin</artifactId>
    <configuration>
        <annotationProcessorPaths>
            <path>
                <groupId>org.framefork</groupId>
                <artifactId>typed-ids-index-java-classes-processor</artifactId>
            </path>
        </annotationProcessorPaths>
    </configuration>
</plugin>With Kotlin, you have to register the indexer as KAPT processor:
dependencies {
    kapt("org.framefork:typed-ids-index-java-classes-processor") // instead of annotationProcessor(...)
}Without the Typed IDs indexing at compile time, the field has to be annotated like this (the @Type(...) is the important part):
@Entity
public class User
{
    @jakarta.persistence.Id
    @Column(nullable = false)
    @Type(ObjectUuidType.class)
    private Id id;
    // ...
    public static final class Id extends ObjectUuid<Id>
    {
        // ...But with the index, the ObjectUuidTypesContributor reads it, and registers the types automatically when Hibernate ORM is initialized.
With the classes indexed, the system will know that the User.Id should be handled by ObjectUuidType and the @Type(...) can be dropped.
This also simplifies usage on every other place, where Hibernate might need to resolve a type for the Id instance, like queries.
This library provides ObjectBigIntIdJacksonModule and ObjectUuidJacksonModule, which can be registered automatically via the standard java.util.ServiceLoader mechanism, or explicitly.
Please note that in Spring, the instance of Jackson's ObjectMapper used for (de)serializing requests and responses of controllers by default ignores the modules provided via ServiceLoader,
so to make it work, you have to either register the modules as beans, or add the following customizer:
@Bean
public Jackson2ObjectMapperBuilderCustomizer enableServiceLoaderModules()
{
    return builder -> builder.findModulesViaServiceLoader(true);
}This library provides ObjectBigIntIdTypeAdapterFactory and ObjectUuidTypeAdapterFactory, which can be registered automatically via the standard java.util.ServiceLoader mechanism, or explicitly.
This library supports two mechanism for the standard Kotlin Serialization.
This mechanism requires explicit setup for each ID class
@Serializable(with = UserId.Serializer::class)
class UserId private constructor(id: UUID) : ObjectUuid<UserId>(id) {
    // standard boilerplate ...
    object Serializer : ObjectUuidSerializer<UserId>(::UserId)
}but in return every time you use it, it just works
@Serializable
data class UserDto(val id: UserId)With contextual, the standard ID boiler can be used, but you have to register the module
val json = Json {
    serializersModule = ObjectUuidKotlinxSerializationModule.fromIndex
}and then mark the type as @Contextual on every usage
@Serializable
data class UserDto(@Contextual val id: UserId)This library provides support for OpenAPI schema generation, so that you don't have to annotate the types with @Schema manually (you still can, if you want to).
The org.framefork:typed-ids-openapi-swagger-jakarta artifact
provides a TypedIdsModelConverter, which should be automatically picked up by the standard Swagger v3 Jakarta implementation,
because it's exposed via the standard java.util.ServiceLoader mechanism.
There is a single configurable property - idsAsRef, which is the equivalent of enumsAsRef in the standard Swagger v3 implementation.
You can set override it via TypedIdsModelConverter.idsAsRef, or using a system property framefork.typed-ids.openapi.as-ref.
Providing a non-jakarta variant would be straightforward, but given that javax has been deprecated for a long time, I decided to not bother with it.
The org.framefork:typed-ids-openapi-springdoc artifact builds on the Swagger v3 Jakarta integration, and registers it as a standard Spring bean.
With SpringDoc, you shouldn't configure the converter explicitly,
and instead you should use the standard spring configuration to set it via the framefork.typed-ids.openapi.as-ref property in config/ENV/etc.
You may notice that the artifact depends on a quite old version of the SpringDoc. That is because I needed compatibility with Spring Boot 3.0.x. However, it works seamlessly with newer Spring Boot versions.
You may want to check the working example in testing/testing-typed-ids-springdoc-openapi, with a recent Spring Boot version, and also with a working OpenApi spec generation, and TypeScript client generation.
To learn more you can explore the testing/ directory of this library,
for example the testing/testing-typed-ids-hibernate-66-indexed
has several working use-cases written in the form of tests that you can study for inspiration.