What are UUIDs?

January 14, 2023

2,105 words

This article is part of a series: Explaining UUIDs

Post contents

Oftentimes, in computer programming, you'll find yourself needing a way to give a unique identifier to a digital asset. For example, items such as books in your "to read" digital bookshelf, computers on your network, or rows in your database need a quick and easy way to access them. This is typically done with a distinct ID for each item.

If you've done much research on this problem, you'll likely have heard of a "universally unique identifier" (UUID) or "globally unique identifier" (GUID); a way to generate IDs for these assets quickly. After all, UUID usage is wide-reaching in the software industry.

While it's good to recognize popular technologies, a few questions remain:

  • What is a UUID?
  • What are the various versions of UUID?
  • Why are they so widely utilized?

What is a UUID?

Broadly speaking, a UUID is a numerical value of 128 bits that can be used to provide an identification number for a resource.

While there are multiple different kinds of UUIDs, which we'll touch on shortly, all UUID formats follow a few general ideas.

First, UUIDs are, for all intents and purposes, unique. While that might seem obvious, here's the part that isn't: The guarantee that a number is unique does not require previous knowledge of other generated UUIDs.

This differs from a simple counter that counts up from 0, where you would need to look up the previous number stored to generate a new one.

Second, while it's not literally impossible to generate two UUIDs with the same value, it's generally safe to assume that it's nearly impossible with the proper inputs.

Finally, all UUIDs are formatted similarly. The numerical value of the UUID is encoded into a string of hexadecimal numbers and dashes (-) when displayed as a string. An example UUIDv4 might be formatted to look something like this:

a5abec44-7ce0-437c-972a-cf451b4fde2b

Two of these characters include information about which kind of UUID it is:

xxxxxxxx-xxxx-Mxxx-Nxxx-xxxxxxxxxxxx

Here, M is the UUID version number, while N is the variant of the UUID version. Think of this as information encoded within the UUID about the UUID generation process that yielded the output value.

While we'll explain what the UUID "version" is in the next section, let's quickly explain what the "variant" means:

  • If the variant is 0 through 7, the UUID is backward compatible with very old computer systems from the 1980s.
  • If the variant is 8 through b, it means the UUID is part of the "RFC 4122" standard
  • If the variant is c or d, it means the UUID is compatible with early Windows systems
  • Variants e and f are reserved for future UUID versions

This means that we can take the previous UUID:

a5abec44-7ce0-437c-972a-cf451b4fde2b

And determine that this is a UUID with the version of 4 and the variant of 9.

Speaking of "UUID version numbers", what are those? I know we outlined earlier that there were different ways of generating a UUID; what are they?

I'm glad you asked.

What are the different types of UUID?

A "UUID version" outlines which type of UUID you're generating; each version of UUID has a different generation mechanism and, therefore, different use cases.

At the time of writing, there are 5 different types of UUIDs that are part of the official UUID specification:

  • UUIDv1
    • A machine's network card information + a timestamp
  • UUIDv2
  • UUIDv3
    • Encode a string using MD5
  • UUIDv4
    • Random UUID with effectively zero chance of producing the same number twice
  • UUIDv5
    • UUIDv3 but more secure (uses SHA-1)

Track Network Systems Using UUIDv1

As mentioned previously, a UUID is composed of:

  • A timestamp
  • A UUID version of 1
  • A UUID variant value
  • A random number called a "Clock Sequence"
  • A unique ID number of a computer's network card - called a "MAC address"

This data is then converted to raw numerical values and encoded into UUID with the following format:

A UUID broken down into "Low Time", a dash, "Mid Time", a dash, "Version", "High Time", a dash, "Variant", "Clock", a dash, and finally a "MAC Address". An example UUIDv1 might be "4e2b4d4c-92e8-11ed-86a8-3fdb0085247e"
Pros of UUIDv1
  • Low likelihood of collision: While Mac addresses are not truly unique, thanks to vendor re-use, the combination of timestamp and Mac address makes it extremely unlikely to duplicate the same UUID in multiple generations on the same network system. Furthermore, introducing a clock sequence adds another layer of protection against UUIDv1 collision.
Cons of UUIDv1
  • Anonymity: Because we're encoding a system's Mac address into the UUIDv1, it cannot be used as an anonymous value - since it links the UUID to a system that can be identified by its network card metadata.

  • Easy to guess the sequence: Likewise, let's say you want to use a UUID to generate a reset password link to the user:

    https://example.com/reset/ef5dc6d0-93ca-11ed-acdd-6fb6f98f3ada
    

    An attacker can decode the Mac address of the server (in this case, 6f:b6:f9:8f:3a:da) and use it to brute force the rest of the URL, as demonstrated in this article by Versprite.

UUIDv2s are weird

I know you mentioned it's a long story with UUIDv2 when outlining the different versions of UUID... Surely it can't be that bad, can it?

No, really, it's a long story. It's complex enough that I've written a second blog post explaining what UUIDv2s are, why they're not widely used, and how they came to be.

In the meantime, the short version of UUIDv2 is:


UUIDv2 matches the specification for UUIDv1 but replaces some of the timing and clock information with identification numbers for operating system usage.

They're rarely implemented into most UUID libraries and are used even less. This is because of 2 reasons:

  1. They're not documented as part of the newest UUID specification
  2. Significant pitfalls in UUIDv2's generation schema that regularly leads to collision with other generated UUIDv2s

As a result, we won't talk about UUIDv2s anymore today. Instead, I'd encourage you to read the follow-up article to this one if you're interested.

Namespace Your IDs with UUIDv3 and UUIDv5

Let's say you're creating a database of URLs tracked on your website. You want to use the URL as the primary key in your database but don't want to use the full URL, as it will bloat your key's storage requirements.

On the other side of the coin: You can't use any random data to store as your URL's primary key since it could introduce duplicate URL entries into your database table.

Is there a good way to generate the same unique ID from the same input every single time?

Intro: UUIDv3 and UUIDv5.

Both of these UUID versions take the following information:

  • A constant UUID (called a "namespace")
  • An input string (called a "name")

These UUID versions then output a UUID that contains a hash of the namespace and name concatenated together.

You can think of the generation algorithm for both of these UUID versions as the following:

UUID = hash(NAMESPACE + NAME)

For example, here's UUIDv3, which uses MD5 to hash the concatenated value:

A UUID broken down into "MD5 High", a dash, "MD5 High", a dash, "Version", "MD5 Mid", a dash, "Variant", "MD5 Low", a dash, and finally another "MD5 Low". An example UUIDv3 might be "8d9aeee5-d9ad-3934-84f4-ac533183424d"

Compare this to UUIDv5, which uses SHA-1 to generate the hash:

A UUID broken down seperate parts. These parts reflect the same as UUIDv3, but instead of "MD5" parts, they're "SHA". An example UUIDv5 might be "6b5eb089-463b-5cfe-a881-80f5fd6545b0"
What is a "namespace" in UUIDv3 and UUIDv5?

Some of you might be wondering "What is a namespace and why does it need to be another UUID?"

Great question. Let's consider the following usecase:

Let's say we want to assign a UUID to each blog post on the Playful Programming site. We could simply assign the URL:

/posts/intro-to-uuid

To the input value of the hash function of the UUID generation.

This works if we only have one instance of the website, but we often deploy multiple versions of the website in order to make sure the new versions of Playful Programming are running properly before deploying to production.

For example, we might have:

framework-guide.playfulprogramming.com

To host a preview of the Framework Field Guide book early access.

This isn't the real URL, just an example. Want to have the real early access link? Join our Discord and ask!

As a result of the early access site being a mirror of the main site, many of the URLs might be duplicated. We want to allow seperate comments from the preview site vs. the production site so that issues in preview can be resolved before going live.

If we used the same constant UUID for each site:

00000000-0000-0000-0000-000000000000

Each URL with the same path would lead to the same UUID being generated. This means that if we used this UUID to track comments, they would be duplicated on each deployment of the site.

Luckily for us, we may not need to define our own constant UUID. The UUID specification establishes 4 pre-defined namespaces for common use cases. The pre-defined namespaces are:

  • DNS: 6ba7b810-9dad-11d1-80b4-00c04fd430c8
  • URL: 6ba7b811-9dad-11d1-80b4-00c04fd430c8
  • OID: 6ba7b812-9dad-11d1-80b4-00c04fd430c8
  • X.500 DN: 6ba7b814-9dad-11d1-80b4-00c04fd430c8

Knowing this, we can do something akin to the following:

UU_UUID = uuidv5(DNS_UUID, 'playfulprogramming.com');FFG_UUID = uuidv5(DNS_UUID, 'framework-guide.playfulprogramming.com');

And use this base namespace for each post URL:

FFG_POST_UUID = uuidv5(FFG_UUID, "/posts/what-are-uuids");UU_POST_UUID = uuidv5(UU_UUID, "/posts/what-are-uuids");

Now, each of the posts will have a distinct UUID. This is what the namespace ensures; each distinct environment has their own hash for the same name value.

When should you use UUIDv3 vs. UUIDv5?

When provided the option, when should you use UUIDv3 against UUIDv5? When should you do the opposite?

You should use UUIDv3 when:

  • Performance is key, or system resources are limited
  • You're OK with people knowing the original name and namespace values

This is because UUIDv3 uses the less secure MD5 hashing method for its contents.

Likewise, you should use UUIDv5 when:

  • Performance is less of a concern
  • You're not OK with people knowing the original name and namespace values

As UUIDv5's SHA-1 is significantly more secure with hashing its contents.

Pros of UUIDv3 and UUIDv5
  • Contents are stable: In instances where you need the output to match the input every time, no other UUID versions can fulfill this use case.
Cons of UUIDv3 and UUIDv5
  • Contents are not random. While output stability can be incredibly useful, it has somewhat limited usage in ID generation. As such, it may not fit your needs that other UUIDs may.

Generate non-clashing random IDs with UUIDv4

While the previous versions of UUID have had some kind of input data, UUIDv4 breaks from this pattern. Instead, a UUIDv4 is defined as random data alongside the standard version and variant UUID standards.

A UUIDv4 is defined by 5 different segments of random data separated by a dash. Inside of this UUID is also encoded a UUID version code and a variant code. An example UUIDv4 might be "170e8013-f7b9-4a15-9e2e-6fc86dcb98fd".
Pros of UUIDv4
Cons of UUIDv4
  • No encoded metadata: This UUID does not convey any information about the item it's being generated for. This means you must store this data elsewhere and do additional lookups to find this data.

Conclusion

And that's it; That's all of the UUID versions that are part of the original 2005 specification! Hopefully, this has been helpful in learning about the different UUID versions and when each is useful.

In the next article, we'll talk about how UUIDs came to be, what happened to UUIDv2, and why you shouldn't use it.

Until then, happy hacking ID generating!

Subscribe to our newsletter!

Subscribe to our newsletter to get updates on new content we create, events we have coming up, and more! We'll make sure not to spam you and provide good insights to the content we have.