| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123 |
- package core
- import (
- "crypto/hmac"
- "crypto/md5"
- "encoding/hex"
- mrand "math/rand"
- "time"
- "github.com/v2ray/v2ray-core/log"
- )
- const (
- IDBytesLen = 16
- )
- // The ID of en entity, in the form of an UUID.
- type ID struct {
- String string
- Bytes []byte
- cmdKey []byte
- }
- func NewID(id string) (ID, error) {
- idBytes, err := UUIDToID(id)
- if err != nil {
- return ID{}, log.Error("Failed to parse id %s", id)
- }
- md5hash := md5.New()
- md5hash.Write(idBytes)
- md5hash.Write([]byte("c48619fe-8f02-49e0-b9e9-edf763e17e21"))
- cmdKey := md5.Sum(nil)
- return ID{
- String: id,
- Bytes: idBytes,
- cmdKey: cmdKey[:],
- }, nil
- }
- func (v ID) TimeRangeHash(rangeSec int) ([]byte, int64) {
- nowSec := time.Now().UTC().Unix()
- delta := mrand.Intn(rangeSec*2) - rangeSec
- targetSec := nowSec + int64(delta)
- return v.TimeHash(targetSec), targetSec
- }
- func (v ID) TimeHash(timeSec int64) []byte {
- buffer := []byte{
- byte(timeSec >> 56),
- byte(timeSec >> 48),
- byte(timeSec >> 40),
- byte(timeSec >> 32),
- byte(timeSec >> 24),
- byte(timeSec >> 16),
- byte(timeSec >> 8),
- byte(timeSec),
- }
- return v.Hash(buffer)
- }
- func (v ID) Hash(data []byte) []byte {
- hasher := hmac.New(md5.New, v.Bytes)
- hasher.Write(data)
- return hasher.Sum(nil)
- }
- func (v ID) CmdKey() []byte {
- return v.cmdKey
- }
- func TimestampHash(timeSec int64) []byte {
- md5hash := md5.New()
- buffer := []byte{
- byte(timeSec >> 56),
- byte(timeSec >> 48),
- byte(timeSec >> 40),
- byte(timeSec >> 32),
- byte(timeSec >> 24),
- byte(timeSec >> 16),
- byte(timeSec >> 8),
- byte(timeSec),
- }
- md5hash.Write(buffer)
- md5hash.Write(buffer)
- md5hash.Write(buffer)
- md5hash.Write(buffer)
- return md5hash.Sum(nil)
- }
- var byteGroups = []int{8, 4, 4, 4, 12}
- // TODO: leverage a full functional UUID library
- func UUIDToID(uuid string) (v []byte, err error) {
- v = make([]byte, 16)
- text := []byte(uuid)
- if len(text) < 32 {
- err = log.Error("uuid: invalid UUID string: %s", text)
- return
- }
- b := v[:]
- for _, byteGroup := range byteGroups {
- if text[0] == '-' {
- text = text[1:]
- }
- _, err = hex.Decode(b[:byteGroup/2], text[:byteGroup])
- if err != nil {
- return
- }
- text = text[byteGroup:]
- b = b[byteGroup/2:]
- }
- return
- }
|