/pydomkeys

A python library for domain entity key generation identifiers

Primary LanguagePython

PyDomKeys

 _____                        __          __  __
|     \.-----.--------.---.-.|__|.-----. |  |/  |.-----.--.--.-----.
|  --  |  _  |        |  _  ||  ||     | |     < |  -__|  |  |__ --|
|_____/|_____|__|__|__|___._||__||__|__| |__|\__||_____|___  |_____|
                                                       |_____|

A python library for domain entity key generation identifiers e.g., user, provider, inventory item, etc. based on the rust project.

Overview

Key Size Reduction

Our primary objective is to minimize key sizes when used with key/value stores like Redis or even SQL databases. Domain keys are gauranteed to be unique for a given domain. For example, your set of users may grow into the 10s or 100s of millions. You need a unique identifier for each user. But, you don't need a globally unique identifier because these keys are restricted to just your domain of users. Or a domain of customers, orders, etc.

Rather than use a V4 UUID with 36 characters, it's much better to use a shorter key based on the same technology--a combination of date-time and random numbers. If you restrict a key to a specific domain, e.g., users, customers, businesses, etc, the key can be reduced to a combination of a millisecond time stamp and either a random number, or a counter--especially if the counter range is in the 200K. Add a bit of random number generation and a two character domain prefix, like US, CU, BU.

So, let say you do reach that 10M users goal? If you use a UUID, the keys will consume 360,000,000+ bytes.
Compare that to 160,000,000 and you have cut your memory requirements by more than one half.

Routing for Sharding

The composition of the DomainKey includes two character domain followed by a two character random number to provide automatic shard-routing for 256 shards, or any divisable of 256.
Lets say your user base starts to approach 1M. If you use domain keys with random rounting, it's easy to split this into two, four, or eight shards. With four shards, each database instance now has 250K users--a much more manageable number.

So the idea is to start with a single instance but include the uniformally random routing as the key. When it's time to shard, you simply implement shard-routing logic and you are good to go. This is a life saver for startups that should think about sharding, but don't really need to shard until the time comes.

Domain Routing Key Features...

  • fast, uniformly distributed random number generation based on large range (10^40?) of values
  • time based to the microsecond
  • base62 encoded for size reduction: [0-9][A-Z][a-z]
  • routing key is always 16 characters, 9 date and 7 random including routing key (first two chars)
  • similar to UUID V7 where a timestamp is mixed with random, specifically random + timestamp(micros) + random
  • route-able, not sortable (although sort_by could be implemented for the timestamp portion of the key)
  • short, time based keys from txkey generate 12 character keys.

The goal of the random number generation is speed and uniformity--not security. Domain keys are suitable for identifying elements in a specific domain. Uniformaty is important for routing to insure equally.

When to use

When you...

  • need to create unique identifiers for specified domains e.g. users with the minimum key size that will support billions of entities without collision. You also may want to extract the UTC datetime from the key.
  • need to decode a portion of the key to implement data routing to dozens of destinations or database shards.
  • generate your keys on the rust (or other) application's server side.

When not to use

If you need to generate a key that is truely globally unique, then use v4 UUID. You also are not concerned with key size or being compatible with RFC4122 (UUID standard).

Installation

NOTE: this package is still in development and not available yet

pip install pydomkeys

Use

Examples for time based generator txkey()...

    >>> from pydomkeys.keys import KeyGen
    >>> keygen = KeyGen()
    >>> keygen.txkey()
    '7l0QKqIlDTME'
    >>> key = keygen.txkey()
    >>> assert len(key) == 12
    >>> key2 = keygen.txkey()
    >>> assert key2 > key

Examples for routing key generator route_key()...

    >>> from pydomkeys.keys import KeyGen, DomainRouter
    >>> router = DomainRouter("us")
    >>> keygen = KeyGen(router=router)
    >>> keygen.route_key()
    'usH67l0fKBYkbOc1'
    >>> key = keygen.route_key()
    >>> assert len(key) == 16

Or, use the factory method create to get a new instance...

    >>> from pydomkeys.keys import KeyGen
    >>> keygen = KeyGen.create("US")
    >>> keygen.route_key()
    'USH67l0fKBYkbOc1'
    >>> key = keygen.route_key()
    >>> route_number = int(key[2:4], 16)
    >>> assert route_number < 256

Contributing

Interested in contributing to this project? If so, check our our contributing guidelines document as well as our code of conduct.

Changelog

We are currently in alpha, so lots of changes are possible. But the project is small, so it won't be long until we go to beta.
We will do a minor version bump when we come out of alpha.

We won't have a specific changelog until we are in beta.

References

License

Except where noted (below and/or in individual files), all code in this repository is dual-licensed under either:

PyDomKeys | 2023.09.06