Skip to content

sdrapkin/SecurityDriven.FastGuid

Repository files navigation

FastGuid NuGet

10 times faster than Guid.NewGuid()

Static APIs

  • Guid FastGuid.NewGuid()

    • Returns a cryptographically random GUID.
    • ~10x (900%) faster than Guid.NewGuid().
  • FastGuid.Fill(Span<byte> data)

    • Fills a span with cryptographically strong random bytes.
    • ~1.5x (50%) to 9x (800%) faster for <512 bytes, otherwise calls RandomNumberGenerator.Fill()
  • Guid FastGuid.NewSqlServerGuid()

    • Returns new Guid optimized for use as a SQL-Server clustered key.
    • Guid structure is [8 random bytes] [8 bytes of SQL-Server-ordered DateTime.UtcNow]
    • Each Guid is sequential accross 100-nanosecond UtcNow precision limits.
    • 64-bit cryptographic randomness adds uniqueness for timestamp collisions and provides reasonable unguessability and protection against online brute-force attacks.
  • Guid FastGuid.NewPostgreSqlGuid()

    • Returns new Guid optimized for use as a PostgreSQL primary key or index.
    • Guid structure is [8 bytes of PostgreSQL-ordered DateTime.UtcNow] [8 random bytes]
    • Each Guid is sequential accross 100-nanosecond UtcNow precision limits.
    • 64-bit cryptographic randomness adds uniqueness for timestamp collisions and provides reasonable unguessability and protection against online brute-force attacks.
  • Helper methods for Guids generated by NewSqlServerGuid():

    • DateTime FastGuid.SqlServer.GetTimestamp(Guid guid)
      • Extracts SqlServer Guid creation timestamp (UTC). Full DateTime.UtcNow precision.
    • Guid FastGuid.SqlServer.MinGuidForTimestamp(DateTime timestampUtc)
    • Guid FastGuid.SqlServer.MaxGuidForTimestamp(DateTime timestampUtc)
      • Return the smallest/largest Guid for a given timestamp (useful for time-based SQL-Server range searches).
  • Helper methods for Guids generated by NewPostgreSqlGuid():

    • DateTime FastGuid.PostgreSql.GetTimestamp(Guid guid)
      • Extracts PostgreSQL Guid creation timestamp (UTC). Full DateTime.UtcNow precision.
    • Guid FastGuid.PostgreSql.MinGuidForTimestamp(DateTime timestampUtc)
    • Guid FastGuid.PostgreSql.MaxGuidForTimestamp(DateTime timestampUtc)
      • Return the smallest/largest Guid for a given timestamp (useful for time-based PostgreSQL range searches).

Usage

Replace all calls to Guid.NewGuid() with FastGuid.NewGuid()

..from this:

Guid guid = Guid.NewGuid(); // your current code

..to this:

// using SecurityDriven;
Guid guid = FastGuid.NewGuid(); // 10x faster
  • Thread-safe
  • 128 bits of cryptographically-strong randomness

Switch from this:

Span<byte> key = stackalloc byte[32];
RandomNumberGenerator.Fill(key); // 145 nanoseconds

..to this:

Span<byte> key = stackalloc byte[32];
FastGuid.Fill(key); // 20 nanoseconds (7x faster)

Important

Guid.CreateVersion7() causes page-fragmentation in SQL-Server and PostgreSQL. DO NOT USE Guid.CreateVersion7() for database Guids. Yes, I have tested it with SQL-Server and PostgreSQL and it causes severe page-fragmentation. Use FastGuid.NewSqlServerGuid() or FastGuid.NewPostgreSqlGuid() instead. Internet advice to use Guid.CreateVersion7() for database Guids is wrong.

Benchmark #1:

public class Bench
{
	[Benchmark(Baseline = true)]
	public void FastGuid_NewGuid() // 12 calls
	{
		FastGuid.NewGuid(); FastGuid.NewGuid(); FastGuid.NewGuid(); FastGuid.NewGuid();
		FastGuid.NewGuid(); FastGuid.NewGuid(); FastGuid.NewGuid(); FastGuid.NewGuid();
		FastGuid.NewGuid(); FastGuid.NewGuid(); FastGuid.NewGuid(); FastGuid.NewGuid();
	}

	[Benchmark]
	public void Guid_NewGuid() // 12 calls
	{
		Guid.NewGuid(); Guid.NewGuid(); Guid.NewGuid(); Guid.NewGuid();
		Guid.NewGuid(); Guid.NewGuid(); Guid.NewGuid(); Guid.NewGuid();
		Guid.NewGuid(); Guid.NewGuid(); Guid.NewGuid(); Guid.NewGuid();
	}
}//class Bench
BenchmarkDotNet=v0.13.2, OS=Windows 10 (10.0.19045.2364)
Intel Core i7-10510U CPU 1.80GHz, 1 CPU, 8 logical and 4 physical cores
.NET SDK=7.0.101
  [Host] : .NET 7.0.1 (7.0.122.56804), X64 RyuJIT AVX2
Method Mean Error StdDev Ratio
FastGuid_NewGuid 116.6 ns 2.26 ns 5.19 ns 1.00
Guid_NewGuid 1,215.9 ns 23.85 ns 45.96 ns 10.39

Benchmark #2:

static Guid Guid_NewGuid() => Guid.NewGuid();
static Guid Guid_CreateVersion7() => Guid.CreateVersion7();
static Guid FastGuid_NewGuid() => SecurityDriven.FastGuid.NewGuid();
static Guid FastGuid_NewSqlServerGuid() => SecurityDriven.FastGuid.NewSqlServerGuid();
static Guid FastGuid_NewPostgreSqlGuid() => SecurityDriven.FastGuid.NewPostgreSqlGuid();
BenchmarkDotNet v0.13.8, Windows 10 (10.0.19045.5371/22H2/2022Update)
Intel Core i7-10510U CPU 1.80GHz, 1 CPU, 8 logical and 4 physical cores
.NET SDK 9.0.102
  [Host] : .NET 9.0.1 (9.0.124.61010), X64 RyuJIT AVX2
Method Mean Error StdDev Ratio
Guid_NewGuid 84.479 ns 1.5090 ns 3.3751 ns 9x
Guid_CreateVersion7 116.798 ns 1.5795 ns 1.8190 ns 12x
FastGuid_NewGuid 9.676 ns 0.2318 ns 0.5278 ns 1x
FastGuid_NewSqlServerGuid 36.710 ns 0.6097 ns 0.8546 ns 4x
FastGuid_NewPostgreSqlGuid 37.292 ns 0.7558 ns 1.0839 ns 4x

image

About

Fast replacement for Guid.NewGuid().

Topics

Resources

License

Stars

Watchers

Forks

Packages

No packages published

Languages