prom.go 8.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308
  1. package prom
  2. import (
  3. "math/rand"
  4. "runtime"
  5. "sync"
  6. "time"
  7. "go-common/app/interface/openplatform/monitor-end/conf"
  8. "github.com/prometheus/client_golang/prometheus"
  9. )
  10. const (
  11. _department = "open"
  12. _typeCommon = "Common"
  13. _typeDetailed = "Detailed"
  14. _NaN = "NaN"
  15. )
  16. var (
  17. // HTTPClientSum HTTP Client request cost sum.
  18. HTTPClientSum *Prom
  19. // HTTPClientCount HTTP Client request count.
  20. HTTPClientCount *Prom
  21. // HTTPClientCode HTTP Client request server code count.
  22. HTTPClientCode *Prom
  23. // HTTPClientStatus HTTP Client request status.
  24. HTTPClientStatus *Prom
  25. // HTTPClientSummary HTTP Client request quantiles.
  26. HTTPClientSummary *Prom
  27. )
  28. var c *conf.Config
  29. // Prom struct info
  30. type Prom struct {
  31. timer *prometheus.HistogramVec
  32. counter *prometheus.CounterVec
  33. state *prometheus.GaugeVec
  34. summary *prometheus.SummaryVec
  35. }
  36. // Init .
  37. func Init(ce *conf.Config) {
  38. c = ce
  39. NewProm(true)
  40. go clearMemory()
  41. }
  42. func clearMemory() {
  43. for {
  44. // default 512MB accloc
  45. var limit = uint64(512 * 1024 * 1024)
  46. memStat := new(runtime.MemStats)
  47. runtime.ReadMemStats(memStat)
  48. used := memStat.Alloc
  49. if c.Prom.Limit > 0 && c.Prom.Limit < 3072 {
  50. limit = uint64(c.Prom.Limit * 1024 * 1024)
  51. }
  52. if used > limit {
  53. NewProm(false)
  54. }
  55. time.Sleep(time.Minute)
  56. }
  57. }
  58. // NewProm .
  59. func NewProm(isFirst bool) {
  60. if isFirst {
  61. HTTPClientSum = New().WithCounter("http_client_sum", []string{"target", "client_app", "department", "type", "method", "event", "version", "detail"})
  62. HTTPClientCount = New().WithCounter("http_client_count", []string{"target", "client_app", "department", "type", "method", "event", "version", "detail"})
  63. HTTPClientCode = New().WithCounter("http_client_code", []string{"target", "client_app", "department", "type", "method", "event", "version", "code"})
  64. HTTPClientStatus = New().WithCounter("http_client_status", []string{"target", "client_app", "department", "type", "method", "event", "version", "status"})
  65. HTTPClientSummary = New().WithQuantile("http_client_summary", []string{"target", "client_app", "department", "type", "method", "event", "version", "detail"})
  66. return
  67. }
  68. var mutex sync.Mutex
  69. mutex.Lock()
  70. HTTPClientSum.Unregister()
  71. HTTPClientSum = New().WithCounter("http_client_sum", []string{"target", "client_app", "department", "type", "method", "event", "version", "detail"})
  72. HTTPClientCount.Unregister()
  73. HTTPClientCount = New().WithCounter("http_client_count", []string{"target", "client_app", "department", "type", "method", "event", "version", "detail"})
  74. HTTPClientCode.Unregister()
  75. HTTPClientCode = New().WithCounter("http_client_code", []string{"target", "client_app", "department", "type", "method", "event", "version", "code"})
  76. HTTPClientStatus.Unregister()
  77. HTTPClientStatus = New().WithCounter("http_client_status", []string{"target", "client_app", "department", "type", "method", "event", "version", "status"})
  78. HTTPClientSummary.Unregister()
  79. HTTPClientSummary = New().WithQuantile("http_client_summary", []string{"target", "client_app", "department", "type", "method", "event", "version", "detail"})
  80. mutex.Unlock()
  81. }
  82. // New creates a Prom instance.
  83. func New() *Prom {
  84. return &Prom{}
  85. }
  86. // WithTimer with summary timer
  87. func (p *Prom) WithTimer(name string, labels []string) *Prom {
  88. if p == nil || p.timer != nil {
  89. return p
  90. }
  91. p.timer = prometheus.NewHistogramVec(
  92. prometheus.HistogramOpts{
  93. Name: name,
  94. Help: name,
  95. }, labels)
  96. prometheus.MustRegister(p.timer)
  97. return p
  98. }
  99. // WithCounter sets counter.
  100. func (p *Prom) WithCounter(name string, labels []string) *Prom {
  101. if p == nil || p.counter != nil {
  102. return p
  103. }
  104. p.counter = prometheus.NewCounterVec(
  105. prometheus.CounterOpts{
  106. Name: name,
  107. Help: name,
  108. }, labels)
  109. prometheus.MustRegister(p.counter)
  110. return p
  111. }
  112. // WithState sets state.
  113. func (p *Prom) WithState(name string, labels []string) *Prom {
  114. if p == nil || p.state != nil {
  115. return p
  116. }
  117. p.state = prometheus.NewGaugeVec(
  118. prometheus.GaugeOpts{
  119. Name: name,
  120. Help: name,
  121. }, labels)
  122. prometheus.MustRegister(p.state)
  123. return p
  124. }
  125. // WithQuantile sets quantiles.
  126. func (p *Prom) WithQuantile(name string, labels []string) *Prom {
  127. if p == nil || p.summary != nil {
  128. return p
  129. }
  130. p.summary = prometheus.NewSummaryVec(
  131. prometheus.SummaryOpts{
  132. Name: name,
  133. Help: name,
  134. }, labels)
  135. prometheus.MustRegister(p.summary)
  136. return p
  137. }
  138. // Unregister .
  139. func (p *Prom) Unregister() {
  140. if p.counter != nil {
  141. prometheus.Unregister(p.counter)
  142. }
  143. if p.state != nil {
  144. prometheus.Unregister(p.state)
  145. }
  146. if p.timer != nil {
  147. prometheus.Unregister(p.timer)
  148. }
  149. if p.summary != nil {
  150. prometheus.Unregister(p.summary)
  151. }
  152. }
  153. // Timing log timing information (in milliseconds) without sampling
  154. func (p *Prom) Timing(name string, time int64, extra ...string) {
  155. label := append([]string{name}, extra...)
  156. if p.timer != nil {
  157. p.timer.WithLabelValues(label...).Observe(float64(time))
  158. }
  159. }
  160. // Incr increments one stat counter without sampling
  161. func (p *Prom) Incr(name string, extra ...string) {
  162. label := append([]string{name}, extra...)
  163. if p.counter != nil {
  164. p.counter.WithLabelValues(label...).Inc()
  165. }
  166. if p.state != nil {
  167. p.state.WithLabelValues(label...).Inc()
  168. }
  169. }
  170. // Decr decrements one stat counter without sampling
  171. func (p *Prom) Decr(name string, extra ...string) {
  172. if p.state != nil {
  173. label := append([]string{name}, extra...)
  174. p.state.WithLabelValues(label...).Dec()
  175. }
  176. }
  177. // State set state
  178. func (p *Prom) State(name string, v int64, extra ...string) {
  179. if p.state != nil {
  180. label := append([]string{name}, extra...)
  181. p.state.WithLabelValues(label...).Set(float64(v))
  182. }
  183. }
  184. // Add add count v must > 0
  185. func (p *Prom) Add(name string, v int64, extra ...string) {
  186. label := append([]string{name}, extra...)
  187. if p.counter != nil {
  188. p.counter.WithLabelValues(label...).Add(float64(v))
  189. }
  190. if p.state != nil {
  191. p.state.WithLabelValues(label...).Add(float64(v))
  192. }
  193. }
  194. // AddCommonLog .
  195. func AddCommonLog(target string, app string, method string, event string, version string, v int64) {
  196. if HTTPClientCount.counter == nil || HTTPClientSum.counter == nil || HTTPClientSummary.summary == nil {
  197. return
  198. }
  199. if target == "" || app == "" || method == "" || event == "" {
  200. return
  201. }
  202. var ok = true
  203. if c.Prom.Factor > 0 && c.Prom.Factor < 100 {
  204. if rand.Intn(100) >= c.Prom.Factor {
  205. ok = false
  206. }
  207. }
  208. if version == "" {
  209. version = _NaN
  210. }
  211. // default 1ms per request
  212. i := float64(v)
  213. if i <= 0 {
  214. i = 1
  215. }
  216. labels := []string{target, app, _department, _typeCommon, method, event, version, _NaN}
  217. // labels := append(label, _NaN)
  218. HTTPClientCount.counter.WithLabelValues(labels...).Inc()
  219. HTTPClientSum.counter.WithLabelValues(labels...).Add(i)
  220. if ok {
  221. HTTPClientSummary.summary.WithLabelValues(labels...).Observe(i)
  222. }
  223. }
  224. // AddDetailedLog .
  225. func AddDetailedLog(target string, app string, method string, event string, version string, details map[string]int64) {
  226. if HTTPClientCount.counter == nil || HTTPClientSum.counter == nil || HTTPClientSummary.summary == nil {
  227. return
  228. }
  229. if target == "" || app == "" || method == "" || event == "" || details == nil {
  230. return
  231. }
  232. var ok = true
  233. if c.Prom.Factor > 0 && c.Prom.Factor < 100 {
  234. if rand.Intn(100) >= c.Prom.Factor {
  235. ok = false
  236. }
  237. }
  238. if version == "" {
  239. version = _NaN
  240. }
  241. label := []string{target, app, _department, _typeDetailed, method, event, version}
  242. for k, v := range details {
  243. labels := append(label, k)
  244. // default 1ms per request
  245. i := float64(v)
  246. if i <= 0 {
  247. i = 1
  248. }
  249. HTTPClientCount.counter.WithLabelValues(labels...).Inc()
  250. HTTPClientSum.counter.WithLabelValues(labels...).Add(i)
  251. if ok {
  252. HTTPClientSummary.summary.WithLabelValues(labels...).Observe(i)
  253. }
  254. }
  255. }
  256. // AddHTTPCode .
  257. func AddHTTPCode(target string, app string, method string, event string, version string, code string) {
  258. if HTTPClientStatus.counter == nil {
  259. return
  260. }
  261. if target == "" || app == "" || method == "" || event == "" || code == "" {
  262. return
  263. }
  264. if version == "" {
  265. version = _NaN
  266. }
  267. label := []string{target, app, _department, _typeCommon, method, event, version, code}
  268. HTTPClientStatus.counter.WithLabelValues(label...).Inc()
  269. }
  270. // AddCode .
  271. func AddCode(target string, app string, method string, event string, version string, code string) {
  272. if HTTPClientCode.counter == nil {
  273. return
  274. }
  275. if target == "" || app == "" || method == "" || event == "" || code == "" {
  276. return
  277. }
  278. if version == "" {
  279. version = _NaN
  280. }
  281. label := []string{target, app, _department, _typeCommon, method, event, version, code}
  282. HTTPClientCode.counter.WithLabelValues(label...).Inc()
  283. }