Medo.Uuid7.EntityFrameworkCore 3.0.0-beta1

Prefix Reserved
This is a prerelease version of Medo.Uuid7.EntityFrameworkCore.
There is a newer prerelease version of this package available.
See the version list below for details.
dotnet add package Medo.Uuid7.EntityFrameworkCore --version 3.0.0-beta1                
NuGet\Install-Package Medo.Uuid7.EntityFrameworkCore -Version 3.0.0-beta1                
This command is intended to be used within the Package Manager Console in Visual Studio, as it uses the NuGet module's version of Install-Package.
<PackageReference Include="Medo.Uuid7.EntityFrameworkCore" Version="3.0.0-beta1" />                
For projects that support PackageReference, copy this XML node into the project file to reference the package.
paket add Medo.Uuid7.EntityFrameworkCore --version 3.0.0-beta1                
#r "nuget: Medo.Uuid7.EntityFrameworkCore, 3.0.0-beta1"                
#r directive can be used in F# Interactive and Polyglot Notebooks. Copy this into the interactive tool or source code of the script to reference the package.
// Install Medo.Uuid7.EntityFrameworkCore as a Cake Addin
#addin nuget:?package=Medo.Uuid7.EntityFrameworkCore&version=3.0.0-beta1&prerelease

// Install Medo.Uuid7.EntityFrameworkCore as a Cake Tool
#tool nuget:?package=Medo.Uuid7.EntityFrameworkCore&version=3.0.0-beta1&prerelease                

Medo.Uuid7.EntityFrameworkCore

The UUID7 library is an implementation of the UUID version 7 (time-ordered) and version 4 (fully random) as defined in the RFC 9562. It offers improved entropy characteristics compared to versions 1 or 6 of the UUID standard. The inherent monotonicity of UUID version 7 makes it an excellent choice for utilization as a binary database key.

Features:

  • Time-ordered value field: UUID7 utilizes the widely implemented Unix Epoch timestamp source to generate a time-ordered value field. This enables easy sorting and indexing of resources based on their creation time.
  • Enhanced entropy characteristics: UUID7 provides improved entropy characteristics over UUID versions 1 or 6. The inclusion of the timestamp ensures a high level of uniqueness, minimizing the chances of collisions across different systems or instances.
  • Multiple string representations: In addition to the standard UUID string formatting, library also offers ID22 and ID25 string conversions.
  • Wide compatibility: Support for .NET Standard 2.0 makes this library compatible with .NET Framework 4.6.1 or higher.
  • High performance: Speed comparable to the optimized built-in GUID generator in both single-threaded and multi-threaded scenarios under Windows and Linux.
  • Hardware acceleration: Vector128 support for Equals method.
  • Microsoft SQL Server support (NewMsSqlUniqueIdentifier()).
  • Support for UUID version 4 (fully random UUID)
  • Conversion from and to System.Guid
  • .NET 8 AOT support
  • Also available as .NET Core library.

You can find packaged library at NuGet.

Usage

To generate a new database-friendly UUID v7, simply call NewUuid7 method:

using System;
using Medo;

var uuid = Uuid7.NewUuid7();  // or 'Uuid7.NewGuid()'
Console.WriteLine($"UUID : {uuid}");

Alternatively, if a fully random UUID v4 is desired, call NewUuid4 method:

using System;
using Medo;

var uuid = Uuid7.NewUuid4();
Console.WriteLine($"UUID : {uuid}");

If higher multi-thread performance is needed and per-thread seqencing is sufficient, you can instantiate UUID directly:

using System;
using Medo;

var uuid = new Uuid7();
Console.WriteLine($"UUID : {uuid}");

Converting to Guid

Converting to and from System.Guid is a complicated story. There are two ways it can be done. One is by preserving binary equivalency and the other is by preserving textual respresentations. Since Microsoft's Guid implementation in .NET 9.0 keeps Microsoft's weird endianess behavior even with UUIDv7, this is selected as default

Please note that, prior to 3.0, priority was given to maintaining binary representation. This is a breaking change for older versions. If you are upgrading from older version, use ToGuid(bigEndian: true) overload.

If we want to preserve textual representation and follow behavior as defined in .NET 9, we can useToGuid() function or its ToGuid(bigEndian: false) overload as this one takes internal Guid endianess into account.

using Medo;

var uuid = Uuid7.NewUuid7();
Console.WriteLine($"{uuid}");

var guid = uuid.ToGuid();
Console.WriteLine($"{guid}");

Textual output in this case would be equal but at the cost of raw binary bytes differing.

01904d33-d262-7531-b71c-05555c63df91
01904d33-d262-7531-b71c-05555c63df91

On other hand, code below will retain binary compatibility during conversion (if running on LE system).

using Medo;

var uuid = Uuid7.NewUuid7();
Console.WriteLine($"{uuid}");

var guid = uuid.ToGuid(bigEndian: true);
Console.WriteLine($"{guid}");

Note this means that textual respresentations look different since Microsoft prints logically numeric Guid elements in little-endian order instead of arguably more common big-endian order.

01904d33-d262-7531-b71c-05555c63df91
334d9001-62d2-3175-b71c-05555c63df91

I view this as a damn-if-you-do-damn-if-you-don't scenario and prior to version 3.0, default was to keep them equivalent in binary. Since .NET 9 decided to go other way, the default was changed.

Entity Framework Core

Converters

In order to use the Uuid7 type in your models, you must configure your DbContext in Entity Framework Core.

You can choose from various converters from the Medo.Uuid7.EntityFrameworkCore package. These are used for your EF database provider to read from or write to a property's associated DB column type:

  • Uuid7ToGuidConverter
  • Uuid7ToStringConverter
  • Uuid7ToBytesConverter
  • Uuid7ToId22Converter
  • Uuid7ToId25Converter

To set a converter globally for all properties of type Uuid7, override the ConfigureConventions method:

protected override void ConfigureConventions(ModelConfigurationBuilder configurationBuilder) {
    configurationBuilder.Properties<Uuid7>().HaveConversion<Uuid7ToGuidConverter>();
}

Alternatively, if you prefer to do it per entity, override the OnModelCreating method:

protected override void OnModelCreating(ModelBuilder modelBuilder) {
    modelBuilder.Entity<User>().Property(x => x.Id).HasConversion<Uuid7ToGuidConverter>();
}

Configuration

Disable RNG Buffering

Buffering of random numbers significantly increases performance at the cost of less frequent but bigger requests toward random number generator. If buffering is not desired (e.g. only a small count of UUIDs is needed), you can disable it using UUID7_NO_RANDOM_BUFFER preprocessor constant.

<PropertyGroup>
    <DefineConstants>UUID7_NO_RANDOM_BUFFER</DefineConstants>
</PropertyGroup>

Note that this will decrease performance significantly.

UUID Format

The format of UUIDv7 is as specified below.

 0                   1                   2                   3
 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|                           unix_ts_ms                          |
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|          unix_ts_ms           |  ver  |       rand_a          |
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|var|                        rand_b                             |
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|                            rand_b                             |
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+

unix_tx_ms: 48 bit big-endian unsigned number of Unix epoch timestamp.

ver: 4 bit UUIDv7 version. Always 0111.

rand_a: 12 bits of pseudo-random data.

var: 2 bit variant. Always 10.

rand_b: Additional 62 bits of pseudo-random data.

Implementation

As monotonicity is important for UUID version 7 generation, this implementation implements most of monotonic random counter recommendations.

Implementation uses randomly seeded 26 bit monotonic counter (25 random bits + 1 rollover guard bit) with a 4-bit increment.

Counter uses 12-bits from rand_a field and it "steals" 14 bits from rand_b field. Counter will have its 25 bits fully randomized each millisecond tick. Within the same millisecond tick, counter will be randomly increased using 4 bit increment.

In the case of multithreaded use, the counter seed is different for each thread.

In the worst case, this implementation guarantees at least 2^21 monotonically increasing UUIDs per millisecond. Up to 2^23 monotonically increasing UUID values per millisecond can be expected on average. Monotonic increase for each generated value is guaranteed on per thread basis.

The last 48 bits are filled with random data that is different for each generated UUID.

As each UUID uses 48 random bits in addition to 25 random bits from the seeded counter, this means we have at least 73 bits of entropy (without taking 48-bit timestamp into account).

With those implementation details in mind, the final layout is defined as below.

 0                   1                   2                   3
 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|                           unix_ts_ms                          |
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|          unix_ts_ms           |  ver  |        counter        |
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|var|          counter          |            random             |
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|                            random                             |
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+

unix_tx_ms: 48 bit big-endian unsigned number of Unix epoch timestamp.

ver: 4 bit UUIDv7 version. Always 0111.

var: 2 bit variant. Always 10.

counter: 26 bit big-endian unsigned counter.

random: 48 bits of random data.

Textual Representation

While this UUID should be handled and stored in its binary 128 bit form, it's often useful to provide a textual representation.

UUID Format

This is a standard hexadecimal representation of UUID with dashes separating various components. Please note that this component separation doesn't necessarily correlate with any internal fields.

Example:

0185aee1-4413-7023-9109-bde493efe31d

Id25

Alternative string representation is Id25 (Base-35), courtesy of stevesimmons. While I have seen similar encodings used before, his implementation is the first one I saw being used on UUIDs. Since it uses only numbers and lowercase characters, it actually retains lexicographical sorting property the default UUID text format has.

UUID will always fit in 25 characters.

Example:

0672016s27hx3fjxmn5ic1hzq

Id22

If more compact string representation is needed, one can use Id22 (Base-58) encoding. This is the same encoding Bitcoin uses for its keys.

UUID will always fit in 22 characters.

Example:

1BuKkq6yWzmN2fCaHBjCRr
Product Compatible and additional computed target framework versions.
.NET net5.0 was computed.  net5.0-windows was computed.  net6.0 was computed.  net6.0-android was computed.  net6.0-ios was computed.  net6.0-maccatalyst was computed.  net6.0-macos was computed.  net6.0-tvos was computed.  net6.0-windows was computed.  net7.0 was computed.  net7.0-android was computed.  net7.0-ios was computed.  net7.0-maccatalyst was computed.  net7.0-macos was computed.  net7.0-tvos was computed.  net7.0-windows was computed.  net8.0 is compatible.  net8.0-android was computed.  net8.0-browser was computed.  net8.0-ios was computed.  net8.0-maccatalyst was computed.  net8.0-macos was computed.  net8.0-tvos was computed.  net8.0-windows was computed.  net9.0 is compatible. 
.NET Core netcoreapp2.0 was computed.  netcoreapp2.1 was computed.  netcoreapp2.2 was computed.  netcoreapp3.0 was computed.  netcoreapp3.1 was computed. 
.NET Standard netstandard2.0 is compatible.  netstandard2.1 was computed. 
.NET Framework net461 was computed.  net462 was computed.  net463 was computed.  net47 was computed.  net471 was computed.  net472 was computed.  net48 was computed.  net481 was computed. 
MonoAndroid monoandroid was computed. 
MonoMac monomac was computed. 
MonoTouch monotouch was computed. 
Tizen tizen40 was computed.  tizen60 was computed. 
Xamarin.iOS xamarinios was computed. 
Xamarin.Mac xamarinmac was computed. 
Xamarin.TVOS xamarintvos was computed. 
Xamarin.WatchOS xamarinwatchos was computed. 
Compatible target framework(s)
Included target framework(s) (in package)
Learn more about Target Frameworks and .NET Standard.

NuGet packages (1)

Showing the top 1 NuGet packages that depend on Medo.Uuid7.EntityFrameworkCore:

Package Downloads
Platter.DomainCore

DomainCore

GitHub repositories

This package is not used by any popular GitHub repositories.

Version Downloads Last updated
3.0.0-beta3 26 11/23/2024
3.0.0-beta2 35 11/22/2024
3.0.0-beta1 30 11/19/2024
2.1.1 120 11/9/2024
2.1.0 91 11/6/2024
2.0.0 7,781 6/30/2024
1.9.1 6,719 1/21/2024
1.9.0 165 1/7/2024
1.8.2 173 11/25/2023
1.8.1 131 11/15/2023
1.8.0 120 11/15/2023
1.7.0 241 7/7/2023
1.6.1 168 7/5/2023
1.6.0 170 7/4/2023

$'CHANGELOG\n\n[3.0.0]\n\n- .NET 9 suppport\n- **BREAKING CHANGE**: Guid conversion is done in native endianess (was always big endian before)\n  - this was done to match Guid.CreateVersion7() behavior by default\n  - old behavior possible by using ToGuid(bigEndian:true)\n- **BREAKING CHANGE**: all overloads that used matchGuidEndianess were changed to bigEndian parameter.\n  - this was done to match most of .NET 9 overloads\n  - it is possible to get the old behavior by using opposite boolean value\n- removed obsoleted functions\n- removed official .NET 6 support\n\n\n[2.1.1]\n\n- Fixed handling of non-initialized struct\n\n\n[2.1.0]\n\n- Added timestamp support for FillMsSqlUniqueIdentifier\n- Removed explicit .NET 7 support\n- Fixed handling of non-initialized struct\n\n\n[2.0.0]\n\n- Compatible with the final [RFC 9562](https://datatracker.ietf.org/doc/rfc9562/)\n- Fixed compatibility with Microsoft SQL Server UniqueIdentifier\n- Added support for Guid endianness when converting from and to System.Guid\n- Added support for creating Uuid7 with custom timestamp\n\n\n[1.9.1] (2024-01-21)\n\n- More parsing performance improvement - courtesy of [Joel Mueller](https://github.com/jtmueller)\n\n\n[1.9.0] (2024-01-06)\n\n- Parsing performance improvement - courtesy of [Joel Mueller](https://github.com/jtmueller)\n- Improved ToString performance - courtesy of [Joel Mueller](https://github.com/jtmueller)\n\n\n[1.8.2] (2023-11-24)\n\n- Added ToDateTime and ToDateTimeOffset methods\n\n\n[1.8.1] (2023-11-15)\n\n- Fixed .NET 8 AOT compilation support\n\n\n[1.8.0] (2023-11-15)\n\n- Added .NET 8 AOT compilation support\n- ToString optimizations\n\n\n[1.7.0] (2023-07-06)\n\n- Improved performance\n- Added ISpanFormattable interface\n- Added ISpanParsable interface\n- Added format specifiers for Id25 (\'5\') and Id22 (\'2\')\n- Added Fill method for .NET Standard 2.0 targets\n\n\n[1.6.1] (2023-07-04)\n\n- Static methods use locks to allow for sequencing even when called from\n  different threads (e.g. usage with async)\n\n\n[1.6.0] (2023-07-02)\n\n- Added Entity Framework Core version (separate NuGet package)\n- Added NewGuid() method\n- Added MS SQL Guid format support (NewGuidMsSql and ToGuidMsSql)\n- Added implicit conversion from and to Guid\n\n\n[1.5.0] (2023-06-29)\n\n- Added TryWriteBytes method\n- Updated GetHashCode method\n- HW acceleration for Equals method\n\n\n[1.4.0] (2023-06-19)\n\n- Major optimizations (buffered RandomNumberGenerator calls)\n- Changed NuGet package name to Medo.Uuid7\n- Obsoleted NuGet package Uuid7\n\n\n[1.3.5] (2023-06-19)\n\n- Updated readme\n\n\n[1.3.4] (2023-06-12)\n\n- Minor optimizations\n- Updated readme\n\n\n[1.3.3] (2023-06-07)\n\n- Minor optimizations\n- Added fully random v4 UUID support\n\n\n[1.3.2] (2023-05-17)\n\n- .NET Standard 2.0 support\n- ToString() performance optimizations\n\n\n[1.3.1] (2023-05-16)\n\n- Performance optimizations\n\n\n[1.3.0] (2023-04-30)\n\n- Added IFormattable interface\n- Fixed EF NullReferenceException when CompareArrays gets a null input\n\n\n[1.2.0] (2023-04-12)\n\n- Timestamps are monotonically increasing even if time goes backward\n\n\n[1.1.1] (2023-01-14)\n\n- Fixed monotonicity bug\n\n\n[1.1.0] (2023-01-14)\n\n- Using random increment (was tick based before)\n- Performance improvements\n\n\n[1.0.2] (2023-01-13)\n\n- Fixed docs\n\n\n[1.0.1] (2023-01-13)\n\n- Added readme\n\n\n[1.0.0] (2023-01-13)\n\n- First release\n\n\n\n[unreleased]: https://github.com/medo64/Medo.uuid7\n[1.9.1]: Medo.Uuid7/1.9.1\n[1.9.0]: Medo.Uuid7/1.9.0\n[1.8.2]: Medo.Uuid7/1.8.2\n[1.8.1]: Medo.Uuid7/1.8.1\n[1.8.0]: Medo.Uuid7/1.8.0\n[1.7.0]: Medo.Uuid7/1.7.0\n[1.6.1]: Medo.Uuid7/1.6.1\n[1.6.0]: Medo.Uuid7/1.6.0\n[1.5.0]: Medo.Uuid7/1.5.0\n[1.4.0]: Medo.Uuid7/1.4.0\n[1.3.5]: Uuid7/1.3.5\n[1.3.4]: Uuid7/1.3.4\n[1.3.3]: Uuid7/1.3.3\n[1.3.2]: Uuid7/1.3.2\n[1.3.1]: Uuid7/1.3.1\n[1.3.0]: Uuid7/1.3.0\n[1.2.0]: Uuid7/1.2.0\n[1.1.1]: Uuid7/1.1.1\n[1.1.0]: Uuid7/1.1.0\n[1.0.2]: Uuid7/1.0.2\n[1.0.1]: Uuid7/1.0.1\n[1.0.0]: Uuid7/1.0.0'