breaker.go 3.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176
  1. package breaker
  2. import (
  3. "sync"
  4. "time"
  5. xtime "go-common/library/time"
  6. )
  7. // Config broker config.
  8. type Config struct {
  9. SwitchOff bool // breaker switch,default off.
  10. // Hystrix
  11. Ratio float32
  12. Sleep xtime.Duration
  13. // Google
  14. K float64
  15. Window xtime.Duration
  16. Bucket int
  17. Request int64
  18. }
  19. func (conf *Config) fix() {
  20. if conf.K == 0 {
  21. conf.K = 1.5
  22. }
  23. if conf.Request == 0 {
  24. conf.Request = 100
  25. }
  26. if conf.Ratio == 0 {
  27. conf.Ratio = 0.5
  28. }
  29. if conf.Sleep == 0 {
  30. conf.Sleep = xtime.Duration(500 * time.Millisecond)
  31. }
  32. if conf.Bucket == 0 {
  33. conf.Bucket = 10
  34. }
  35. if conf.Window == 0 {
  36. conf.Window = xtime.Duration(3 * time.Second)
  37. }
  38. }
  39. // Breaker is a CircuitBreaker pattern.
  40. // FIXME on int32 atomic.LoadInt32(&b.on) == _switchOn
  41. type Breaker interface {
  42. Allow() error
  43. MarkSuccess()
  44. MarkFailed()
  45. }
  46. // Group represents a class of CircuitBreaker and forms a namespace in which
  47. // units of CircuitBreaker.
  48. type Group struct {
  49. mu sync.RWMutex
  50. brks map[string]Breaker
  51. conf *Config
  52. }
  53. const (
  54. // StateOpen when circuit breaker open, request not allowed, after sleep
  55. // some duration, allow one single request for testing the health, if ok
  56. // then state reset to closed, if not continue the step.
  57. StateOpen int32 = iota
  58. // StateClosed when circuit breaker closed, request allowed, the breaker
  59. // calc the succeed ratio, if request num greater request setting and
  60. // ratio lower than the setting ratio, then reset state to open.
  61. StateClosed
  62. // StateHalfopen when circuit breaker open, after slepp some duration, allow
  63. // one request, but not state closed.
  64. StateHalfopen
  65. //_switchOn int32 = iota
  66. // _switchOff
  67. )
  68. var (
  69. _mu sync.RWMutex
  70. _conf = &Config{
  71. Window: xtime.Duration(3 * time.Second),
  72. Bucket: 10,
  73. Request: 100,
  74. Sleep: xtime.Duration(500 * time.Millisecond),
  75. Ratio: 0.5,
  76. // Percentage of failures must be lower than 33.33%
  77. K: 1.5,
  78. // Pattern: "",
  79. }
  80. _group = NewGroup(_conf)
  81. )
  82. // Init init global breaker config, also can reload config after first time call.
  83. func Init(conf *Config) {
  84. if conf == nil {
  85. return
  86. }
  87. _mu.Lock()
  88. _conf = conf
  89. _mu.Unlock()
  90. }
  91. // Go runs your function while tracking the breaker state of default group.
  92. func Go(name string, run, fallback func() error) error {
  93. breaker := _group.Get(name)
  94. if err := breaker.Allow(); err != nil {
  95. return fallback()
  96. }
  97. return run()
  98. }
  99. // newBreaker new a breaker.
  100. func newBreaker(c *Config) (b Breaker) {
  101. // factory
  102. return newSRE(c)
  103. }
  104. // NewGroup new a breaker group container, if conf nil use default conf.
  105. func NewGroup(conf *Config) *Group {
  106. if conf == nil {
  107. _mu.RLock()
  108. conf = _conf
  109. _mu.RUnlock()
  110. } else {
  111. conf.fix()
  112. }
  113. return &Group{
  114. conf: conf,
  115. brks: make(map[string]Breaker),
  116. }
  117. }
  118. // Get get a breaker by a specified key, if breaker not exists then make a new one.
  119. func (g *Group) Get(key string) Breaker {
  120. g.mu.RLock()
  121. brk, ok := g.brks[key]
  122. conf := g.conf
  123. g.mu.RUnlock()
  124. if ok {
  125. return brk
  126. }
  127. // NOTE here may new multi breaker for rarely case, let gc drop it.
  128. brk = newBreaker(conf)
  129. g.mu.Lock()
  130. if _, ok = g.brks[key]; !ok {
  131. g.brks[key] = brk
  132. }
  133. g.mu.Unlock()
  134. return brk
  135. }
  136. // Reload reload the group by specified config, this may let all inner breaker
  137. // reset to a new one.
  138. func (g *Group) Reload(conf *Config) {
  139. if conf == nil {
  140. return
  141. }
  142. conf.fix()
  143. g.mu.Lock()
  144. g.conf = conf
  145. g.brks = make(map[string]Breaker, len(g.brks))
  146. g.mu.Unlock()
  147. }
  148. // Go runs your function while tracking the breaker state of group.
  149. func (g *Group) Go(name string, run, fallback func() error) error {
  150. breaker := g.Get(name)
  151. if err := breaker.Allow(); err != nil {
  152. return fallback()
  153. }
  154. return run()
  155. }