correlation/generator.go (38 lines of code) (raw):

package correlation import ( "crypto/rand" "io" "sync" "time" "github.com/oklog/ulid/v2" ) // Replaceable for testing purposes. var ulidEntropySource io.Reader = &safeMonotonicReader{ delegate: ulid.Monotonic(rand.Reader, 0), } func generatePseudorandomCorrelationID() string { return "E:" + encodeReverseBase62(time.Now().UnixNano()) } // generateRandomCorrelationID will attempt to generate a correlationid randomly // if this fails, will log a message and fallback to a pseudorandom approach. func generateRandomCorrelationIDWithFallback() string { uid, err := ulid.New(ulid.Timestamp(time.Now()), ulidEntropySource) if err != nil { // Swallow the error and return a pseudorandom correlation_id. // Operators can determine that an error occurred by the shape of the // correlation_id, which will be prefixed with a `E:` return generatePseudorandomCorrelationID() } return uid.String() } // RandomID generates a random correlation ID. // Deprecated: use SafeRandomID instead. // Note, that this method will not return an error, it is here for compatibility reasons only. func RandomID() (string, error) { return generateRandomCorrelationIDWithFallback(), nil } // SafeRandomID generates a random correlation ID. func SafeRandomID() string { return generateRandomCorrelationIDWithFallback() } // safeMonotonicReader is a thread-safe wrapper around a ulid.Monotonic instance, which is not safe for concurrent use by itself. // See https://godoc.org/github.com/oklog/ulid#Monotonic. type safeMonotonicReader struct { mtx sync.Mutex delegate ulid.MonotonicReader } var _ ulid.MonotonicReader = &safeMonotonicReader{} func (r *safeMonotonicReader) MonotonicRead(ms uint64, p []byte) error { r.mtx.Lock() defer r.mtx.Unlock() return r.delegate.MonotonicRead(ms, p) } func (r *safeMonotonicReader) Read(p []byte) (int, error) { r.mtx.Lock() defer r.mtx.Unlock() return r.delegate.Read(p) }