mirror of
				https://github.com/openimsdk/open-im-server.git
				synced 2025-10-26 05:02:11 +08:00 
			
		
		
		
	refactor: gin prometheus for api (#1371)
Signed-off-by: rfyiamcool <rfyiamcool@163.com>
This commit is contained in:
		
							parent
							
								
									2496a16a88
								
							
						
					
					
						commit
						65c1c412da
					
				| @ -3,7 +3,7 @@ package ginprometheus | |||||||
| import ( | import ( | ||||||
| 	"bytes" | 	"bytes" | ||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"io/ioutil" | 	"io" | ||||||
| 	"net/http" | 	"net/http" | ||||||
| 	"os" | 	"os" | ||||||
| 	"strconv" | 	"strconv" | ||||||
| @ -17,15 +17,16 @@ import ( | |||||||
| var defaultMetricPath = "/metrics" | var defaultMetricPath = "/metrics" | ||||||
| 
 | 
 | ||||||
| // counter, counter_vec, gauge, gauge_vec, | // counter, counter_vec, gauge, gauge_vec, | ||||||
| // histogram, histogram_vec, summary, summary_vec | // histogram, histogram_vec, summary, summary_vec. | ||||||
| var reqCnt = &Metric{ | var ( | ||||||
|  | 	reqCounter = &Metric{ | ||||||
| 		ID:          "reqCnt", | 		ID:          "reqCnt", | ||||||
| 		Name:        "requests_total", | 		Name:        "requests_total", | ||||||
| 		Description: "How many HTTP requests processed, partitioned by status code and HTTP method.", | 		Description: "How many HTTP requests processed, partitioned by status code and HTTP method.", | ||||||
| 		Type:        "counter_vec", | 		Type:        "counter_vec", | ||||||
| 		Args:        []string{"code", "method", "handler", "host", "url"}} | 		Args:        []string{"code", "method", "handler", "host", "url"}} | ||||||
| 
 | 
 | ||||||
| var reqDur = &Metric{ | 	reqDuration = &Metric{ | ||||||
| 		ID:          "reqDur", | 		ID:          "reqDur", | ||||||
| 		Name:        "request_duration_seconds", | 		Name:        "request_duration_seconds", | ||||||
| 		Description: "The HTTP request latencies in seconds.", | 		Description: "The HTTP request latencies in seconds.", | ||||||
| @ -33,24 +34,25 @@ var reqDur = &Metric{ | |||||||
| 		Args:        []string{"code", "method", "url"}, | 		Args:        []string{"code", "method", "url"}, | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| var resSz = &Metric{ | 	resSize = &Metric{ | ||||||
| 		ID:          "resSz", | 		ID:          "resSz", | ||||||
| 		Name:        "response_size_bytes", | 		Name:        "response_size_bytes", | ||||||
| 		Description: "The HTTP response sizes in bytes.", | 		Description: "The HTTP response sizes in bytes.", | ||||||
| 		Type:        "summary"} | 		Type:        "summary"} | ||||||
| 
 | 
 | ||||||
| var reqSz = &Metric{ | 	reqSize = &Metric{ | ||||||
| 		ID:          "reqSz", | 		ID:          "reqSz", | ||||||
| 		Name:        "request_size_bytes", | 		Name:        "request_size_bytes", | ||||||
| 		Description: "The HTTP request sizes in bytes.", | 		Description: "The HTTP request sizes in bytes.", | ||||||
| 		Type:        "summary"} | 		Type:        "summary"} | ||||||
| 
 | 
 | ||||||
| var standardMetrics = []*Metric{ | 	standardMetrics = []*Metric{ | ||||||
| 	reqCnt, | 		reqCounter, | ||||||
| 	reqDur, | 		reqDuration, | ||||||
| 	resSz, | 		resSize, | ||||||
| 	reqSz, | 		reqSize, | ||||||
| 	} | 	} | ||||||
|  | ) | ||||||
| 
 | 
 | ||||||
| /* | /* | ||||||
| RequestCounterURLLabelMappingFn is a function which can be supplied to the middleware to control | RequestCounterURLLabelMappingFn is a function which can be supplied to the middleware to control | ||||||
| @ -74,7 +76,7 @@ which would map "/customer/alice" and "/customer/bob" to their template "/custom | |||||||
| type RequestCounterURLLabelMappingFn func(c *gin.Context) string | type RequestCounterURLLabelMappingFn func(c *gin.Context) string | ||||||
| 
 | 
 | ||||||
| // Metric is a definition for the name, description, type, ID, and | // Metric is a definition for the name, description, type, ID, and | ||||||
| // prometheus.Collector type (i.e. CounterVec, Summary, etc) of each metric | // prometheus.Collector type (i.e. CounterVec, Summary, etc) of each metric. | ||||||
| type Metric struct { | type Metric struct { | ||||||
| 	MetricCollector prometheus.Collector | 	MetricCollector prometheus.Collector | ||||||
| 	ID              string | 	ID              string | ||||||
| @ -84,7 +86,7 @@ type Metric struct { | |||||||
| 	Args            []string | 	Args            []string | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // Prometheus contains the metrics gathered by the instance and its path | // Prometheus contains the metrics gathered by the instance and its path. | ||||||
| type Prometheus struct { | type Prometheus struct { | ||||||
| 	reqCnt        *prometheus.CounterVec | 	reqCnt        *prometheus.CounterVec | ||||||
| 	reqDur        *prometheus.HistogramVec | 	reqDur        *prometheus.HistogramVec | ||||||
| @ -102,7 +104,7 @@ type Prometheus struct { | |||||||
| 	URLLabelFromContext string | 	URLLabelFromContext string | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // PrometheusPushGateway contains the configuration for pushing to a Prometheus pushgateway (optional) | // PrometheusPushGateway contains the configuration for pushing to a Prometheus pushgateway (optional). | ||||||
| type PrometheusPushGateway struct { | type PrometheusPushGateway struct { | ||||||
| 
 | 
 | ||||||
| 	// Push interval in seconds | 	// Push interval in seconds | ||||||
| @ -112,7 +114,7 @@ type PrometheusPushGateway struct { | |||||||
| 	// where JOBNAME can be any string of your choice | 	// where JOBNAME can be any string of your choice | ||||||
| 	PushGatewayURL string | 	PushGatewayURL string | ||||||
| 
 | 
 | ||||||
| 	// Local metrics URL where metrics are fetched from, this could be ommited in the future | 	// Local metrics URL where metrics are fetched from, this could be omitted in the future | ||||||
| 	// if implemented using prometheus common/expfmt instead | 	// if implemented using prometheus common/expfmt instead | ||||||
| 	MetricsURL string | 	MetricsURL string | ||||||
| 
 | 
 | ||||||
| @ -120,9 +122,11 @@ type PrometheusPushGateway struct { | |||||||
| 	Job string | 	Job string | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // NewPrometheus generates a new set of metrics with a certain subsystem name | // NewPrometheus generates a new set of metrics with a certain subsystem name. | ||||||
| func NewPrometheus(subsystem string, customMetricsList ...[]*Metric) *Prometheus { | func NewPrometheus(subsystem string, customMetricsList ...[]*Metric) *Prometheus { | ||||||
|  | 	if subsystem == "" { | ||||||
| 		subsystem = "app" | 		subsystem = "app" | ||||||
|  | 	} | ||||||
| 
 | 
 | ||||||
| 	var metricsList []*Metric | 	var metricsList []*Metric | ||||||
| 
 | 
 | ||||||
| @ -131,16 +135,13 @@ func NewPrometheus(subsystem string, customMetricsList ...[]*Metric) *Prometheus | |||||||
| 	} else if len(customMetricsList) == 1 { | 	} else if len(customMetricsList) == 1 { | ||||||
| 		metricsList = customMetricsList[0] | 		metricsList = customMetricsList[0] | ||||||
| 	} | 	} | ||||||
| 
 | 	metricsList = append(metricsList, standardMetrics...) | ||||||
| 	for _, metric := range standardMetrics { |  | ||||||
| 		metricsList = append(metricsList, metric) |  | ||||||
| 	} |  | ||||||
| 
 | 
 | ||||||
| 	p := &Prometheus{ | 	p := &Prometheus{ | ||||||
| 		MetricsList: metricsList, | 		MetricsList: metricsList, | ||||||
| 		MetricsPath: defaultMetricPath, | 		MetricsPath: defaultMetricPath, | ||||||
| 		ReqCntURLLabelMappingFn: func(c *gin.Context) string { | 		ReqCntURLLabelMappingFn: func(c *gin.Context) string { | ||||||
| 			return c.Request.URL.Path | 			return c.FullPath() // e.g. /user/:id , /user/:id/info | ||||||
| 		}, | 		}, | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| @ -150,7 +151,7 @@ func NewPrometheus(subsystem string, customMetricsList ...[]*Metric) *Prometheus | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // SetPushGateway sends metrics to a remote pushgateway exposed on pushGatewayURL | // SetPushGateway sends metrics to a remote pushgateway exposed on pushGatewayURL | ||||||
| // every pushIntervalSeconds. Metrics are fetched from metricsURL | // every pushIntervalSeconds. Metrics are fetched from metricsURL. | ||||||
| func (p *Prometheus) SetPushGateway(pushGatewayURL, metricsURL string, pushIntervalSeconds time.Duration) { | func (p *Prometheus) SetPushGateway(pushGatewayURL, metricsURL string, pushIntervalSeconds time.Duration) { | ||||||
| 	p.Ppg.PushGatewayURL = pushGatewayURL | 	p.Ppg.PushGatewayURL = pushGatewayURL | ||||||
| 	p.Ppg.MetricsURL = metricsURL | 	p.Ppg.MetricsURL = metricsURL | ||||||
| @ -158,13 +159,13 @@ func (p *Prometheus) SetPushGateway(pushGatewayURL, metricsURL string, pushInter | |||||||
| 	p.startPushTicker() | 	p.startPushTicker() | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // SetPushGatewayJob job name, defaults to "gin" | // SetPushGatewayJob job name, defaults to "gin". | ||||||
| func (p *Prometheus) SetPushGatewayJob(j string) { | func (p *Prometheus) SetPushGatewayJob(j string) { | ||||||
| 	p.Ppg.Job = j | 	p.Ppg.Job = j | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // SetListenAddress for exposing metrics on address. If not set, it will be exposed at the | // SetListenAddress for exposing metrics on address. If not set, it will be exposed at the | ||||||
| // same address of the gin engine that is being used | // same address of the gin engine that is being used. | ||||||
| func (p *Prometheus) SetListenAddress(address string) { | func (p *Prometheus) SetListenAddress(address string) { | ||||||
| 	p.listenAddress = address | 	p.listenAddress = address | ||||||
| 	if p.listenAddress != "" { | 	if p.listenAddress != "" { | ||||||
| @ -181,7 +182,7 @@ func (p *Prometheus) SetListenAddressWithRouter(listenAddress string, r *gin.Eng | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // SetMetricsPath set metrics paths | // SetMetricsPath set metrics paths. | ||||||
| func (p *Prometheus) SetMetricsPath(e *gin.Engine) { | func (p *Prometheus) SetMetricsPath(e *gin.Engine) { | ||||||
| 
 | 
 | ||||||
| 	if p.listenAddress != "" { | 	if p.listenAddress != "" { | ||||||
| @ -192,7 +193,7 @@ func (p *Prometheus) SetMetricsPath(e *gin.Engine) { | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // SetMetricsPathWithAuth set metrics paths with authentication | // SetMetricsPathWithAuth set metrics paths with authentication. | ||||||
| func (p *Prometheus) SetMetricsPathWithAuth(e *gin.Engine, accounts gin.Accounts) { | func (p *Prometheus) SetMetricsPathWithAuth(e *gin.Engine, accounts gin.Accounts) { | ||||||
| 
 | 
 | ||||||
| 	if p.listenAddress != "" { | 	if p.listenAddress != "" { | ||||||
| @ -205,34 +206,43 @@ func (p *Prometheus) SetMetricsPathWithAuth(e *gin.Engine, accounts gin.Accounts | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (p *Prometheus) runServer() { | func (p *Prometheus) runServer() { | ||||||
| 	if p.listenAddress != "" { |  | ||||||
| 	go p.router.Run(p.listenAddress) | 	go p.router.Run(p.listenAddress) | ||||||
| } | } | ||||||
| } |  | ||||||
| 
 | 
 | ||||||
| func (p *Prometheus) getMetrics() []byte { | func (p *Prometheus) getMetrics() []byte { | ||||||
| 	response, _ := http.Get(p.Ppg.MetricsURL) | 	response, err := http.Get(p.Ppg.MetricsURL) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
| 
 | 
 | ||||||
| 	defer response.Body.Close() | 	defer response.Body.Close() | ||||||
| 	body, _ := ioutil.ReadAll(response.Body) |  | ||||||
| 
 | 
 | ||||||
|  | 	body, _ := io.ReadAll(response.Body) | ||||||
| 	return body | 	return body | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | var hostname, _ = os.Hostname() | ||||||
|  | 
 | ||||||
| func (p *Prometheus) getPushGatewayURL() string { | func (p *Prometheus) getPushGatewayURL() string { | ||||||
| 	h, _ := os.Hostname() |  | ||||||
| 	if p.Ppg.Job == "" { | 	if p.Ppg.Job == "" { | ||||||
| 		p.Ppg.Job = "gin" | 		p.Ppg.Job = "gin" | ||||||
| 	} | 	} | ||||||
| 	return p.Ppg.PushGatewayURL + "/metrics/job/" + p.Ppg.Job + "/instance/" + h | 	return p.Ppg.PushGatewayURL + "/metrics/job/" + p.Ppg.Job + "/instance/" + hostname | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (p *Prometheus) sendMetricsToPushGateway(metrics []byte) { | func (p *Prometheus) sendMetricsToPushGateway(metrics []byte) { | ||||||
| 	req, err := http.NewRequest("POST", p.getPushGatewayURL(), bytes.NewBuffer(metrics)) | 	req, err := http.NewRequest("POST", p.getPushGatewayURL(), bytes.NewBuffer(metrics)) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	client := &http.Client{} | 	client := &http.Client{} | ||||||
| 	if _, err = client.Do(req); err != nil { | 	resp, err := client.Do(req) | ||||||
|  | 	if err != nil { | ||||||
| 		fmt.Println("Error sending to push gateway error:", err.Error()) | 		fmt.Println("Error sending to push gateway error:", err.Error()) | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
|  | 	resp.Body.Close() | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (p *Prometheus) startPushTicker() { | func (p *Prometheus) startPushTicker() { | ||||||
| @ -244,7 +254,7 @@ func (p *Prometheus) startPushTicker() { | |||||||
| 	}() | 	}() | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // NewMetric associates prometheus.Collector based on Metric.Type | // NewMetric associates prometheus.Collector based on Metric.Type. | ||||||
| func NewMetric(m *Metric, subsystem string) prometheus.Collector { | func NewMetric(m *Metric, subsystem string) prometheus.Collector { | ||||||
| 	var metric prometheus.Collector | 	var metric prometheus.Collector | ||||||
| 	switch m.Type { | 	switch m.Type { | ||||||
| @ -321,20 +331,20 @@ func NewMetric(m *Metric, subsystem string) prometheus.Collector { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (p *Prometheus) registerMetrics(subsystem string) { | func (p *Prometheus) registerMetrics(subsystem string) { | ||||||
| 
 |  | ||||||
| 	for _, metricDef := range p.MetricsList { | 	for _, metricDef := range p.MetricsList { | ||||||
| 		metric := NewMetric(metricDef, subsystem) | 		metric := NewMetric(metricDef, subsystem) | ||||||
| 		if err := prometheus.Register(metric); err != nil { | 		if err := prometheus.Register(metric); err != nil { | ||||||
| 			fmt.Println("could not be registered in Prometheus,metricDef.Name:", metricDef.Name, "   error:", err.Error()) | 			fmt.Println("could not be registered in Prometheus,metricDef.Name:", metricDef.Name, "   error:", err.Error()) | ||||||
| 		} | 		} | ||||||
|  | 
 | ||||||
| 		switch metricDef { | 		switch metricDef { | ||||||
| 		case reqCnt: | 		case reqCounter: | ||||||
| 			p.reqCnt = metric.(*prometheus.CounterVec) | 			p.reqCnt = metric.(*prometheus.CounterVec) | ||||||
| 		case reqDur: | 		case reqDuration: | ||||||
| 			p.reqDur = metric.(*prometheus.HistogramVec) | 			p.reqDur = metric.(*prometheus.HistogramVec) | ||||||
| 		case resSz: | 		case resSize: | ||||||
| 			p.resSz = metric.(prometheus.Summary) | 			p.resSz = metric.(prometheus.Summary) | ||||||
| 		case reqSz: | 		case reqSize: | ||||||
| 			p.reqSz = metric.(prometheus.Summary) | 			p.reqSz = metric.(prometheus.Summary) | ||||||
| 		} | 		} | ||||||
| 		metricDef.MetricCollector = metric | 		metricDef.MetricCollector = metric | ||||||
| @ -353,7 +363,7 @@ func (p *Prometheus) UseWithAuth(e *gin.Engine, accounts gin.Accounts) { | |||||||
| 	p.SetMetricsPathWithAuth(e, accounts) | 	p.SetMetricsPathWithAuth(e, accounts) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // HandlerFunc defines handler function for middleware | // HandlerFunc defines handler function for middleware. | ||||||
| func (p *Prometheus) HandlerFunc() gin.HandlerFunc { | func (p *Prometheus) HandlerFunc() gin.HandlerFunc { | ||||||
| 	return func(c *gin.Context) { | 	return func(c *gin.Context) { | ||||||
| 		if c.Request.URL.Path == p.MetricsPath { | 		if c.Request.URL.Path == p.MetricsPath { | ||||||
| @ -393,7 +403,7 @@ func prometheusHandler() gin.HandlerFunc { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func computeApproximateRequestSize(r *http.Request) int { | func computeApproximateRequestSize(r *http.Request) int { | ||||||
| 	s := 0 | 	var s int | ||||||
| 	if r.URL != nil { | 	if r.URL != nil { | ||||||
| 		s = len(r.URL.Path) | 		s = len(r.URL.Path) | ||||||
| 	} | 	} | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user