121 lines
3.8 KiB
Go
121 lines
3.8 KiB
Go
|
// Written in 2011-2014 by Dmitry Chestnykh
|
||
|
//
|
||
|
// The author(s) have dedicated all copyright and related and
|
||
|
// neighboring rights to this software to the public domain
|
||
|
// worldwide. Distributed without any warranty.
|
||
|
// http://creativecommons.org/publicdomain/zero/1.0/
|
||
|
|
||
|
// Package uniuri generates random strings good for use in URIs to identify
|
||
|
// unique objects.
|
||
|
//
|
||
|
// Example usage:
|
||
|
//
|
||
|
// s := uniuri.New() // s is now "apHCJBl7L1OmC57n"
|
||
|
//
|
||
|
// A standard string created by New() is 16 bytes in length and consists of
|
||
|
// Latin upper and lowercase letters, and numbers (from the set of 62 allowed
|
||
|
// characters), which means that it has ~95 bits of entropy. To get more
|
||
|
// entropy, you can use NewLen(UUIDLen), which returns 20-byte string, giving
|
||
|
// ~119 bits of entropy, or any other desired length.
|
||
|
//
|
||
|
// Functions read from crypto/rand random source, and panic if they fail to
|
||
|
// read from it.
|
||
|
package uniuri
|
||
|
|
||
|
import (
|
||
|
"crypto/rand"
|
||
|
"math"
|
||
|
)
|
||
|
|
||
|
const (
|
||
|
// StdLen is a standard length of uniuri string to achive ~95 bits of entropy.
|
||
|
StdLen = 16
|
||
|
// UUIDLen is a length of uniuri string to achive ~119 bits of entropy, closest
|
||
|
// to what can be losslessly converted to UUIDv4 (122 bits).
|
||
|
UUIDLen = 20
|
||
|
)
|
||
|
|
||
|
// StdChars is a set of standard characters allowed in uniuri string.
|
||
|
var StdChars = []byte("ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789")
|
||
|
|
||
|
// New returns a new random string of the standard length, consisting of
|
||
|
// standard characters.
|
||
|
func New() string {
|
||
|
return NewLenChars(StdLen, StdChars)
|
||
|
}
|
||
|
|
||
|
// NewLen returns a new random string of the provided length, consisting of
|
||
|
// standard characters.
|
||
|
func NewLen(length int) string {
|
||
|
return NewLenChars(length, StdChars)
|
||
|
}
|
||
|
|
||
|
// maxBufLen is the maximum length of a temporary buffer for random bytes.
|
||
|
const maxBufLen = 2048
|
||
|
|
||
|
// minRegenBufLen is the minimum length of temporary buffer for random bytes
|
||
|
// to fill after the first rand.Read request didn't produce the full result.
|
||
|
// If the initial buffer is smaller, this value is ignored.
|
||
|
// Rationale: for performance, assume it's pointless to request fewer bytes from rand.Read.
|
||
|
const minRegenBufLen = 16
|
||
|
|
||
|
// estimatedBufLen returns the estimated number of random bytes to request
|
||
|
// given that byte values greater than maxByte will be rejected.
|
||
|
func estimatedBufLen(need, maxByte int) int {
|
||
|
return int(math.Ceil(float64(need) * (255 / float64(maxByte))))
|
||
|
}
|
||
|
|
||
|
// NewLenCharsBytes returns a new random byte slice of the provided length, consisting
|
||
|
// of the provided byte slice of allowed characters (maximum 256).
|
||
|
func NewLenCharsBytes(length int, chars []byte) []byte {
|
||
|
if length == 0 {
|
||
|
return nil
|
||
|
}
|
||
|
clen := len(chars)
|
||
|
if clen < 2 || clen > 256 {
|
||
|
panic("uniuri: wrong charset length for NewLenChars")
|
||
|
}
|
||
|
maxrb := 255 - (256 % clen)
|
||
|
buflen := estimatedBufLen(length, maxrb)
|
||
|
if buflen < length {
|
||
|
buflen = length
|
||
|
}
|
||
|
if buflen > maxBufLen {
|
||
|
buflen = maxBufLen
|
||
|
}
|
||
|
buf := make([]byte, buflen) // storage for random bytes
|
||
|
out := make([]byte, length) // storage for result
|
||
|
i := 0
|
||
|
for {
|
||
|
if _, err := rand.Read(buf[:buflen]); err != nil {
|
||
|
panic("uniuri: error reading random bytes: " + err.Error())
|
||
|
}
|
||
|
for _, rb := range buf[:buflen] {
|
||
|
c := int(rb)
|
||
|
if c > maxrb {
|
||
|
// Skip this number to avoid modulo bias.
|
||
|
continue
|
||
|
}
|
||
|
out[i] = chars[c%clen]
|
||
|
i++
|
||
|
if i == length {
|
||
|
return out
|
||
|
}
|
||
|
}
|
||
|
// Adjust new requested length, but no smaller than minRegenBufLen.
|
||
|
buflen = estimatedBufLen(length-i, maxrb)
|
||
|
if buflen < minRegenBufLen && minRegenBufLen < cap(buf) {
|
||
|
buflen = minRegenBufLen
|
||
|
}
|
||
|
if buflen > maxBufLen {
|
||
|
buflen = maxBufLen
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// NewLenChars returns a new random string of the provided length, consisting
|
||
|
// of the provided byte slice of allowed characters (maximum 256).
|
||
|
func NewLenChars(length int, chars []byte) string {
|
||
|
return string(NewLenCharsBytes(length, chars))
|
||
|
}
|