76 lines
		
	
	
		
			2.1 KiB
		
	
	
	
		
			Go
		
	
	
			
		
		
	
	
			76 lines
		
	
	
		
			2.1 KiB
		
	
	
	
		
			Go
		
	
	
package protocol
 | 
						|
 | 
						|
import (
 | 
						|
	"crypto/rand"
 | 
						|
	"fmt"
 | 
						|
	"reflect"
 | 
						|
)
 | 
						|
 | 
						|
// RandReader is the random reader the protocol package will use to read
 | 
						|
// random bytes from. This is exported for testing, and should not be used.
 | 
						|
var RandReader = rand.Reader
 | 
						|
 | 
						|
const idempotencyTokenFillTag = `idempotencyToken`
 | 
						|
 | 
						|
// CanSetIdempotencyToken returns true if the struct field should be
 | 
						|
// automatically populated with a Idempotency token.
 | 
						|
//
 | 
						|
// Only *string and string type fields that are tagged with idempotencyToken
 | 
						|
// which are not already set can be auto filled.
 | 
						|
func CanSetIdempotencyToken(v reflect.Value, f reflect.StructField) bool {
 | 
						|
	switch u := v.Interface().(type) {
 | 
						|
	// To auto fill an Idempotency token the field must be a string,
 | 
						|
	// tagged for auto fill, and have a zero value.
 | 
						|
	case *string:
 | 
						|
		return u == nil && len(f.Tag.Get(idempotencyTokenFillTag)) != 0
 | 
						|
	case string:
 | 
						|
		return len(u) == 0 && len(f.Tag.Get(idempotencyTokenFillTag)) != 0
 | 
						|
	}
 | 
						|
 | 
						|
	return false
 | 
						|
}
 | 
						|
 | 
						|
// GetIdempotencyToken returns a randomly generated idempotency token.
 | 
						|
func GetIdempotencyToken() string {
 | 
						|
	b := make([]byte, 16)
 | 
						|
	RandReader.Read(b)
 | 
						|
 | 
						|
	return UUIDVersion4(b)
 | 
						|
}
 | 
						|
 | 
						|
// SetIdempotencyToken will set the value provided with a Idempotency Token.
 | 
						|
// Given that the value can be set. Will panic if value is not setable.
 | 
						|
func SetIdempotencyToken(v reflect.Value) {
 | 
						|
	if v.Kind() == reflect.Ptr {
 | 
						|
		if v.IsNil() && v.CanSet() {
 | 
						|
			v.Set(reflect.New(v.Type().Elem()))
 | 
						|
		}
 | 
						|
		v = v.Elem()
 | 
						|
	}
 | 
						|
	v = reflect.Indirect(v)
 | 
						|
 | 
						|
	if !v.CanSet() {
 | 
						|
		panic(fmt.Sprintf("unable to set idempotnecy token %v", v))
 | 
						|
	}
 | 
						|
 | 
						|
	b := make([]byte, 16)
 | 
						|
	_, err := rand.Read(b)
 | 
						|
	if err != nil {
 | 
						|
		// TODO handle error
 | 
						|
		return
 | 
						|
	}
 | 
						|
 | 
						|
	v.Set(reflect.ValueOf(UUIDVersion4(b)))
 | 
						|
}
 | 
						|
 | 
						|
// UUIDVersion4 returns a Version 4 random UUID from the byte slice provided
 | 
						|
func UUIDVersion4(u []byte) string {
 | 
						|
	// https://en.wikipedia.org/wiki/Universally_unique_identifier#Version_4_.28random.29
 | 
						|
	// 13th character is "4"
 | 
						|
	u[6] = (u[6] | 0x40) & 0x4F
 | 
						|
	// 17th character is "8", "9", "a", or "b"
 | 
						|
	u[8] = (u[8] | 0x80) & 0xBF
 | 
						|
 | 
						|
	return fmt.Sprintf(`%X-%X-%X-%X-%X`, u[0:4], u[4:6], u[6:8], u[8:10], u[10:])
 | 
						|
}
 |