version7.go

  1// Copyright 2023 Google Inc.  All rights reserved.
  2// Use of this source code is governed by a BSD-style
  3// license that can be found in the LICENSE file.
  4
  5package uuid
  6
  7import (
  8	"io"
  9)
 10
 11// UUID version 7 features a time-ordered value field derived from the widely
 12// implemented and well known Unix Epoch timestamp source,
 13// the number of milliseconds seconds since midnight 1 Jan 1970 UTC, leap seconds excluded.
 14// As well as improved entropy characteristics over versions 1 or 6.
 15//
 16// see https://datatracker.ietf.org/doc/html/draft-peabody-dispatch-new-uuid-format-03#name-uuid-version-7
 17//
 18// Implementations SHOULD utilize UUID version 7 over UUID version 1 and 6 if possible.
 19//
 20// NewV7 returns a Version 7 UUID based on the current time(Unix Epoch).
 21// Uses the randomness pool if it was enabled with EnableRandPool.
 22// On error, NewV7 returns Nil and an error
 23func NewV7() (UUID, error) {
 24	uuid, err := NewRandom()
 25	if err != nil {
 26		return uuid, err
 27	}
 28	makeV7(uuid[:])
 29	return uuid, nil
 30}
 31
 32// NewV7FromReader returns a Version 7 UUID based on the current time(Unix Epoch).
 33// it use NewRandomFromReader fill random bits.
 34// On error, NewV7FromReader returns Nil and an error.
 35func NewV7FromReader(r io.Reader) (UUID, error) {
 36	uuid, err := NewRandomFromReader(r)
 37	if err != nil {
 38		return uuid, err
 39	}
 40
 41	makeV7(uuid[:])
 42	return uuid, nil
 43}
 44
 45// makeV7 fill 48 bits time (uuid[0] - uuid[5]), set version b0111 (uuid[6])
 46// uuid[8] already has the right version number (Variant is 10)
 47// see function NewV7 and NewV7FromReader
 48func makeV7(uuid []byte) {
 49	/*
 50		 0                   1                   2                   3
 51		 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
 52		+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
 53		|                           unix_ts_ms                          |
 54		+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
 55		|          unix_ts_ms           |  ver  |  rand_a (12 bit seq)  |
 56		+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
 57		|var|                        rand_b                             |
 58		+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
 59		|                            rand_b                             |
 60		+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
 61	*/
 62	_ = uuid[15] // bounds check
 63
 64	t, s := getV7Time()
 65
 66	uuid[0] = byte(t >> 40)
 67	uuid[1] = byte(t >> 32)
 68	uuid[2] = byte(t >> 24)
 69	uuid[3] = byte(t >> 16)
 70	uuid[4] = byte(t >> 8)
 71	uuid[5] = byte(t)
 72
 73	uuid[6] = 0x70 | (0x0F & byte(s>>8))
 74	uuid[7] = byte(s)
 75}
 76
 77// lastV7time is the last time we returned stored as:
 78//
 79//	52 bits of time in milliseconds since epoch
 80//	12 bits of (fractional nanoseconds) >> 8
 81var lastV7time int64
 82
 83const nanoPerMilli = 1000000
 84
 85// getV7Time returns the time in milliseconds and nanoseconds / 256.
 86// The returned (milli << 12 + seq) is guarenteed to be greater than
 87// (milli << 12 + seq) returned by any previous call to getV7Time.
 88func getV7Time() (milli, seq int64) {
 89	timeMu.Lock()
 90	defer timeMu.Unlock()
 91
 92	nano := timeNow().UnixNano()
 93	milli = nano / nanoPerMilli
 94	// Sequence number is between 0 and 3906 (nanoPerMilli>>8)
 95	seq = (nano - milli*nanoPerMilli) >> 8
 96	now := milli<<12 + seq
 97	if now <= lastV7time {
 98		now = lastV7time + 1
 99		milli = now >> 12
100		seq = now & 0xfff
101	}
102	lastV7time = now
103	return milli, seq
104}