HotGuid 1.0.2

dotnet add package HotGuid --version 1.0.2                
NuGet\Install-Package HotGuid -Version 1.0.2                
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="HotGuid" Version="1.0.2" />                
For projects that support PackageReference, copy this XML node into the project file to reference the package.
paket add HotGuid --version 1.0.2                
#r "nuget: HotGuid, 1.0.2"                
#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 HotGuid as a Cake Addin
#addin nuget:?package=HotGuid&version=1.0.2

// Install HotGuid as a Cake Tool
#tool nuget:?package=HotGuid&version=1.0.2                

HotGuid

.NETRelease to NuGetLicense: MIT

Generates time-ordered ID in Guid (UUID) format, which includes 4-byte shard key.

The feature of an ID generated by the this piece of software is that the shard key can be extracted out of an ID anywhere in the distributed system. This is helpful if you use ID alone to request an entity from a backend, but expect the backend to know which shard to look for the requested entity.

Why time-order is important?

When you massively generate transactional entities (like orders, offers, missions) and insert them to a database the order is important to avoid page splits in ordered storage structures (indexes and tables). The ordered key at least divides in half the cost of an insert versus random key.

If your database is larger than a server memory size the ordered keys saves a lot, because caching is only effective if the hot points in a storage are stable.

Check the articles:

How to use a shard key?

Primary case is the sharding / partitioning in a database. You can use the shard key extracted from ID to query the logical partition / chunk of data by using key in query parameter.

Another case is the request routing at the API gateway level. For instance, having a GraphQL API with node(id: ID!) field or REST API with {id} path parameter, in the backend you can take a region key out from ID and route request to the proper regional cluster.

Two level sharding example

A combination of cases (request routing + database sharding) can be implemented by combining region key and shard key into single shard key value inside ID.

For instance, UN M.49 region codes, which might be a reasonable choice for request routing, are 10 bit long. Having a 4-byte shard key, you can use another 22 bits or the whole 32-bit value as a shard / partition key at the database level. The remaining 22 bits support quite high key cardinality.

sharding

How to use the generator?

Generate a Guid for a given shard key:

Guid id = HotGuidGenerator.Instance.NewGuid(shardKey);

It's your responsibility to use the shard key, which is meaningful and effective for your sharding scheme.

Take shard key out of such Guid:

uint shardKey = id.ExtractShardKey();

Don't extract shard key out from Guid not generated by HotGuidGenerator, you'll get meaningless value.

How ID is built?

The 16 bytes value in Hot Guid consists of:

  • a 4-byte timestamp value measured in seconds since the Unix epoch. Unsigned integer.
  • a 5-byte random value generated once per process. This value is unique to the machine and process.
  • a 3-byte incrementing counter, initialized to a random value.
  • a 4-byte shard key, passed as an argument to HotGuidGenerator.Instance.NewGuid(uint shardKey).

Take a note, in the first 3 parts Hot Guid reproduces MongoDB Object Id, which is 12-bytes.

How probable collisions are?

Like with MongoDB Object Id you will not get the same ID generating less than 16,777,216 (2^24) ID values per second (because the counter is 3-byte long) by a single generator instance.

Take a note, that using several generators in your distributed system (usually you have an instance of the generator per service instance) is quite safe, because Mongo algorithm uses 5-byte random value to reduce probability of collisions between generators.

To have 1% probability of getting collision between generators working each at full capacity you need 23,786,20 generator instances according to the formula:

probability_formula

Check https://en.wikipedia.org/wiki/Birthday_problem, see generalized formula at https://stackoverflow.com/questions/184869/are-guid-collisions-possible.

What timestamp issues might happen?

January 19, 2038?

Nothing happen because we use unsigned 4-byte timestamp value (uint), which is capable to increase up to 4,294,967,295 (2106-02-07 06:28:15 UTC). Covered with tests.

February 7, 2106?

Due to 32-bit overflow the unix timestamp will start from 0.

Due to reusing the same range collision probability for IDs increased since the current timestamp reaches the first seen timestamp (after the year 2158 if you start using Hot Guid in 2022).

But you are still protected from collisions by 5-byte random value and 3-byte incrementing counter which makes it low if you still generate several orders under 16,777,216 (2^24) ID values per second.

Gratitude

Portion of my code is borrowed from SequentialGuid project (https://github.com/buvinghausen/SequentialGuid/blob/master/README.md). I appreciate Brian Buvinghausen work.

Product Compatible and additional computed target framework versions.
.NET net6.0 is compatible.  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 was computed.  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 was computed.  net9.0-android was computed.  net9.0-browser was computed.  net9.0-ios was computed.  net9.0-maccatalyst was computed.  net9.0-macos was computed.  net9.0-tvos was computed.  net9.0-windows was computed. 
Compatible target framework(s)
Included target framework(s) (in package)
Learn more about Target Frameworks and .NET Standard.
  • net6.0

    • No dependencies.

NuGet packages

This package is not used by any NuGet packages.

GitHub repositories

This package is not used by any popular GitHub repositories.

Version Downloads Last updated
1.0.2 276 2/6/2022
1.0.1 231 2/6/2022
1.0.0 243 2/6/2022