mirror of
https://github.com/moby/moby.git
synced 2026-01-11 18:51:37 +00:00
The stringid package is used in many places; while it's trivial to implement a similar utility, let's just provide it as a utility package in the client, removing the daemon-specific logic. For integration tests, I opted to use the implementation in the client, as those should not ideally not make assumptions about the daemon implementation. Signed-off-by: Sebastiaan van Stijn <github@gone.nl>
71 lines
2.0 KiB
Go
71 lines
2.0 KiB
Go
// Package stringid provides helper functions for dealing with string identifiers
|
|
package stringid
|
|
|
|
import (
|
|
"crypto/rand"
|
|
"encoding/hex"
|
|
"strings"
|
|
)
|
|
|
|
const (
|
|
shortLen = 12
|
|
fullLen = 64
|
|
)
|
|
|
|
// TruncateID returns a shorthand version of a string identifier for presentation.
|
|
// For convenience, it accepts both digests ("sha256:xxxx") and IDs without an
|
|
// algorithm prefix. It truncates the algorithm (if any) before truncating the
|
|
// ID. The length of the truncated ID is currently fixed, but users should make
|
|
// no assumptions of this to not change; it is merely a prefix of the ID that
|
|
// provides enough uniqueness for common scenarios.
|
|
//
|
|
// Truncated IDs ("ID-prefixes") usually can be used to uniquely identify an
|
|
// object (such as a container or network), but collisions may happen, in
|
|
// which case an "ambiguous result" error is produced. In case of a collision,
|
|
// the caller should try with a longer prefix or the full-length ID.
|
|
func TruncateID(id string) string {
|
|
if i := strings.IndexRune(id, ':'); i >= 0 {
|
|
id = id[i+1:]
|
|
}
|
|
if len(id) > shortLen {
|
|
id = id[:shortLen]
|
|
}
|
|
return id
|
|
}
|
|
|
|
// GenerateRandomID returns a unique, 64-character ID consisting of a-z, 0-9.
|
|
// It guarantees that the ID, when truncated ([TruncateID]) does not consist
|
|
// of numbers only, so that the truncated ID can be used as hostname for
|
|
// containers.
|
|
func GenerateRandomID() string {
|
|
b := make([]byte, 32)
|
|
for {
|
|
if _, err := rand.Read(b); err != nil {
|
|
panic(err) // This shouldn't happen
|
|
}
|
|
id := hex.EncodeToString(b)
|
|
|
|
// make sure that the truncated ID does not consist of only numeric
|
|
// characters, as it's used as default hostname for containers.
|
|
//
|
|
// See:
|
|
// - https://github.com/moby/moby/issues/3869
|
|
// - https://bugzilla.redhat.com/show_bug.cgi?id=1059122
|
|
if allNum(id[:shortLen]) {
|
|
// all numbers; try again
|
|
continue
|
|
}
|
|
return id
|
|
}
|
|
}
|
|
|
|
// allNum checks whether id consists of only numbers (0-9).
|
|
func allNum(id string) bool {
|
|
for _, c := range []byte(id) {
|
|
if c > '9' || c < '0' {
|
|
return false
|
|
}
|
|
}
|
|
return true
|
|
}
|