2018-05-14 16:26:38 +00:00
|
|
|
package snowflake
|
|
|
|
|
|
|
|
import (
|
2018-09-07 22:17:49 +00:00
|
|
|
"errors"
|
2018-05-14 16:26:38 +00:00
|
|
|
"math/rand"
|
2018-09-07 22:17:49 +00:00
|
|
|
"sync"
|
2018-05-14 16:26:38 +00:00
|
|
|
"time"
|
|
|
|
|
2019-01-08 00:37:16 +00:00
|
|
|
platform "github.com/influxdata/influxdb"
|
|
|
|
"github.com/influxdata/influxdb/pkg/snowflake"
|
2018-05-14 16:26:38 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
func init() {
|
|
|
|
rand.Seed(time.Now().UnixNano())
|
2018-09-07 22:17:49 +00:00
|
|
|
SetGlobalMachineID(rand.Intn(1023))
|
|
|
|
}
|
|
|
|
|
|
|
|
var globalmachineID struct {
|
|
|
|
id int
|
|
|
|
set bool
|
|
|
|
sync.RWMutex
|
|
|
|
}
|
|
|
|
|
|
|
|
// ErrGlobalIDBadVal means that the global machine id value wasn't properly set.
|
|
|
|
var ErrGlobalIDBadVal = errors.New("globalID must be a number between (inclusive) 0 and 1023")
|
|
|
|
|
|
|
|
// SetGlobalMachineID returns the global machine id. This number is limited to a number between 0 and 1023 inclusive.
|
|
|
|
func SetGlobalMachineID(id int) error {
|
|
|
|
if id > 1023 || id < 0 {
|
|
|
|
return ErrGlobalIDBadVal
|
|
|
|
}
|
|
|
|
globalmachineID.Lock()
|
|
|
|
globalmachineID.id = id
|
|
|
|
globalmachineID.set = true
|
|
|
|
globalmachineID.Unlock()
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// GlobalMachineID returns the global machine id. This number is limited to a number between 0 and 1023 inclusive.
|
|
|
|
func GlobalMachineID() int {
|
|
|
|
var id int
|
|
|
|
globalmachineID.RLock()
|
|
|
|
id = int(globalmachineID.id)
|
2018-09-13 21:07:29 +00:00
|
|
|
globalmachineID.RUnlock()
|
2018-09-07 22:17:49 +00:00
|
|
|
return id
|
2018-05-14 16:26:38 +00:00
|
|
|
}
|
|
|
|
|
2018-09-07 22:17:49 +00:00
|
|
|
// NewDefaultIDGenerator returns an *IDGenerator that uses the currently set global machine ID.
|
|
|
|
// If you change the global machine id, it will not change the id in any generators that have already been created.
|
|
|
|
func NewDefaultIDGenerator() *IDGenerator {
|
|
|
|
globalmachineID.RLock()
|
|
|
|
defer globalmachineID.RUnlock()
|
|
|
|
if globalmachineID.set {
|
|
|
|
return NewIDGenerator(WithMachineID(globalmachineID.id))
|
|
|
|
}
|
|
|
|
return NewIDGenerator()
|
|
|
|
}
|
|
|
|
|
2018-07-20 10:24:07 +00:00
|
|
|
// IDGenerator holds the ID generator.
|
2018-05-14 16:26:38 +00:00
|
|
|
type IDGenerator struct {
|
|
|
|
Generator *snowflake.Generator
|
|
|
|
}
|
|
|
|
|
2018-09-07 22:17:49 +00:00
|
|
|
// IDGeneratorOp is an option for an IDGenerator.
|
|
|
|
type IDGeneratorOp func(*IDGenerator)
|
|
|
|
|
|
|
|
// WithMachineID uses the low 12 bits of machineID to set the machine ID for the snowflake ID.
|
|
|
|
func WithMachineID(machineID int) IDGeneratorOp {
|
|
|
|
return func(g *IDGenerator) {
|
|
|
|
g.Generator = snowflake.New(machineID & 1023)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewIDGenerator returns a new IDGenerator. Optionally you can use an IDGeneratorOp.
|
|
|
|
// to use a specific Generator
|
|
|
|
func NewIDGenerator(opts ...IDGeneratorOp) *IDGenerator {
|
|
|
|
gen := &IDGenerator{}
|
|
|
|
for _, f := range opts {
|
|
|
|
f(gen)
|
|
|
|
}
|
|
|
|
if gen.Generator == nil {
|
|
|
|
gen.Generator = snowflake.New(rand.Intn(1023))
|
2018-05-14 16:26:38 +00:00
|
|
|
}
|
2018-09-07 22:17:49 +00:00
|
|
|
return gen
|
2018-05-14 16:26:38 +00:00
|
|
|
}
|
|
|
|
|
2018-09-07 22:17:49 +00:00
|
|
|
// ID returns the next platform.ID from an IDGenerator.
|
2018-07-30 14:29:04 +00:00
|
|
|
func (g *IDGenerator) ID() platform.ID {
|
2018-07-20 10:24:07 +00:00
|
|
|
var id platform.ID
|
2018-09-28 13:26:39 +00:00
|
|
|
for !id.Valid() {
|
2018-07-20 10:24:07 +00:00
|
|
|
id = platform.ID(g.Generator.Next())
|
|
|
|
}
|
2018-07-30 14:29:04 +00:00
|
|
|
return id
|
2018-05-14 16:26:38 +00:00
|
|
|
}
|