2018-05-14 16:26:38 +00:00
|
|
|
package platform
|
|
|
|
|
|
|
|
import (
|
2018-07-20 10:24:07 +00:00
|
|
|
"encoding/binary"
|
2018-05-14 16:26:38 +00:00
|
|
|
"encoding/hex"
|
|
|
|
"encoding/json"
|
2018-07-20 10:24:07 +00:00
|
|
|
"errors"
|
2018-05-14 16:26:38 +00:00
|
|
|
)
|
|
|
|
|
2018-10-10 18:57:18 +00:00
|
|
|
// IDLength is the exact length a string (or a byte slice representing it) must have in order to be decoded into a valid ID.
|
|
|
|
const IDLength = 16
|
2018-07-20 10:24:07 +00:00
|
|
|
|
2018-10-10 19:23:29 +00:00
|
|
|
// ErrInvalidID signifies invalid IDs.
|
2018-07-20 10:24:07 +00:00
|
|
|
var ErrInvalidID = errors.New("invalid ID")
|
|
|
|
|
2018-10-10 19:23:29 +00:00
|
|
|
// ErrInvalidIDLength is returned when an ID has the incorrect number of bytes.
|
|
|
|
var ErrInvalidIDLength = errors.New("id must have a length of 16 bytes")
|
2018-07-20 10:24:07 +00:00
|
|
|
|
2018-05-14 16:26:38 +00:00
|
|
|
// ID is a unique identifier.
|
2018-07-20 10:24:07 +00:00
|
|
|
//
|
|
|
|
// Its zero value is not a valid ID.
|
|
|
|
type ID uint64
|
2018-05-14 16:26:38 +00:00
|
|
|
|
2018-05-16 15:32:19 +00:00
|
|
|
// IDGenerator represents a generator for IDs.
|
|
|
|
type IDGenerator interface {
|
|
|
|
// ID creates unique byte slice ID.
|
2018-07-30 11:38:23 +00:00
|
|
|
ID() ID
|
2018-05-16 15:32:19 +00:00
|
|
|
}
|
|
|
|
|
2018-07-20 10:24:07 +00:00
|
|
|
// IDFromString creates an ID from a given string.
|
|
|
|
//
|
|
|
|
// It errors if the input string does not match a valid ID.
|
|
|
|
func IDFromString(str string) (*ID, error) {
|
2018-07-20 15:00:31 +00:00
|
|
|
var id ID
|
2018-07-20 10:24:07 +00:00
|
|
|
err := id.DecodeFromString(str)
|
2018-07-20 15:00:31 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return &id, nil
|
|
|
|
}
|
|
|
|
|
2018-09-28 15:06:21 +00:00
|
|
|
// InvalidID returns a zero ID.
|
|
|
|
func InvalidID() ID {
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
|
2018-05-14 16:26:38 +00:00
|
|
|
// Decode parses b as a hex-encoded byte-slice-string.
|
2018-07-20 10:24:07 +00:00
|
|
|
//
|
|
|
|
// It errors if the input byte slice does not have the correct length
|
|
|
|
// or if it contains all zeros.
|
2018-05-14 16:26:38 +00:00
|
|
|
func (i *ID) Decode(b []byte) error {
|
2018-10-10 18:57:18 +00:00
|
|
|
if len(b) != IDLength {
|
2018-07-20 10:24:07 +00:00
|
|
|
return ErrInvalidIDLength
|
|
|
|
}
|
|
|
|
|
2018-10-10 18:57:18 +00:00
|
|
|
dst := make([]byte, hex.DecodedLen(IDLength))
|
2018-05-14 16:26:38 +00:00
|
|
|
_, err := hex.Decode(dst, b)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2018-07-20 10:24:07 +00:00
|
|
|
*i = ID(binary.BigEndian.Uint64(dst))
|
|
|
|
|
|
|
|
if !i.Valid() {
|
|
|
|
return ErrInvalidID
|
|
|
|
}
|
|
|
|
|
2018-05-14 16:26:38 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// DecodeFromString parses s as a hex-encoded string.
|
|
|
|
func (i *ID) DecodeFromString(s string) error {
|
|
|
|
return i.Decode([]byte(s))
|
|
|
|
}
|
|
|
|
|
|
|
|
// Encode converts ID to a hex-encoded byte-slice-string.
|
2018-07-20 10:24:07 +00:00
|
|
|
//
|
|
|
|
// It errors if the receiving ID holds its zero value.
|
|
|
|
func (i ID) Encode() ([]byte, error) {
|
|
|
|
if !i.Valid() {
|
|
|
|
return nil, ErrInvalidID
|
|
|
|
}
|
|
|
|
|
2018-10-10 18:57:18 +00:00
|
|
|
b := make([]byte, hex.DecodedLen(IDLength))
|
2018-07-20 10:24:07 +00:00
|
|
|
binary.BigEndian.PutUint64(b, uint64(i))
|
|
|
|
|
|
|
|
dst := make([]byte, hex.EncodedLen(len(b)))
|
|
|
|
hex.Encode(dst, b)
|
|
|
|
return dst, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Valid checks whether the receiving ID is a valid one or not.
|
|
|
|
func (i ID) Valid() bool {
|
|
|
|
return i != 0
|
2018-07-30 11:38:23 +00:00
|
|
|
}
|
|
|
|
|
2018-07-20 10:24:07 +00:00
|
|
|
// String returns the ID as a hex encoded string.
|
|
|
|
//
|
|
|
|
// Returns an empty string in the case the ID is invalid.
|
2018-05-14 16:26:38 +00:00
|
|
|
func (i ID) String() string {
|
2018-07-20 10:24:07 +00:00
|
|
|
enc, _ := i.Encode()
|
|
|
|
return string(enc)
|
2018-05-14 16:26:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// UnmarshalJSON implements JSON unmarshaller for IDs.
|
|
|
|
func (i *ID) UnmarshalJSON(b []byte) error {
|
2018-09-28 13:26:39 +00:00
|
|
|
if b[0] == '"' {
|
2018-09-26 11:50:24 +00:00
|
|
|
b = b[1:]
|
|
|
|
}
|
2018-09-28 13:26:39 +00:00
|
|
|
if b[len(b)-1] == '"' {
|
2018-09-26 11:50:24 +00:00
|
|
|
b = b[:len(b)-1]
|
|
|
|
}
|
2018-05-14 16:26:38 +00:00
|
|
|
return i.Decode(b)
|
|
|
|
}
|
|
|
|
|
|
|
|
// MarshalJSON implements JSON marshaller for IDs.
|
|
|
|
func (i ID) MarshalJSON() ([]byte, error) {
|
2018-07-20 10:24:07 +00:00
|
|
|
enc, err := i.Encode()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return json.Marshal(string(enc))
|
2018-05-14 16:26:38 +00:00
|
|
|
}
|