keygen.go 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194
  1. package utils
  2. import (
  3. "errors"
  4. "github.com/spf13/viper"
  5. "net"
  6. "sync"
  7. "time"
  8. )
  9. // 获取新的程序生成的编号
  10. func NextId() Long {
  11. id, e := SingletonSnowflakeKeyGen().NextId()
  12. if nil != e {
  13. println("get next id failed", e)
  14. return 0
  15. }
  16. return Long(id)
  17. }
  18. /*------------------------------------------------------Singleton----------------------------------------------------*/
  19. var snowflakeKeyGen *Snowflake
  20. var once sync.Once
  21. func SingletonSnowflakeKeyGen() *Snowflake {
  22. once.Do(func() {
  23. snowflakeKeyGen = NewSnowflake(SnowflakeSettings{})
  24. })
  25. return snowflakeKeyGen
  26. }
  27. /*----------------------------------------------------SnowflakeKeyGen------------------------------------------------*/
  28. const (
  29. SnowflakeTimeUnit = 1e7 // 时间单位,一纳秒的多少倍,1e6 = 一毫秒,1e7 = 百分之一秒,1e8 = 十分之一秒
  30. BitLenSequence = 8 // 序列号的个数最多256个(0-255),即每单位时间并发数,如时间单位是1e7,则单实例qps = 25600
  31. BitLenDataCenterId = 3 // 数据中心个数最多8个(0-7),即同一个环境(生产、预发布、测试等)的数据中心(假设一个机房相同数据域的应用服务器集群只有一个,则数据中心数等于机房数)最多有8个
  32. BitLenMachineId = 16 // 同一个数据中心下最多65536个应用实例(0-65535),默认是根据实例ip后两段算实例id(k8s环境动态创建Pod,也建议用此方式),所以需要预留255 * 255这么多
  33. BitLenTime = 36 // 时间戳之差最大 = 2的36次方 * 时间单位 / 1e9 秒,目前的设计最多可以用21.79年就需要更新开始时间(随之还需要归档旧数据和更新次新数据id)
  34. // 总共63位,不超过bit64
  35. )
  36. type SnowflakeSettings struct {
  37. StartTime time.Time
  38. DataCenterId func() (uint16, error)
  39. MachineId func() (uint16, error)
  40. CheckMachineId func(uint16) bool
  41. }
  42. type Snowflake struct {
  43. mutex *sync.Mutex
  44. startTime int64
  45. elapsedTime int64
  46. sequence uint16
  47. dataCenterId uint16
  48. machineId uint16
  49. }
  50. func NewSnowflake(st SnowflakeSettings) *Snowflake {
  51. sf := new(Snowflake)
  52. sf.mutex = new(sync.Mutex)
  53. sf.sequence = uint16(1<<BitLenSequence - 1)
  54. if st.StartTime.After(time.Now()) {
  55. return nil
  56. }
  57. if st.StartTime.IsZero() {
  58. sf.startTime = toSnowflakeTime(time.Date(2018, 9, 26, 0, 0, 0, 0, time.UTC)) //没有配置默认使用此时间
  59. } else {
  60. sf.startTime = toSnowflakeTime(st.StartTime)
  61. }
  62. var err error
  63. if st.MachineId == nil {
  64. sf.machineId, err = GetPrivateIPv4Id() // 没有配置会读机器内网ip后两段,然后计算出一个值
  65. } else {
  66. sf.machineId, err = st.MachineId()
  67. }
  68. if nil != err {
  69. err = nil
  70. sf.machineId = uint16(0)
  71. }
  72. if st.DataCenterId == nil {
  73. if id := viper.GetInt("data_center_id"); id > 0 { // 没有配置会尝试从配置文件读取数据中心id
  74. sf.dataCenterId = uint16(id)
  75. } else { // 如果配置文件也没有,默认数据中心id为0
  76. sf.dataCenterId = uint16(0)
  77. }
  78. } else {
  79. sf.dataCenterId, err = st.DataCenterId()
  80. if nil != err {
  81. sf.dataCenterId = uint16(0)
  82. }
  83. }
  84. if st.CheckMachineId != nil && !st.CheckMachineId(sf.machineId) {
  85. return nil
  86. }
  87. return sf
  88. }
  89. func (sf *Snowflake) NextId() (uint64, error) {
  90. const maskSequence = uint16(1<<BitLenSequence - 1)
  91. sf.mutex.Lock()
  92. defer sf.mutex.Unlock()
  93. current := currentElapsedTime(sf.startTime)
  94. if sf.elapsedTime < current {
  95. sf.elapsedTime = current
  96. sf.sequence = 0
  97. } else { // sf.elapsedTime >= current
  98. sf.sequence = (sf.sequence + 1) & maskSequence
  99. if sf.sequence == 0 {
  100. sf.elapsedTime++
  101. overtime := sf.elapsedTime - current
  102. time.Sleep(sleepTime(overtime))
  103. }
  104. }
  105. return sf.toId()
  106. }
  107. func toSnowflakeTime(t time.Time) int64 {
  108. return t.UTC().UnixNano() / SnowflakeTimeUnit
  109. }
  110. func currentElapsedTime(startTime int64) int64 {
  111. return toSnowflakeTime(time.Now()) - startTime
  112. }
  113. func sleepTime(overtime int64) time.Duration {
  114. return time.Duration(overtime)*10*time.Millisecond -
  115. time.Duration(time.Now().UTC().UnixNano()%SnowflakeTimeUnit)*time.Nanosecond
  116. }
  117. func (sf *Snowflake) toId() (uint64, error) {
  118. if sf.elapsedTime >= 1<<BitLenTime {
  119. return 0, errors.New("over the time limit")
  120. }
  121. return uint64(sf.elapsedTime)<<(BitLenSequence+BitLenDataCenterId+BitLenMachineId) |
  122. uint64(sf.sequence)<<(BitLenDataCenterId+BitLenMachineId) |
  123. uint64(sf.dataCenterId)<<BitLenMachineId |
  124. uint64(sf.machineId), nil
  125. }
  126. func privateIPv4() (net.IP, error) {
  127. as, err := net.InterfaceAddrs()
  128. if err != nil {
  129. return nil, err
  130. }
  131. for _, a := range as {
  132. ipnet, ok := a.(*net.IPNet)
  133. if !ok || ipnet.IP.IsLoopback() {
  134. continue
  135. }
  136. ip := ipnet.IP.To4()
  137. if isPrivateIPv4(ip) {
  138. return ip, nil
  139. }
  140. }
  141. return nil, errors.New("no private ip address")
  142. }
  143. func isPrivateIPv4(ip net.IP) bool {
  144. return ip != nil &&
  145. (ip[0] == 10 || ip[0] == 172 && (ip[1] >= 16 && ip[1] < 32) || ip[0] == 192 && ip[1] == 168)
  146. }
  147. func GetPrivateIPv4Id() (uint16, error) {
  148. ip, err := privateIPv4()
  149. if err != nil {
  150. return 0, err
  151. }
  152. return uint16(ip[2])<<8 + uint16(ip[3]), nil
  153. }
  154. func Decompose(id uint64) map[string]uint64 {
  155. const maskDataCenterId = uint64((1<<BitLenDataCenterId - 1) << BitLenMachineId)
  156. const maskMachineId = uint64(1<<BitLenMachineId - 1)
  157. dataCenterId := id & maskDataCenterId >> BitLenMachineId
  158. machineId := id & maskMachineId
  159. return map[string]uint64{
  160. "id": id,
  161. "dataCenterId": dataCenterId,
  162. "machineId": machineId,
  163. }
  164. }