From a79a9b6cbb77859ba57e6ba9ffdce93723f20e9f Mon Sep 17 00:00:00 2001 From: Bangseungjae Date: Sat, 12 Oct 2024 14:48:23 +0900 Subject: [PATCH 1/5] #828 Feature Prometheus API --- api/prometheus/prometheus.go | 359 +++++++++++------- api/restapi/configure_loxilb_rest_api.go | 3 + api/restapi/embedded_spec.go | 220 +++++++++++ api/restapi/handler/prometheus.go | 28 ++ api/restapi/operations/loxilb_rest_api_api.go | 38 +- api/swagger.yml | 72 ++++ 6 files changed, 578 insertions(+), 142 deletions(-) diff --git a/api/prometheus/prometheus.go b/api/prometheus/prometheus.go index 2e3d8e255..f24932917 100644 --- a/api/prometheus/prometheus.go +++ b/api/prometheus/prometheus.go @@ -16,7 +16,11 @@ package prometheus import ( + "context" "fmt" + "github.com/go-openapi/errors" + "github.com/loxilb-io/loxilb/options" + "net/http" "strings" "sync" "time" @@ -46,6 +50,8 @@ var ( PromethusDefaultPeriod = 10 * time.Second PromethusPartialPeriod = (PromethusDefaultPeriod / 6) PromethusLongPeriod = (PromethusDefaultPeriod * 600) // To reset Period + prometheusCtx context.Context + prometheusCancel context.CancelFunc activeConntrackCount = promauto.NewGauge( prometheus.GaugeOpts{ Name: "active_conntrack_count", @@ -143,215 +149,286 @@ func PrometheusRegister(hook cmn.NetHookInterface) { } func Init() { + prometheusCtx, prometheusCancel = context.WithCancel(context.Background()) + // Make Conntrack Statistic map ConntrackStats = make(map[ConntrackKey]Stats) mutex = &sync.Mutex{} - go RunGetConntrack() - go RunGetEndpoint() - go RunActiveConntrackCount() - go RunHostCount() - go RunProcessedStatistic() - go RunNewFlowCount() - go RunResetCounts() - go RunGetLBRule() - go RunLcusCalculator() + go RunGetConntrack(prometheusCtx) + go RunGetEndpoint(prometheusCtx) + go RunActiveConntrackCount(prometheusCtx) + go RunHostCount(prometheusCtx) + go RunProcessedStatistic(prometheusCtx) + go RunNewFlowCount(prometheusCtx) + go RunResetCounts(prometheusCtx) + go RunGetLBRule(prometheusCtx) + go RunLcusCalculator(prometheusCtx) +} + +func Off() error { + if !options.Opts.Prometheus { + return errors.New(http.StatusBadRequest, "already prometheus turned off") + } + options.Opts.Prometheus = false + prometheusCancel() + return nil +} + +func TurnOn() error { + if options.Opts.Prometheus { + return errors.New(http.StatusBadRequest, "already prometheus turned on") + } + options.Opts.Prometheus = true + Init() + return nil } func MakeConntrackKey(c cmn.CtInfo) (key ConntrackKey) { return ConntrackKey(fmt.Sprintf("%s|%05d|%s|%05d|%v", c.Sip, c.Sport, c.Dip, c.Dport, c.Proto)) } -func RunResetCounts() { +func RunResetCounts(ctx context.Context) { for { // Statistic reset time.Sleep(PromethusLongPeriod) - mutex.Lock() - ConntrackStats = map[ConntrackKey]Stats{} - mutex.Unlock() + select { + case <-ctx.Done(): + return + default: + mutex.Lock() + ConntrackStats = map[ConntrackKey]Stats{} + mutex.Unlock() + } } } -func RunGetConntrack() { +func RunGetConntrack(ctx context.Context) { for { - mutex.Lock() - ConntrackInfo, err = hooks.NetCtInfoGet() - if err != nil { - tk.LogIt(tk.LogDebug, "[Prometheus] Error occur : %v\n", err) - } - - for _, ct := range ConntrackInfo { - k := MakeConntrackKey(ct) - var tmpStats Stats - _, ok := ConntrackStats[k] - if ok { - tmpStats = Stats{ - Bytes: ConntrackStats[k].Bytes + ct.Bytes, - Packets: ConntrackStats[k].Packets + ct.Pkts, - } - } else { - tmpStats = Stats{ - Bytes: ct.Bytes, - Packets: ct.Pkts, - } + select { + case <-ctx.Done(): + return + default: + mutex.Lock() + ConntrackInfo, err = hooks.NetCtInfoGet() + if err != nil { + tk.LogIt(tk.LogDebug, "[Prometheus] Error occur : %v\n", err) } - ConntrackStats[k] = tmpStats + for _, ct := range ConntrackInfo { + k := MakeConntrackKey(ct) + var tmpStats Stats + _, ok := ConntrackStats[k] + if ok { + tmpStats = Stats{ + Bytes: ConntrackStats[k].Bytes + ct.Bytes, + Packets: ConntrackStats[k].Packets + ct.Pkts, + } + } else { + tmpStats = Stats{ + Bytes: ct.Bytes, + Packets: ct.Pkts, + } + } + ConntrackStats[k] = tmpStats + } + mutex.Unlock() } - mutex.Unlock() + time.Sleep(PromethusDefaultPeriod) } } -func RunGetEndpoint() { +func RunGetEndpoint(ctx context.Context) { for { - mutex.Lock() - EndPointInfo, err = hooks.NetEpHostGet() - if err != nil { - tk.LogIt(tk.LogDebug, "[Prometheus] Error occur : %v\n", err) + select { + case <-ctx.Done(): + return + default: + mutex.Lock() + EndPointInfo, err = hooks.NetEpHostGet() + if err != nil { + tk.LogIt(tk.LogDebug, "[Prometheus] Error occur : %v\n", err) + } + mutex.Unlock() } - mutex.Unlock() + time.Sleep(PromethusDefaultPeriod) } } -func RunGetLBRule() { +func RunGetLBRule(ctx context.Context) { for { - mutex.Lock() - LBRuleInfo, err = hooks.NetLbRuleGet() - if err != nil { - tk.LogIt(tk.LogDebug, "[Prometheus] Error occur : %v\n", err) + select { + case <-ctx.Done(): + return + default: + mutex.Lock() + LBRuleInfo, err = hooks.NetLbRuleGet() + if err != nil { + tk.LogIt(tk.LogDebug, "[Prometheus] Error occur : %v\n", err) + } + ruleCount.Set(float64(len(LBRuleInfo))) + mutex.Unlock() } - ruleCount.Set(float64(len(LBRuleInfo))) - mutex.Unlock() + time.Sleep(PromethusDefaultPeriod) } } -func RunActiveConntrackCount() { +func RunActiveConntrackCount(ctx context.Context) { for { - mutex.Lock() - // init Counts - activeFlowCountTcp.Set(0) - activeFlowCountUdp.Set(0) - activeFlowCountSctp.Set(0) - inActiveFlowCount.Set(0) + select { + case <-ctx.Done(): + return + default: + mutex.Lock() + // init Counts + activeFlowCountTcp.Set(0) + activeFlowCountUdp.Set(0) + activeFlowCountSctp.Set(0) + inActiveFlowCount.Set(0) - // Total flow count - activeConntrackCount.Set(float64(len(ConntrackInfo))) + // Total flow count + activeConntrackCount.Set(float64(len(ConntrackInfo))) - for _, ct := range ConntrackInfo { - // TCP flow count - if ct.Proto == "tcp" { - activeFlowCountTcp.Inc() - } - // UDP flow count - if ct.Proto == "udp" { - activeFlowCountUdp.Inc() - } - // SCTP flow count - if ct.Proto == "sctp" { - activeFlowCountSctp.Inc() - } - // Closed flow count - if ct.CState == "closed" { - inActiveFlowCount.Inc() + for _, ct := range ConntrackInfo { + // TCP flow count + if ct.Proto == "tcp" { + activeFlowCountTcp.Inc() + } + // UDP flow count + if ct.Proto == "udp" { + activeFlowCountUdp.Inc() + } + // SCTP flow count + if ct.Proto == "sctp" { + activeFlowCountSctp.Inc() + } + // Closed flow count + if ct.CState == "closed" { + inActiveFlowCount.Inc() + } } + mutex.Unlock() } - mutex.Unlock() + time.Sleep(PromethusDefaultPeriod) } } -func RunHostCount() { +func RunHostCount(ctx context.Context) { for { - mutex.Lock() - healthyHostCount.Set(0) - unHealthyHostCount.Set(0) - for _, ep := range EndPointInfo { - if ep.CurrState == "ok" { - healthyHostCount.Inc() - } - if ep.CurrState == "nok" { - unHealthyHostCount.Inc() + select { + case <-ctx.Done(): + return + default: + mutex.Lock() + healthyHostCount.Set(0) + unHealthyHostCount.Set(0) + for _, ep := range EndPointInfo { + if ep.CurrState == "ok" { + healthyHostCount.Inc() + } + if ep.CurrState == "nok" { + unHealthyHostCount.Inc() + } } + mutex.Unlock() } - mutex.Unlock() + time.Sleep(PromethusDefaultPeriod) } } -func RunProcessedStatistic() { +func RunProcessedStatistic(ctx context.Context) { for { - mutex.Lock() - // Init Stats - processedPackets.Set(0) - processedBytes.Set(0) - processedTCPBytes.Set(0) - processedUDPBytes.Set(0) - processedSCTPBytes.Set(0) - for k, ct := range ConntrackStats { - if strings.Contains(string(k), "tcp") { - processedTCPBytes.Add(float64(ct.Bytes)) - } - if strings.Contains(string(k), "udp") { - processedUDPBytes.Add(float64(ct.Bytes)) - } - if strings.Contains(string(k), "sctp") { - processedSCTPBytes.Add(float64(ct.Bytes)) + select { + case <-ctx.Done(): + return + default: + mutex.Lock() + // Init Stats + processedPackets.Set(0) + processedBytes.Set(0) + processedTCPBytes.Set(0) + processedUDPBytes.Set(0) + processedSCTPBytes.Set(0) + for k, ct := range ConntrackStats { + if strings.Contains(string(k), "tcp") { + processedTCPBytes.Add(float64(ct.Bytes)) + } + if strings.Contains(string(k), "udp") { + processedUDPBytes.Add(float64(ct.Bytes)) + } + if strings.Contains(string(k), "sctp") { + processedSCTPBytes.Add(float64(ct.Bytes)) + } + processedPackets.Add(float64(ct.Packets)) + processedBytes.Add(float64(ct.Bytes)) } - processedPackets.Add(float64(ct.Packets)) - processedBytes.Add(float64(ct.Bytes)) + mutex.Unlock() } - mutex.Unlock() + time.Sleep(PromethusDefaultPeriod) } } -func RunNewFlowCount() { +func RunNewFlowCount(ctx context.Context) { PreFlowCounts = 0 for { - mutex.Lock() - // Total new flow count - CurrentFlowCounts := len(ConntrackInfo) - diff := CurrentFlowCounts - PreFlowCounts - if diff > 0 { - newFlowCount.Set(float64(diff)) - } else { - newFlowCount.Set(0) + select { + case <-ctx.Done(): + return + default: + mutex.Lock() + // Total new flow count + CurrentFlowCounts := len(ConntrackInfo) + diff := CurrentFlowCounts - PreFlowCounts + if diff > 0 { + newFlowCount.Set(float64(diff)) + } else { + newFlowCount.Set(0) + } + PreFlowCounts = CurrentFlowCounts + mutex.Unlock() } - PreFlowCounts = CurrentFlowCounts - mutex.Unlock() + time.Sleep(PromethusDefaultPeriod) } } -func RunLcusCalculator() { +func RunLcusCalculator(ctx context.Context) { for { time.Sleep(PromethusDefaultPeriod) - mutex.Lock() - var LCUNewFlowCount = &dto.Metric{} - var LCUActiveFlowCount = &dto.Metric{} - var LCURuleCount = &dto.Metric{} - var LCUProcessedBytes = &dto.Metric{} - if err := newFlowCount.Write(LCUNewFlowCount); err != nil { - tk.LogIt(tk.LogDebug, "[Prometheus] Error occur : %v\n", err) - } - if err := activeConntrackCount.Write(LCUActiveFlowCount); err != nil { - tk.LogIt(tk.LogDebug, "[Prometheus] Error occur : %v\n", err) - } - if err := ruleCount.Write(LCURuleCount); err != nil { - tk.LogIt(tk.LogDebug, "[Prometheus] Error occur : %v\n", err) - } - if err := processedBytes.Write(LCUProcessedBytes); err != nil { - tk.LogIt(tk.LogDebug, "[Prometheus] Error occur : %v\n", err) + select { + case <-ctx.Done(): + return + default: + mutex.Lock() + var LCUNewFlowCount = &dto.Metric{} + var LCUActiveFlowCount = &dto.Metric{} + var LCURuleCount = &dto.Metric{} + var LCUProcessedBytes = &dto.Metric{} + if err := newFlowCount.Write(LCUNewFlowCount); err != nil { + tk.LogIt(tk.LogDebug, "[Prometheus] Error occur : %v\n", err) + } + if err := activeConntrackCount.Write(LCUActiveFlowCount); err != nil { + tk.LogIt(tk.LogDebug, "[Prometheus] Error occur : %v\n", err) + } + if err := ruleCount.Write(LCURuleCount); err != nil { + tk.LogIt(tk.LogDebug, "[Prometheus] Error occur : %v\n", err) + } + if err := processedBytes.Write(LCUProcessedBytes); err != nil { + tk.LogIt(tk.LogDebug, "[Prometheus] Error occur : %v\n", err) + } + // LCU of accumulated Flow count = Flowcount / 2160000 + // LCU of Rule = ruleCount/1000 + // LCU of Byte = processedBytes(Gb)/1h + consumedLcus.Set(float64(len(ConntrackStats))/2160000 + + *LCURuleCount.Gauge.Value/1000 + + (*LCUProcessedBytes.Gauge.Value*8)/360000000000) // (byte * 8)/ (60*60*1G)/10 + mutex.Unlock() } - // LCU of accumulated Flow count = Flowcount / 2160000 - // LCU of Rule = ruleCount/1000 - // LCU of Byte = processedBytes(Gb)/1h - consumedLcus.Set(float64(len(ConntrackStats))/2160000 + - *LCURuleCount.Gauge.Value/1000 + - (*LCUProcessedBytes.Gauge.Value*8)/360000000000) // (byte * 8)/ (60*60*1G)/10 - mutex.Unlock() } } diff --git a/api/restapi/configure_loxilb_rest_api.go b/api/restapi/configure_loxilb_rest_api.go index 8d2735c45..1db9e93e1 100644 --- a/api/restapi/configure_loxilb_rest_api.go +++ b/api/restapi/configure_loxilb_rest_api.go @@ -158,6 +158,9 @@ func configureAPI(api *operations.LoxilbRestAPIAPI) http.Handler { // Prometheus api.GetMetricsHandler = operations.GetMetricsHandlerFunc(handler.ConfigGetPrometheusCounter) + api.GetConfigMetricsHandler = operations.GetConfigMetricsHandlerFunc(handler.ConfigGetPrometheusOption) + api.PostConfigMetricsHandler = operations.PostConfigMetricsHandlerFunc(handler.ConfigPostPrometheus) + api.DeleteConfigMetricsHandler = operations.DeleteConfigMetricsHandlerFunc(handler.ConfigDeletePrometheus) // BGP Peer api.GetConfigBgpNeighAllHandler = operations.GetConfigBgpNeighAllHandlerFunc(handler.ConfigGetBGPNeigh) diff --git a/api/restapi/embedded_spec.go b/api/restapi/embedded_spec.go index 2ea838773..7925fbcac 100644 --- a/api/restapi/embedded_spec.go +++ b/api/restapi/embedded_spec.go @@ -2281,6 +2281,107 @@ func init() { } } }, + "/config/metrics": { + "get": { + "summary": "Get prometheus config value", + "responses": { + "200": { + "description": "prometheus config value", + "schema": { + "$ref": "#/definitions/MetricsConfig" + } + }, + "400": { + "description": "Malformed arguments for API call", + "schema": { + "$ref": "#/definitions/Error" + } + }, + "401": { + "description": "Invalid authentication credentials", + "schema": { + "$ref": "#/definitions/Error" + } + }, + "500": { + "description": "Internal service error", + "schema": { + "$ref": "#/definitions/Error" + } + }, + "503": { + "description": "Maintenance mode", + "schema": { + "$ref": "#/definitions/Error" + } + } + } + }, + "post": { + "summary": "turn on prometheus option", + "responses": { + "204": { + "description": "OK" + }, + "400": { + "description": "Malformed arguments for API call", + "schema": { + "$ref": "#/definitions/Error" + } + }, + "401": { + "description": "Invalid authentication credentials", + "schema": { + "$ref": "#/definitions/Error" + } + }, + "500": { + "description": "Internal service error", + "schema": { + "$ref": "#/definitions/Error" + } + }, + "503": { + "description": "Maintenance mode", + "schema": { + "$ref": "#/definitions/Error" + } + } + } + }, + "delete": { + "summary": "turn off prometheus option", + "responses": { + "204": { + "description": "OK" + }, + "400": { + "description": "Malformed arguments for API call", + "schema": { + "$ref": "#/definitions/Error" + } + }, + "401": { + "description": "Invalid authentication credentials", + "schema": { + "$ref": "#/definitions/Error" + } + }, + "500": { + "description": "Internal service error", + "schema": { + "$ref": "#/definitions/Error" + } + }, + "503": { + "description": "Maintenance mode", + "schema": { + "$ref": "#/definitions/Error" + } + } + } + } + }, "/config/mirror": { "post": { "description": "Create a new Mirror config.", @@ -5105,6 +5206,15 @@ func init() { } } }, + "MetricsConfig": { + "type": "object", + "properties": { + "prometheus": { + "description": "value for prometheus enable or not", + "type": "boolean" + } + } + }, "MirrorEntry": { "type": "object", "properties": { @@ -7992,6 +8102,107 @@ func init() { } } }, + "/config/metrics": { + "get": { + "summary": "Get prometheus config value", + "responses": { + "200": { + "description": "prometheus config value", + "schema": { + "$ref": "#/definitions/MetricsConfig" + } + }, + "400": { + "description": "Malformed arguments for API call", + "schema": { + "$ref": "#/definitions/Error" + } + }, + "401": { + "description": "Invalid authentication credentials", + "schema": { + "$ref": "#/definitions/Error" + } + }, + "500": { + "description": "Internal service error", + "schema": { + "$ref": "#/definitions/Error" + } + }, + "503": { + "description": "Maintenance mode", + "schema": { + "$ref": "#/definitions/Error" + } + } + } + }, + "post": { + "summary": "turn on prometheus option", + "responses": { + "204": { + "description": "OK" + }, + "400": { + "description": "Malformed arguments for API call", + "schema": { + "$ref": "#/definitions/Error" + } + }, + "401": { + "description": "Invalid authentication credentials", + "schema": { + "$ref": "#/definitions/Error" + } + }, + "500": { + "description": "Internal service error", + "schema": { + "$ref": "#/definitions/Error" + } + }, + "503": { + "description": "Maintenance mode", + "schema": { + "$ref": "#/definitions/Error" + } + } + } + }, + "delete": { + "summary": "turn off prometheus option", + "responses": { + "204": { + "description": "OK" + }, + "400": { + "description": "Malformed arguments for API call", + "schema": { + "$ref": "#/definitions/Error" + } + }, + "401": { + "description": "Invalid authentication credentials", + "schema": { + "$ref": "#/definitions/Error" + } + }, + "500": { + "description": "Internal service error", + "schema": { + "$ref": "#/definitions/Error" + } + }, + "503": { + "description": "Maintenance mode", + "schema": { + "$ref": "#/definitions/Error" + } + } + } + } + }, "/config/mirror": { "post": { "description": "Create a new Mirror config.", @@ -11378,6 +11589,15 @@ func init() { } } }, + "MetricsConfig": { + "type": "object", + "properties": { + "prometheus": { + "description": "value for prometheus enable or not", + "type": "boolean" + } + } + }, "MirrorEntry": { "type": "object", "properties": { diff --git a/api/restapi/handler/prometheus.go b/api/restapi/handler/prometheus.go index 603a9a855..6fc1940b0 100644 --- a/api/restapi/handler/prometheus.go +++ b/api/restapi/handler/prometheus.go @@ -16,6 +16,8 @@ package handler import ( + "github.com/loxilb-io/loxilb/api/models" + "github.com/loxilb-io/loxilb/api/prometheus" "net/http" "github.com/go-openapi/runtime" @@ -35,3 +37,29 @@ func ConfigGetPrometheusCounter(params operations.GetMetricsParams) middleware.R promhttp.Handler().ServeHTTP(w, params.HTTPRequest) }) } + +func ConfigGetPrometheusOption(params operations.GetConfigMetricsParams) middleware.Responder { + tk.LogIt(tk.LogDebug, "[API] Prometheus %s API called. url : %s\n", params.HTTPRequest.Method, params.HTTPRequest.URL) + return operations.NewGetConfigMetricsOK().WithPayload(&models.MetricsConfig{Prometheus: options.Opts.Prometheus}) +} + +func ConfigPostPrometheus(params operations.PostConfigMetricsParams) middleware.Responder { + tk.LogIt(tk.LogDebug, "[API] Prometheus %s API called. url : %s\n", params.HTTPRequest.Method, params.HTTPRequest.URL) + err := prometheus.TurnOn() + if err != nil { + tk.LogIt(tk.LogDebug, "[API] Error occur : %v\n", err) + return &ResultResponse{Result: err.Error()} + } + return &ResultResponse{Result: "Success"} +} + +func ConfigDeletePrometheus(params operations.DeleteConfigMetricsParams) middleware.Responder { + tk.LogIt(tk.LogDebug, "[API] Prometheus %s API called. url : %s\n", params.HTTPRequest.Method, params.HTTPRequest.URL) + err := prometheus.Off() + if err != nil { + tk.LogIt(tk.LogDebug, "[API] Error occur : %v\n", err) + return &ResultResponse{Result: err.Error()} + } + + return &ResultResponse{Result: "Success"} +} diff --git a/api/restapi/operations/loxilb_rest_api_api.go b/api/restapi/operations/loxilb_rest_api_api.go index 4d0909b75..fa9d67095 100644 --- a/api/restapi/operations/loxilb_rest_api_api.go +++ b/api/restapi/operations/loxilb_rest_api_api.go @@ -81,6 +81,9 @@ func NewLoxilbRestAPIAPI(spec *loads.Document) *LoxilbRestAPIAPI { DeleteConfigLoadbalancerNameLbNameHandler: DeleteConfigLoadbalancerNameLbNameHandlerFunc(func(params DeleteConfigLoadbalancerNameLbNameParams) middleware.Responder { return middleware.NotImplemented("operation DeleteConfigLoadbalancerNameLbName has not yet been implemented") }), + DeleteConfigMetricsHandler: DeleteConfigMetricsHandlerFunc(func(params DeleteConfigMetricsParams) middleware.Responder { + return middleware.NotImplemented("operation DeleteConfigMetrics has not yet been implemented") + }), DeleteConfigMirrorIdentIdentHandler: DeleteConfigMirrorIdentIdentHandlerFunc(func(params DeleteConfigMirrorIdentIdentParams) middleware.Responder { return middleware.NotImplemented("operation DeleteConfigMirrorIdentIdent has not yet been implemented") }), @@ -144,6 +147,9 @@ func NewLoxilbRestAPIAPI(spec *loads.Document) *LoxilbRestAPIAPI { GetConfigLoadbalancerAllHandler: GetConfigLoadbalancerAllHandlerFunc(func(params GetConfigLoadbalancerAllParams) middleware.Responder { return middleware.NotImplemented("operation GetConfigLoadbalancerAll has not yet been implemented") }), + GetConfigMetricsHandler: GetConfigMetricsHandlerFunc(func(params GetConfigMetricsParams) middleware.Responder { + return middleware.NotImplemented("operation GetConfigMetrics has not yet been implemented") + }), GetConfigMirrorAllHandler: GetConfigMirrorAllHandlerFunc(func(params GetConfigMirrorAllParams) middleware.Responder { return middleware.NotImplemented("operation GetConfigMirrorAll has not yet been implemented") }), @@ -222,6 +228,9 @@ func NewLoxilbRestAPIAPI(spec *loads.Document) *LoxilbRestAPIAPI { PostConfigLoadbalancerHandler: PostConfigLoadbalancerHandlerFunc(func(params PostConfigLoadbalancerParams) middleware.Responder { return middleware.NotImplemented("operation PostConfigLoadbalancer has not yet been implemented") }), + PostConfigMetricsHandler: PostConfigMetricsHandlerFunc(func(params PostConfigMetricsParams) middleware.Responder { + return middleware.NotImplemented("operation PostConfigMetrics has not yet been implemented") + }), PostConfigMirrorHandler: PostConfigMirrorHandlerFunc(func(params PostConfigMirrorParams) middleware.Responder { return middleware.NotImplemented("operation PostConfigMirror has not yet been implemented") }), @@ -317,6 +326,8 @@ type LoxilbRestAPIAPI struct { DeleteConfigLoadbalancerHosturlHosturlExternalipaddressIPAddressPortPortProtocolProtoHandler DeleteConfigLoadbalancerHosturlHosturlExternalipaddressIPAddressPortPortProtocolProtoHandler // DeleteConfigLoadbalancerNameLbNameHandler sets the operation handler for the delete config loadbalancer name lb name operation DeleteConfigLoadbalancerNameLbNameHandler DeleteConfigLoadbalancerNameLbNameHandler + // DeleteConfigMetricsHandler sets the operation handler for the delete config metrics operation + DeleteConfigMetricsHandler DeleteConfigMetricsHandler // DeleteConfigMirrorIdentIdentHandler sets the operation handler for the delete config mirror ident ident operation DeleteConfigMirrorIdentIdentHandler DeleteConfigMirrorIdentIdentHandler // DeleteConfigNeighborIPAddressDevIfNameHandler sets the operation handler for the delete config neighbor IP address dev if name operation @@ -359,6 +370,8 @@ type LoxilbRestAPIAPI struct { GetConfigIpv4addressAllHandler GetConfigIpv4addressAllHandler // GetConfigLoadbalancerAllHandler sets the operation handler for the get config loadbalancer all operation GetConfigLoadbalancerAllHandler GetConfigLoadbalancerAllHandler + // GetConfigMetricsHandler sets the operation handler for the get config metrics operation + GetConfigMetricsHandler GetConfigMetricsHandler // GetConfigMirrorAllHandler sets the operation handler for the get config mirror all operation GetConfigMirrorAllHandler GetConfigMirrorAllHandler // GetConfigNeighborAllHandler sets the operation handler for the get config neighbor all operation @@ -411,6 +424,8 @@ type LoxilbRestAPIAPI struct { PostConfigIpv4addressHandler PostConfigIpv4addressHandler // PostConfigLoadbalancerHandler sets the operation handler for the post config loadbalancer operation PostConfigLoadbalancerHandler PostConfigLoadbalancerHandler + // PostConfigMetricsHandler sets the operation handler for the post config metrics operation + PostConfigMetricsHandler PostConfigMetricsHandler // PostConfigMirrorHandler sets the operation handler for the post config mirror operation PostConfigMirrorHandler PostConfigMirrorHandler // PostConfigNeighborHandler sets the operation handler for the post config neighbor operation @@ -549,6 +564,9 @@ func (o *LoxilbRestAPIAPI) Validate() error { if o.DeleteConfigLoadbalancerNameLbNameHandler == nil { unregistered = append(unregistered, "DeleteConfigLoadbalancerNameLbNameHandler") } + if o.DeleteConfigMetricsHandler == nil { + unregistered = append(unregistered, "DeleteConfigMetricsHandler") + } if o.DeleteConfigMirrorIdentIdentHandler == nil { unregistered = append(unregistered, "DeleteConfigMirrorIdentIdentHandler") } @@ -612,6 +630,9 @@ func (o *LoxilbRestAPIAPI) Validate() error { if o.GetConfigLoadbalancerAllHandler == nil { unregistered = append(unregistered, "GetConfigLoadbalancerAllHandler") } + if o.GetConfigMetricsHandler == nil { + unregistered = append(unregistered, "GetConfigMetricsHandler") + } if o.GetConfigMirrorAllHandler == nil { unregistered = append(unregistered, "GetConfigMirrorAllHandler") } @@ -690,6 +711,9 @@ func (o *LoxilbRestAPIAPI) Validate() error { if o.PostConfigLoadbalancerHandler == nil { unregistered = append(unregistered, "PostConfigLoadbalancerHandler") } + if o.PostConfigMetricsHandler == nil { + unregistered = append(unregistered, "PostConfigMetricsHandler") + } if o.PostConfigMirrorHandler == nil { unregistered = append(unregistered, "PostConfigMirrorHandler") } @@ -866,6 +890,10 @@ func (o *LoxilbRestAPIAPI) initHandlerCache() { if o.handlers["DELETE"] == nil { o.handlers["DELETE"] = make(map[string]http.Handler) } + o.handlers["DELETE"]["/config/metrics"] = NewDeleteConfigMetrics(o.context, o.DeleteConfigMetricsHandler) + if o.handlers["DELETE"] == nil { + o.handlers["DELETE"] = make(map[string]http.Handler) + } o.handlers["DELETE"]["/config/mirror/ident/{ident}"] = NewDeleteConfigMirrorIdentIdent(o.context, o.DeleteConfigMirrorIdentIdentHandler) if o.handlers["DELETE"] == nil { o.handlers["DELETE"] = make(map[string]http.Handler) @@ -950,6 +978,10 @@ func (o *LoxilbRestAPIAPI) initHandlerCache() { if o.handlers["GET"] == nil { o.handlers["GET"] = make(map[string]http.Handler) } + o.handlers["GET"]["/config/metrics"] = NewGetConfigMetrics(o.context, o.GetConfigMetricsHandler) + if o.handlers["GET"] == nil { + o.handlers["GET"] = make(map[string]http.Handler) + } o.handlers["GET"]["/config/mirror/all"] = NewGetConfigMirrorAll(o.context, o.GetConfigMirrorAllHandler) if o.handlers["GET"] == nil { o.handlers["GET"] = make(map[string]http.Handler) @@ -1054,6 +1086,10 @@ func (o *LoxilbRestAPIAPI) initHandlerCache() { if o.handlers["POST"] == nil { o.handlers["POST"] = make(map[string]http.Handler) } + o.handlers["POST"]["/config/metrics"] = NewPostConfigMetrics(o.context, o.PostConfigMetricsHandler) + if o.handlers["POST"] == nil { + o.handlers["POST"] = make(map[string]http.Handler) + } o.handlers["POST"]["/config/mirror"] = NewPostConfigMirror(o.context, o.PostConfigMirrorHandler) if o.handlers["POST"] == nil { o.handlers["POST"] = make(map[string]http.Handler) @@ -1136,6 +1172,6 @@ func (o *LoxilbRestAPIAPI) AddMiddlewareFor(method, path string, builder middlew } o.Init() if h, ok := o.handlers[um][path]; ok { - o.handlers[method][path] = builder(h) + o.handlers[um][path] = builder(h) } } diff --git a/api/swagger.yml b/api/swagger.yml index 2e930bfef..98ebff548 100644 --- a/api/swagger.yml +++ b/api/swagger.yml @@ -2742,6 +2742,72 @@ paths: description: Metrics in prometheus text format schema: type: string + '/config/metrics': + get: + summary: Get prometheus config value + responses: + '200': + description: prometheus config value + schema: + $ref: '#/definitions/MetricsConfig' + '400': + description: Malformed arguments for API call + schema: + $ref: '#/definitions/Error' + '401': + description: Invalid authentication credentials + schema: + $ref: '#/definitions/Error' + '500': + description: Internal service error + schema: + $ref: '#/definitions/Error' + '503': + description: Maintenance mode + schema: + $ref: '#/definitions/Error' + post: + summary: turn on prometheus option + responses: + '204': + description: OK + '400': + description: Malformed arguments for API call + schema: + $ref: '#/definitions/Error' + '401': + description: Invalid authentication credentials + schema: + $ref: '#/definitions/Error' + '500': + description: Internal service error + schema: + $ref: '#/definitions/Error' + '503': + description: Maintenance mode + schema: + $ref: '#/definitions/Error' + delete: + summary: turn off prometheus option + responses: + '204': + description: OK + '400': + description: Malformed arguments for API call + schema: + $ref: '#/definitions/Error' + '401': + description: Invalid authentication credentials + schema: + $ref: '#/definitions/Error' + '500': + description: Internal service error + schema: + $ref: '#/definitions/Error' + '503': + description: Maintenance mode + schema: + $ref: '#/definitions/Error' #---------------------------------------------- # BFD @@ -4032,3 +4098,9 @@ definitions: type: integer format: uint8 description: Retry Count to detect failure + MetricsConfig: + type: object + properties: + prometheus: + type: boolean + description: value for prometheus enable or not From effdf7cc3d020d06009b36480e0288c6195bdc88 Mon Sep 17 00:00:00 2001 From: Bangseungjae Date: Sat, 12 Oct 2024 14:59:12 +0900 Subject: [PATCH 2/5] #828 Feature Prometheus API Signed-off-by: Bangseungjae --- .../post_config_metrics_urlbuilder.go | 87 +++++++++++++++++++ 1 file changed, 87 insertions(+) create mode 100644 api/restapi/operations/post_config_metrics_urlbuilder.go diff --git a/api/restapi/operations/post_config_metrics_urlbuilder.go b/api/restapi/operations/post_config_metrics_urlbuilder.go new file mode 100644 index 000000000..51660a76d --- /dev/null +++ b/api/restapi/operations/post_config_metrics_urlbuilder.go @@ -0,0 +1,87 @@ +// Code generated by go-swagger; DO NOT EDIT. + +package operations + +// This file was generated by the swagger tool. +// Editing this file might prove futile when you re-run the generate command + +import ( + "errors" + "net/url" + golangswaggerpaths "path" +) + +// PostConfigMetricsURL generates an URL for the post config metrics operation +type PostConfigMetricsURL struct { + _basePath string +} + +// WithBasePath sets the base path for this url builder, only required when it's different from the +// base path specified in the swagger spec. +// When the value of the base path is an empty string +func (o *PostConfigMetricsURL) WithBasePath(bp string) *PostConfigMetricsURL { + o.SetBasePath(bp) + return o +} + +// SetBasePath sets the base path for this url builder, only required when it's different from the +// base path specified in the swagger spec. +// When the value of the base path is an empty string +func (o *PostConfigMetricsURL) SetBasePath(bp string) { + o._basePath = bp +} + +// Build a url path and query string +func (o *PostConfigMetricsURL) Build() (*url.URL, error) { + var _result url.URL + + var _path = "/config/metrics" + + _basePath := o._basePath + if _basePath == "" { + _basePath = "/netlox/v1" + } + _result.Path = golangswaggerpaths.Join(_basePath, _path) + + return &_result, nil +} + +// Must is a helper function to panic when the url builder returns an error +func (o *PostConfigMetricsURL) Must(u *url.URL, err error) *url.URL { + if err != nil { + panic(err) + } + if u == nil { + panic("url can't be nil") + } + return u +} + +// String returns the string representation of the path with query string +func (o *PostConfigMetricsURL) String() string { + return o.Must(o.Build()).String() +} + +// BuildFull builds a full url with scheme, host, path and query string +func (o *PostConfigMetricsURL) BuildFull(scheme, host string) (*url.URL, error) { + if scheme == "" { + return nil, errors.New("scheme is required for a full url on PostConfigMetricsURL") + } + if host == "" { + return nil, errors.New("host is required for a full url on PostConfigMetricsURL") + } + + base, err := o.Build() + if err != nil { + return nil, err + } + + base.Scheme = scheme + base.Host = host + return base, nil +} + +// StringFull returns the string representation of a complete url +func (o *PostConfigMetricsURL) StringFull(scheme, host string) string { + return o.Must(o.BuildFull(scheme, host)).String() +} From b3e6f582ab0cf939567a47368f710747068d09ff Mon Sep 17 00:00:00 2001 From: Bangseungjae Date: Sat, 12 Oct 2024 14:59:18 +0900 Subject: [PATCH 3/5] Revert "#828 Feature Prometheus API" This reverts commit a79a9b6cbb77859ba57e6ba9ffdce93723f20e9f. --- api/prometheus/prometheus.go | 359 +++++++----------- api/restapi/configure_loxilb_rest_api.go | 3 - api/restapi/embedded_spec.go | 220 ----------- api/restapi/handler/prometheus.go | 28 -- api/restapi/operations/loxilb_rest_api_api.go | 38 +- api/swagger.yml | 72 ---- 6 files changed, 142 insertions(+), 578 deletions(-) diff --git a/api/prometheus/prometheus.go b/api/prometheus/prometheus.go index f24932917..2e3d8e255 100644 --- a/api/prometheus/prometheus.go +++ b/api/prometheus/prometheus.go @@ -16,11 +16,7 @@ package prometheus import ( - "context" "fmt" - "github.com/go-openapi/errors" - "github.com/loxilb-io/loxilb/options" - "net/http" "strings" "sync" "time" @@ -50,8 +46,6 @@ var ( PromethusDefaultPeriod = 10 * time.Second PromethusPartialPeriod = (PromethusDefaultPeriod / 6) PromethusLongPeriod = (PromethusDefaultPeriod * 600) // To reset Period - prometheusCtx context.Context - prometheusCancel context.CancelFunc activeConntrackCount = promauto.NewGauge( prometheus.GaugeOpts{ Name: "active_conntrack_count", @@ -149,286 +143,215 @@ func PrometheusRegister(hook cmn.NetHookInterface) { } func Init() { - prometheusCtx, prometheusCancel = context.WithCancel(context.Background()) - // Make Conntrack Statistic map ConntrackStats = make(map[ConntrackKey]Stats) mutex = &sync.Mutex{} - go RunGetConntrack(prometheusCtx) - go RunGetEndpoint(prometheusCtx) - go RunActiveConntrackCount(prometheusCtx) - go RunHostCount(prometheusCtx) - go RunProcessedStatistic(prometheusCtx) - go RunNewFlowCount(prometheusCtx) - go RunResetCounts(prometheusCtx) - go RunGetLBRule(prometheusCtx) - go RunLcusCalculator(prometheusCtx) -} - -func Off() error { - if !options.Opts.Prometheus { - return errors.New(http.StatusBadRequest, "already prometheus turned off") - } - options.Opts.Prometheus = false - prometheusCancel() - return nil -} - -func TurnOn() error { - if options.Opts.Prometheus { - return errors.New(http.StatusBadRequest, "already prometheus turned on") - } - options.Opts.Prometheus = true - Init() - return nil + go RunGetConntrack() + go RunGetEndpoint() + go RunActiveConntrackCount() + go RunHostCount() + go RunProcessedStatistic() + go RunNewFlowCount() + go RunResetCounts() + go RunGetLBRule() + go RunLcusCalculator() } func MakeConntrackKey(c cmn.CtInfo) (key ConntrackKey) { return ConntrackKey(fmt.Sprintf("%s|%05d|%s|%05d|%v", c.Sip, c.Sport, c.Dip, c.Dport, c.Proto)) } -func RunResetCounts(ctx context.Context) { +func RunResetCounts() { for { // Statistic reset time.Sleep(PromethusLongPeriod) - select { - case <-ctx.Done(): - return - default: - mutex.Lock() - ConntrackStats = map[ConntrackKey]Stats{} - mutex.Unlock() - } + mutex.Lock() + ConntrackStats = map[ConntrackKey]Stats{} + mutex.Unlock() } } -func RunGetConntrack(ctx context.Context) { +func RunGetConntrack() { for { - select { - case <-ctx.Done(): - return - default: - mutex.Lock() - ConntrackInfo, err = hooks.NetCtInfoGet() - if err != nil { - tk.LogIt(tk.LogDebug, "[Prometheus] Error occur : %v\n", err) - } + mutex.Lock() + ConntrackInfo, err = hooks.NetCtInfoGet() + if err != nil { + tk.LogIt(tk.LogDebug, "[Prometheus] Error occur : %v\n", err) + } - for _, ct := range ConntrackInfo { - k := MakeConntrackKey(ct) - var tmpStats Stats - _, ok := ConntrackStats[k] - if ok { - tmpStats = Stats{ - Bytes: ConntrackStats[k].Bytes + ct.Bytes, - Packets: ConntrackStats[k].Packets + ct.Pkts, - } - } else { - tmpStats = Stats{ - Bytes: ct.Bytes, - Packets: ct.Pkts, - } + for _, ct := range ConntrackInfo { + k := MakeConntrackKey(ct) + var tmpStats Stats + _, ok := ConntrackStats[k] + if ok { + tmpStats = Stats{ + Bytes: ConntrackStats[k].Bytes + ct.Bytes, + Packets: ConntrackStats[k].Packets + ct.Pkts, + } + } else { + tmpStats = Stats{ + Bytes: ct.Bytes, + Packets: ct.Pkts, } - - ConntrackStats[k] = tmpStats } - mutex.Unlock() - } + ConntrackStats[k] = tmpStats + + } + mutex.Unlock() time.Sleep(PromethusDefaultPeriod) } } -func RunGetEndpoint(ctx context.Context) { +func RunGetEndpoint() { for { - select { - case <-ctx.Done(): - return - default: - mutex.Lock() - EndPointInfo, err = hooks.NetEpHostGet() - if err != nil { - tk.LogIt(tk.LogDebug, "[Prometheus] Error occur : %v\n", err) - } - mutex.Unlock() + mutex.Lock() + EndPointInfo, err = hooks.NetEpHostGet() + if err != nil { + tk.LogIt(tk.LogDebug, "[Prometheus] Error occur : %v\n", err) } - + mutex.Unlock() time.Sleep(PromethusDefaultPeriod) } } -func RunGetLBRule(ctx context.Context) { +func RunGetLBRule() { for { - select { - case <-ctx.Done(): - return - default: - mutex.Lock() - LBRuleInfo, err = hooks.NetLbRuleGet() - if err != nil { - tk.LogIt(tk.LogDebug, "[Prometheus] Error occur : %v\n", err) - } - ruleCount.Set(float64(len(LBRuleInfo))) - mutex.Unlock() + mutex.Lock() + LBRuleInfo, err = hooks.NetLbRuleGet() + if err != nil { + tk.LogIt(tk.LogDebug, "[Prometheus] Error occur : %v\n", err) } - + ruleCount.Set(float64(len(LBRuleInfo))) + mutex.Unlock() time.Sleep(PromethusDefaultPeriod) } } -func RunActiveConntrackCount(ctx context.Context) { +func RunActiveConntrackCount() { for { - select { - case <-ctx.Done(): - return - default: - mutex.Lock() - // init Counts - activeFlowCountTcp.Set(0) - activeFlowCountUdp.Set(0) - activeFlowCountSctp.Set(0) - inActiveFlowCount.Set(0) + mutex.Lock() + // init Counts + activeFlowCountTcp.Set(0) + activeFlowCountUdp.Set(0) + activeFlowCountSctp.Set(0) + inActiveFlowCount.Set(0) - // Total flow count - activeConntrackCount.Set(float64(len(ConntrackInfo))) + // Total flow count + activeConntrackCount.Set(float64(len(ConntrackInfo))) - for _, ct := range ConntrackInfo { - // TCP flow count - if ct.Proto == "tcp" { - activeFlowCountTcp.Inc() - } - // UDP flow count - if ct.Proto == "udp" { - activeFlowCountUdp.Inc() - } - // SCTP flow count - if ct.Proto == "sctp" { - activeFlowCountSctp.Inc() - } - // Closed flow count - if ct.CState == "closed" { - inActiveFlowCount.Inc() - } + for _, ct := range ConntrackInfo { + // TCP flow count + if ct.Proto == "tcp" { + activeFlowCountTcp.Inc() + } + // UDP flow count + if ct.Proto == "udp" { + activeFlowCountUdp.Inc() + } + // SCTP flow count + if ct.Proto == "sctp" { + activeFlowCountSctp.Inc() + } + // Closed flow count + if ct.CState == "closed" { + inActiveFlowCount.Inc() } - mutex.Unlock() } - + mutex.Unlock() time.Sleep(PromethusDefaultPeriod) } } -func RunHostCount(ctx context.Context) { +func RunHostCount() { for { - select { - case <-ctx.Done(): - return - default: - mutex.Lock() - healthyHostCount.Set(0) - unHealthyHostCount.Set(0) - for _, ep := range EndPointInfo { - if ep.CurrState == "ok" { - healthyHostCount.Inc() - } - if ep.CurrState == "nok" { - unHealthyHostCount.Inc() - } + mutex.Lock() + healthyHostCount.Set(0) + unHealthyHostCount.Set(0) + for _, ep := range EndPointInfo { + if ep.CurrState == "ok" { + healthyHostCount.Inc() + } + if ep.CurrState == "nok" { + unHealthyHostCount.Inc() } - mutex.Unlock() } - + mutex.Unlock() time.Sleep(PromethusDefaultPeriod) } } -func RunProcessedStatistic(ctx context.Context) { +func RunProcessedStatistic() { for { - select { - case <-ctx.Done(): - return - default: - mutex.Lock() - // Init Stats - processedPackets.Set(0) - processedBytes.Set(0) - processedTCPBytes.Set(0) - processedUDPBytes.Set(0) - processedSCTPBytes.Set(0) - for k, ct := range ConntrackStats { - if strings.Contains(string(k), "tcp") { - processedTCPBytes.Add(float64(ct.Bytes)) - } - if strings.Contains(string(k), "udp") { - processedUDPBytes.Add(float64(ct.Bytes)) - } - if strings.Contains(string(k), "sctp") { - processedSCTPBytes.Add(float64(ct.Bytes)) - } - processedPackets.Add(float64(ct.Packets)) - processedBytes.Add(float64(ct.Bytes)) + mutex.Lock() + // Init Stats + processedPackets.Set(0) + processedBytes.Set(0) + processedTCPBytes.Set(0) + processedUDPBytes.Set(0) + processedSCTPBytes.Set(0) + for k, ct := range ConntrackStats { + if strings.Contains(string(k), "tcp") { + processedTCPBytes.Add(float64(ct.Bytes)) + } + if strings.Contains(string(k), "udp") { + processedUDPBytes.Add(float64(ct.Bytes)) } - mutex.Unlock() + if strings.Contains(string(k), "sctp") { + processedSCTPBytes.Add(float64(ct.Bytes)) + } + processedPackets.Add(float64(ct.Packets)) + processedBytes.Add(float64(ct.Bytes)) } - + mutex.Unlock() time.Sleep(PromethusDefaultPeriod) } } -func RunNewFlowCount(ctx context.Context) { +func RunNewFlowCount() { PreFlowCounts = 0 for { - select { - case <-ctx.Done(): - return - default: - mutex.Lock() - // Total new flow count - CurrentFlowCounts := len(ConntrackInfo) - diff := CurrentFlowCounts - PreFlowCounts - if diff > 0 { - newFlowCount.Set(float64(diff)) - } else { - newFlowCount.Set(0) - } - PreFlowCounts = CurrentFlowCounts - mutex.Unlock() + mutex.Lock() + // Total new flow count + CurrentFlowCounts := len(ConntrackInfo) + diff := CurrentFlowCounts - PreFlowCounts + if diff > 0 { + newFlowCount.Set(float64(diff)) + } else { + newFlowCount.Set(0) } - + PreFlowCounts = CurrentFlowCounts + mutex.Unlock() time.Sleep(PromethusDefaultPeriod) } } -func RunLcusCalculator(ctx context.Context) { +func RunLcusCalculator() { for { time.Sleep(PromethusDefaultPeriod) - select { - case <-ctx.Done(): - return - default: - mutex.Lock() - var LCUNewFlowCount = &dto.Metric{} - var LCUActiveFlowCount = &dto.Metric{} - var LCURuleCount = &dto.Metric{} - var LCUProcessedBytes = &dto.Metric{} - if err := newFlowCount.Write(LCUNewFlowCount); err != nil { - tk.LogIt(tk.LogDebug, "[Prometheus] Error occur : %v\n", err) - } - if err := activeConntrackCount.Write(LCUActiveFlowCount); err != nil { - tk.LogIt(tk.LogDebug, "[Prometheus] Error occur : %v\n", err) - } - if err := ruleCount.Write(LCURuleCount); err != nil { - tk.LogIt(tk.LogDebug, "[Prometheus] Error occur : %v\n", err) - } - if err := processedBytes.Write(LCUProcessedBytes); err != nil { - tk.LogIt(tk.LogDebug, "[Prometheus] Error occur : %v\n", err) - } - // LCU of accumulated Flow count = Flowcount / 2160000 - // LCU of Rule = ruleCount/1000 - // LCU of Byte = processedBytes(Gb)/1h - consumedLcus.Set(float64(len(ConntrackStats))/2160000 + - *LCURuleCount.Gauge.Value/1000 + - (*LCUProcessedBytes.Gauge.Value*8)/360000000000) // (byte * 8)/ (60*60*1G)/10 - mutex.Unlock() + mutex.Lock() + var LCUNewFlowCount = &dto.Metric{} + var LCUActiveFlowCount = &dto.Metric{} + var LCURuleCount = &dto.Metric{} + var LCUProcessedBytes = &dto.Metric{} + if err := newFlowCount.Write(LCUNewFlowCount); err != nil { + tk.LogIt(tk.LogDebug, "[Prometheus] Error occur : %v\n", err) + } + if err := activeConntrackCount.Write(LCUActiveFlowCount); err != nil { + tk.LogIt(tk.LogDebug, "[Prometheus] Error occur : %v\n", err) + } + if err := ruleCount.Write(LCURuleCount); err != nil { + tk.LogIt(tk.LogDebug, "[Prometheus] Error occur : %v\n", err) + } + if err := processedBytes.Write(LCUProcessedBytes); err != nil { + tk.LogIt(tk.LogDebug, "[Prometheus] Error occur : %v\n", err) } + // LCU of accumulated Flow count = Flowcount / 2160000 + // LCU of Rule = ruleCount/1000 + // LCU of Byte = processedBytes(Gb)/1h + consumedLcus.Set(float64(len(ConntrackStats))/2160000 + + *LCURuleCount.Gauge.Value/1000 + + (*LCUProcessedBytes.Gauge.Value*8)/360000000000) // (byte * 8)/ (60*60*1G)/10 + mutex.Unlock() } } diff --git a/api/restapi/configure_loxilb_rest_api.go b/api/restapi/configure_loxilb_rest_api.go index 1db9e93e1..8d2735c45 100644 --- a/api/restapi/configure_loxilb_rest_api.go +++ b/api/restapi/configure_loxilb_rest_api.go @@ -158,9 +158,6 @@ func configureAPI(api *operations.LoxilbRestAPIAPI) http.Handler { // Prometheus api.GetMetricsHandler = operations.GetMetricsHandlerFunc(handler.ConfigGetPrometheusCounter) - api.GetConfigMetricsHandler = operations.GetConfigMetricsHandlerFunc(handler.ConfigGetPrometheusOption) - api.PostConfigMetricsHandler = operations.PostConfigMetricsHandlerFunc(handler.ConfigPostPrometheus) - api.DeleteConfigMetricsHandler = operations.DeleteConfigMetricsHandlerFunc(handler.ConfigDeletePrometheus) // BGP Peer api.GetConfigBgpNeighAllHandler = operations.GetConfigBgpNeighAllHandlerFunc(handler.ConfigGetBGPNeigh) diff --git a/api/restapi/embedded_spec.go b/api/restapi/embedded_spec.go index 7925fbcac..2ea838773 100644 --- a/api/restapi/embedded_spec.go +++ b/api/restapi/embedded_spec.go @@ -2281,107 +2281,6 @@ func init() { } } }, - "/config/metrics": { - "get": { - "summary": "Get prometheus config value", - "responses": { - "200": { - "description": "prometheus config value", - "schema": { - "$ref": "#/definitions/MetricsConfig" - } - }, - "400": { - "description": "Malformed arguments for API call", - "schema": { - "$ref": "#/definitions/Error" - } - }, - "401": { - "description": "Invalid authentication credentials", - "schema": { - "$ref": "#/definitions/Error" - } - }, - "500": { - "description": "Internal service error", - "schema": { - "$ref": "#/definitions/Error" - } - }, - "503": { - "description": "Maintenance mode", - "schema": { - "$ref": "#/definitions/Error" - } - } - } - }, - "post": { - "summary": "turn on prometheus option", - "responses": { - "204": { - "description": "OK" - }, - "400": { - "description": "Malformed arguments for API call", - "schema": { - "$ref": "#/definitions/Error" - } - }, - "401": { - "description": "Invalid authentication credentials", - "schema": { - "$ref": "#/definitions/Error" - } - }, - "500": { - "description": "Internal service error", - "schema": { - "$ref": "#/definitions/Error" - } - }, - "503": { - "description": "Maintenance mode", - "schema": { - "$ref": "#/definitions/Error" - } - } - } - }, - "delete": { - "summary": "turn off prometheus option", - "responses": { - "204": { - "description": "OK" - }, - "400": { - "description": "Malformed arguments for API call", - "schema": { - "$ref": "#/definitions/Error" - } - }, - "401": { - "description": "Invalid authentication credentials", - "schema": { - "$ref": "#/definitions/Error" - } - }, - "500": { - "description": "Internal service error", - "schema": { - "$ref": "#/definitions/Error" - } - }, - "503": { - "description": "Maintenance mode", - "schema": { - "$ref": "#/definitions/Error" - } - } - } - } - }, "/config/mirror": { "post": { "description": "Create a new Mirror config.", @@ -5206,15 +5105,6 @@ func init() { } } }, - "MetricsConfig": { - "type": "object", - "properties": { - "prometheus": { - "description": "value for prometheus enable or not", - "type": "boolean" - } - } - }, "MirrorEntry": { "type": "object", "properties": { @@ -8102,107 +7992,6 @@ func init() { } } }, - "/config/metrics": { - "get": { - "summary": "Get prometheus config value", - "responses": { - "200": { - "description": "prometheus config value", - "schema": { - "$ref": "#/definitions/MetricsConfig" - } - }, - "400": { - "description": "Malformed arguments for API call", - "schema": { - "$ref": "#/definitions/Error" - } - }, - "401": { - "description": "Invalid authentication credentials", - "schema": { - "$ref": "#/definitions/Error" - } - }, - "500": { - "description": "Internal service error", - "schema": { - "$ref": "#/definitions/Error" - } - }, - "503": { - "description": "Maintenance mode", - "schema": { - "$ref": "#/definitions/Error" - } - } - } - }, - "post": { - "summary": "turn on prometheus option", - "responses": { - "204": { - "description": "OK" - }, - "400": { - "description": "Malformed arguments for API call", - "schema": { - "$ref": "#/definitions/Error" - } - }, - "401": { - "description": "Invalid authentication credentials", - "schema": { - "$ref": "#/definitions/Error" - } - }, - "500": { - "description": "Internal service error", - "schema": { - "$ref": "#/definitions/Error" - } - }, - "503": { - "description": "Maintenance mode", - "schema": { - "$ref": "#/definitions/Error" - } - } - } - }, - "delete": { - "summary": "turn off prometheus option", - "responses": { - "204": { - "description": "OK" - }, - "400": { - "description": "Malformed arguments for API call", - "schema": { - "$ref": "#/definitions/Error" - } - }, - "401": { - "description": "Invalid authentication credentials", - "schema": { - "$ref": "#/definitions/Error" - } - }, - "500": { - "description": "Internal service error", - "schema": { - "$ref": "#/definitions/Error" - } - }, - "503": { - "description": "Maintenance mode", - "schema": { - "$ref": "#/definitions/Error" - } - } - } - } - }, "/config/mirror": { "post": { "description": "Create a new Mirror config.", @@ -11589,15 +11378,6 @@ func init() { } } }, - "MetricsConfig": { - "type": "object", - "properties": { - "prometheus": { - "description": "value for prometheus enable or not", - "type": "boolean" - } - } - }, "MirrorEntry": { "type": "object", "properties": { diff --git a/api/restapi/handler/prometheus.go b/api/restapi/handler/prometheus.go index 6fc1940b0..603a9a855 100644 --- a/api/restapi/handler/prometheus.go +++ b/api/restapi/handler/prometheus.go @@ -16,8 +16,6 @@ package handler import ( - "github.com/loxilb-io/loxilb/api/models" - "github.com/loxilb-io/loxilb/api/prometheus" "net/http" "github.com/go-openapi/runtime" @@ -37,29 +35,3 @@ func ConfigGetPrometheusCounter(params operations.GetMetricsParams) middleware.R promhttp.Handler().ServeHTTP(w, params.HTTPRequest) }) } - -func ConfigGetPrometheusOption(params operations.GetConfigMetricsParams) middleware.Responder { - tk.LogIt(tk.LogDebug, "[API] Prometheus %s API called. url : %s\n", params.HTTPRequest.Method, params.HTTPRequest.URL) - return operations.NewGetConfigMetricsOK().WithPayload(&models.MetricsConfig{Prometheus: options.Opts.Prometheus}) -} - -func ConfigPostPrometheus(params operations.PostConfigMetricsParams) middleware.Responder { - tk.LogIt(tk.LogDebug, "[API] Prometheus %s API called. url : %s\n", params.HTTPRequest.Method, params.HTTPRequest.URL) - err := prometheus.TurnOn() - if err != nil { - tk.LogIt(tk.LogDebug, "[API] Error occur : %v\n", err) - return &ResultResponse{Result: err.Error()} - } - return &ResultResponse{Result: "Success"} -} - -func ConfigDeletePrometheus(params operations.DeleteConfigMetricsParams) middleware.Responder { - tk.LogIt(tk.LogDebug, "[API] Prometheus %s API called. url : %s\n", params.HTTPRequest.Method, params.HTTPRequest.URL) - err := prometheus.Off() - if err != nil { - tk.LogIt(tk.LogDebug, "[API] Error occur : %v\n", err) - return &ResultResponse{Result: err.Error()} - } - - return &ResultResponse{Result: "Success"} -} diff --git a/api/restapi/operations/loxilb_rest_api_api.go b/api/restapi/operations/loxilb_rest_api_api.go index fa9d67095..4d0909b75 100644 --- a/api/restapi/operations/loxilb_rest_api_api.go +++ b/api/restapi/operations/loxilb_rest_api_api.go @@ -81,9 +81,6 @@ func NewLoxilbRestAPIAPI(spec *loads.Document) *LoxilbRestAPIAPI { DeleteConfigLoadbalancerNameLbNameHandler: DeleteConfigLoadbalancerNameLbNameHandlerFunc(func(params DeleteConfigLoadbalancerNameLbNameParams) middleware.Responder { return middleware.NotImplemented("operation DeleteConfigLoadbalancerNameLbName has not yet been implemented") }), - DeleteConfigMetricsHandler: DeleteConfigMetricsHandlerFunc(func(params DeleteConfigMetricsParams) middleware.Responder { - return middleware.NotImplemented("operation DeleteConfigMetrics has not yet been implemented") - }), DeleteConfigMirrorIdentIdentHandler: DeleteConfigMirrorIdentIdentHandlerFunc(func(params DeleteConfigMirrorIdentIdentParams) middleware.Responder { return middleware.NotImplemented("operation DeleteConfigMirrorIdentIdent has not yet been implemented") }), @@ -147,9 +144,6 @@ func NewLoxilbRestAPIAPI(spec *loads.Document) *LoxilbRestAPIAPI { GetConfigLoadbalancerAllHandler: GetConfigLoadbalancerAllHandlerFunc(func(params GetConfigLoadbalancerAllParams) middleware.Responder { return middleware.NotImplemented("operation GetConfigLoadbalancerAll has not yet been implemented") }), - GetConfigMetricsHandler: GetConfigMetricsHandlerFunc(func(params GetConfigMetricsParams) middleware.Responder { - return middleware.NotImplemented("operation GetConfigMetrics has not yet been implemented") - }), GetConfigMirrorAllHandler: GetConfigMirrorAllHandlerFunc(func(params GetConfigMirrorAllParams) middleware.Responder { return middleware.NotImplemented("operation GetConfigMirrorAll has not yet been implemented") }), @@ -228,9 +222,6 @@ func NewLoxilbRestAPIAPI(spec *loads.Document) *LoxilbRestAPIAPI { PostConfigLoadbalancerHandler: PostConfigLoadbalancerHandlerFunc(func(params PostConfigLoadbalancerParams) middleware.Responder { return middleware.NotImplemented("operation PostConfigLoadbalancer has not yet been implemented") }), - PostConfigMetricsHandler: PostConfigMetricsHandlerFunc(func(params PostConfigMetricsParams) middleware.Responder { - return middleware.NotImplemented("operation PostConfigMetrics has not yet been implemented") - }), PostConfigMirrorHandler: PostConfigMirrorHandlerFunc(func(params PostConfigMirrorParams) middleware.Responder { return middleware.NotImplemented("operation PostConfigMirror has not yet been implemented") }), @@ -326,8 +317,6 @@ type LoxilbRestAPIAPI struct { DeleteConfigLoadbalancerHosturlHosturlExternalipaddressIPAddressPortPortProtocolProtoHandler DeleteConfigLoadbalancerHosturlHosturlExternalipaddressIPAddressPortPortProtocolProtoHandler // DeleteConfigLoadbalancerNameLbNameHandler sets the operation handler for the delete config loadbalancer name lb name operation DeleteConfigLoadbalancerNameLbNameHandler DeleteConfigLoadbalancerNameLbNameHandler - // DeleteConfigMetricsHandler sets the operation handler for the delete config metrics operation - DeleteConfigMetricsHandler DeleteConfigMetricsHandler // DeleteConfigMirrorIdentIdentHandler sets the operation handler for the delete config mirror ident ident operation DeleteConfigMirrorIdentIdentHandler DeleteConfigMirrorIdentIdentHandler // DeleteConfigNeighborIPAddressDevIfNameHandler sets the operation handler for the delete config neighbor IP address dev if name operation @@ -370,8 +359,6 @@ type LoxilbRestAPIAPI struct { GetConfigIpv4addressAllHandler GetConfigIpv4addressAllHandler // GetConfigLoadbalancerAllHandler sets the operation handler for the get config loadbalancer all operation GetConfigLoadbalancerAllHandler GetConfigLoadbalancerAllHandler - // GetConfigMetricsHandler sets the operation handler for the get config metrics operation - GetConfigMetricsHandler GetConfigMetricsHandler // GetConfigMirrorAllHandler sets the operation handler for the get config mirror all operation GetConfigMirrorAllHandler GetConfigMirrorAllHandler // GetConfigNeighborAllHandler sets the operation handler for the get config neighbor all operation @@ -424,8 +411,6 @@ type LoxilbRestAPIAPI struct { PostConfigIpv4addressHandler PostConfigIpv4addressHandler // PostConfigLoadbalancerHandler sets the operation handler for the post config loadbalancer operation PostConfigLoadbalancerHandler PostConfigLoadbalancerHandler - // PostConfigMetricsHandler sets the operation handler for the post config metrics operation - PostConfigMetricsHandler PostConfigMetricsHandler // PostConfigMirrorHandler sets the operation handler for the post config mirror operation PostConfigMirrorHandler PostConfigMirrorHandler // PostConfigNeighborHandler sets the operation handler for the post config neighbor operation @@ -564,9 +549,6 @@ func (o *LoxilbRestAPIAPI) Validate() error { if o.DeleteConfigLoadbalancerNameLbNameHandler == nil { unregistered = append(unregistered, "DeleteConfigLoadbalancerNameLbNameHandler") } - if o.DeleteConfigMetricsHandler == nil { - unregistered = append(unregistered, "DeleteConfigMetricsHandler") - } if o.DeleteConfigMirrorIdentIdentHandler == nil { unregistered = append(unregistered, "DeleteConfigMirrorIdentIdentHandler") } @@ -630,9 +612,6 @@ func (o *LoxilbRestAPIAPI) Validate() error { if o.GetConfigLoadbalancerAllHandler == nil { unregistered = append(unregistered, "GetConfigLoadbalancerAllHandler") } - if o.GetConfigMetricsHandler == nil { - unregistered = append(unregistered, "GetConfigMetricsHandler") - } if o.GetConfigMirrorAllHandler == nil { unregistered = append(unregistered, "GetConfigMirrorAllHandler") } @@ -711,9 +690,6 @@ func (o *LoxilbRestAPIAPI) Validate() error { if o.PostConfigLoadbalancerHandler == nil { unregistered = append(unregistered, "PostConfigLoadbalancerHandler") } - if o.PostConfigMetricsHandler == nil { - unregistered = append(unregistered, "PostConfigMetricsHandler") - } if o.PostConfigMirrorHandler == nil { unregistered = append(unregistered, "PostConfigMirrorHandler") } @@ -890,10 +866,6 @@ func (o *LoxilbRestAPIAPI) initHandlerCache() { if o.handlers["DELETE"] == nil { o.handlers["DELETE"] = make(map[string]http.Handler) } - o.handlers["DELETE"]["/config/metrics"] = NewDeleteConfigMetrics(o.context, o.DeleteConfigMetricsHandler) - if o.handlers["DELETE"] == nil { - o.handlers["DELETE"] = make(map[string]http.Handler) - } o.handlers["DELETE"]["/config/mirror/ident/{ident}"] = NewDeleteConfigMirrorIdentIdent(o.context, o.DeleteConfigMirrorIdentIdentHandler) if o.handlers["DELETE"] == nil { o.handlers["DELETE"] = make(map[string]http.Handler) @@ -978,10 +950,6 @@ func (o *LoxilbRestAPIAPI) initHandlerCache() { if o.handlers["GET"] == nil { o.handlers["GET"] = make(map[string]http.Handler) } - o.handlers["GET"]["/config/metrics"] = NewGetConfigMetrics(o.context, o.GetConfigMetricsHandler) - if o.handlers["GET"] == nil { - o.handlers["GET"] = make(map[string]http.Handler) - } o.handlers["GET"]["/config/mirror/all"] = NewGetConfigMirrorAll(o.context, o.GetConfigMirrorAllHandler) if o.handlers["GET"] == nil { o.handlers["GET"] = make(map[string]http.Handler) @@ -1086,10 +1054,6 @@ func (o *LoxilbRestAPIAPI) initHandlerCache() { if o.handlers["POST"] == nil { o.handlers["POST"] = make(map[string]http.Handler) } - o.handlers["POST"]["/config/metrics"] = NewPostConfigMetrics(o.context, o.PostConfigMetricsHandler) - if o.handlers["POST"] == nil { - o.handlers["POST"] = make(map[string]http.Handler) - } o.handlers["POST"]["/config/mirror"] = NewPostConfigMirror(o.context, o.PostConfigMirrorHandler) if o.handlers["POST"] == nil { o.handlers["POST"] = make(map[string]http.Handler) @@ -1172,6 +1136,6 @@ func (o *LoxilbRestAPIAPI) AddMiddlewareFor(method, path string, builder middlew } o.Init() if h, ok := o.handlers[um][path]; ok { - o.handlers[um][path] = builder(h) + o.handlers[method][path] = builder(h) } } diff --git a/api/swagger.yml b/api/swagger.yml index 98ebff548..2e930bfef 100644 --- a/api/swagger.yml +++ b/api/swagger.yml @@ -2742,72 +2742,6 @@ paths: description: Metrics in prometheus text format schema: type: string - '/config/metrics': - get: - summary: Get prometheus config value - responses: - '200': - description: prometheus config value - schema: - $ref: '#/definitions/MetricsConfig' - '400': - description: Malformed arguments for API call - schema: - $ref: '#/definitions/Error' - '401': - description: Invalid authentication credentials - schema: - $ref: '#/definitions/Error' - '500': - description: Internal service error - schema: - $ref: '#/definitions/Error' - '503': - description: Maintenance mode - schema: - $ref: '#/definitions/Error' - post: - summary: turn on prometheus option - responses: - '204': - description: OK - '400': - description: Malformed arguments for API call - schema: - $ref: '#/definitions/Error' - '401': - description: Invalid authentication credentials - schema: - $ref: '#/definitions/Error' - '500': - description: Internal service error - schema: - $ref: '#/definitions/Error' - '503': - description: Maintenance mode - schema: - $ref: '#/definitions/Error' - delete: - summary: turn off prometheus option - responses: - '204': - description: OK - '400': - description: Malformed arguments for API call - schema: - $ref: '#/definitions/Error' - '401': - description: Invalid authentication credentials - schema: - $ref: '#/definitions/Error' - '500': - description: Internal service error - schema: - $ref: '#/definitions/Error' - '503': - description: Maintenance mode - schema: - $ref: '#/definitions/Error' #---------------------------------------------- # BFD @@ -4098,9 +4032,3 @@ definitions: type: integer format: uint8 description: Retry Count to detect failure - MetricsConfig: - type: object - properties: - prometheus: - type: boolean - description: value for prometheus enable or not From 7403cea45629ee23a42e0856fc1652848833ac91 Mon Sep 17 00:00:00 2001 From: Bangseungjae Date: Sat, 12 Oct 2024 15:01:39 +0900 Subject: [PATCH 4/5] Revert "Revert "#828 Feature Prometheus API"" This reverts commit b3e6f582ab0cf939567a47368f710747068d09ff. --- api/prometheus/prometheus.go | 359 +++++++++++------- api/restapi/configure_loxilb_rest_api.go | 3 + api/restapi/embedded_spec.go | 220 +++++++++++ api/restapi/handler/prometheus.go | 28 ++ api/restapi/operations/loxilb_rest_api_api.go | 38 +- api/swagger.yml | 72 ++++ 6 files changed, 578 insertions(+), 142 deletions(-) diff --git a/api/prometheus/prometheus.go b/api/prometheus/prometheus.go index 2e3d8e255..f24932917 100644 --- a/api/prometheus/prometheus.go +++ b/api/prometheus/prometheus.go @@ -16,7 +16,11 @@ package prometheus import ( + "context" "fmt" + "github.com/go-openapi/errors" + "github.com/loxilb-io/loxilb/options" + "net/http" "strings" "sync" "time" @@ -46,6 +50,8 @@ var ( PromethusDefaultPeriod = 10 * time.Second PromethusPartialPeriod = (PromethusDefaultPeriod / 6) PromethusLongPeriod = (PromethusDefaultPeriod * 600) // To reset Period + prometheusCtx context.Context + prometheusCancel context.CancelFunc activeConntrackCount = promauto.NewGauge( prometheus.GaugeOpts{ Name: "active_conntrack_count", @@ -143,215 +149,286 @@ func PrometheusRegister(hook cmn.NetHookInterface) { } func Init() { + prometheusCtx, prometheusCancel = context.WithCancel(context.Background()) + // Make Conntrack Statistic map ConntrackStats = make(map[ConntrackKey]Stats) mutex = &sync.Mutex{} - go RunGetConntrack() - go RunGetEndpoint() - go RunActiveConntrackCount() - go RunHostCount() - go RunProcessedStatistic() - go RunNewFlowCount() - go RunResetCounts() - go RunGetLBRule() - go RunLcusCalculator() + go RunGetConntrack(prometheusCtx) + go RunGetEndpoint(prometheusCtx) + go RunActiveConntrackCount(prometheusCtx) + go RunHostCount(prometheusCtx) + go RunProcessedStatistic(prometheusCtx) + go RunNewFlowCount(prometheusCtx) + go RunResetCounts(prometheusCtx) + go RunGetLBRule(prometheusCtx) + go RunLcusCalculator(prometheusCtx) +} + +func Off() error { + if !options.Opts.Prometheus { + return errors.New(http.StatusBadRequest, "already prometheus turned off") + } + options.Opts.Prometheus = false + prometheusCancel() + return nil +} + +func TurnOn() error { + if options.Opts.Prometheus { + return errors.New(http.StatusBadRequest, "already prometheus turned on") + } + options.Opts.Prometheus = true + Init() + return nil } func MakeConntrackKey(c cmn.CtInfo) (key ConntrackKey) { return ConntrackKey(fmt.Sprintf("%s|%05d|%s|%05d|%v", c.Sip, c.Sport, c.Dip, c.Dport, c.Proto)) } -func RunResetCounts() { +func RunResetCounts(ctx context.Context) { for { // Statistic reset time.Sleep(PromethusLongPeriod) - mutex.Lock() - ConntrackStats = map[ConntrackKey]Stats{} - mutex.Unlock() + select { + case <-ctx.Done(): + return + default: + mutex.Lock() + ConntrackStats = map[ConntrackKey]Stats{} + mutex.Unlock() + } } } -func RunGetConntrack() { +func RunGetConntrack(ctx context.Context) { for { - mutex.Lock() - ConntrackInfo, err = hooks.NetCtInfoGet() - if err != nil { - tk.LogIt(tk.LogDebug, "[Prometheus] Error occur : %v\n", err) - } - - for _, ct := range ConntrackInfo { - k := MakeConntrackKey(ct) - var tmpStats Stats - _, ok := ConntrackStats[k] - if ok { - tmpStats = Stats{ - Bytes: ConntrackStats[k].Bytes + ct.Bytes, - Packets: ConntrackStats[k].Packets + ct.Pkts, - } - } else { - tmpStats = Stats{ - Bytes: ct.Bytes, - Packets: ct.Pkts, - } + select { + case <-ctx.Done(): + return + default: + mutex.Lock() + ConntrackInfo, err = hooks.NetCtInfoGet() + if err != nil { + tk.LogIt(tk.LogDebug, "[Prometheus] Error occur : %v\n", err) } - ConntrackStats[k] = tmpStats + for _, ct := range ConntrackInfo { + k := MakeConntrackKey(ct) + var tmpStats Stats + _, ok := ConntrackStats[k] + if ok { + tmpStats = Stats{ + Bytes: ConntrackStats[k].Bytes + ct.Bytes, + Packets: ConntrackStats[k].Packets + ct.Pkts, + } + } else { + tmpStats = Stats{ + Bytes: ct.Bytes, + Packets: ct.Pkts, + } + } + ConntrackStats[k] = tmpStats + } + mutex.Unlock() } - mutex.Unlock() + time.Sleep(PromethusDefaultPeriod) } } -func RunGetEndpoint() { +func RunGetEndpoint(ctx context.Context) { for { - mutex.Lock() - EndPointInfo, err = hooks.NetEpHostGet() - if err != nil { - tk.LogIt(tk.LogDebug, "[Prometheus] Error occur : %v\n", err) + select { + case <-ctx.Done(): + return + default: + mutex.Lock() + EndPointInfo, err = hooks.NetEpHostGet() + if err != nil { + tk.LogIt(tk.LogDebug, "[Prometheus] Error occur : %v\n", err) + } + mutex.Unlock() } - mutex.Unlock() + time.Sleep(PromethusDefaultPeriod) } } -func RunGetLBRule() { +func RunGetLBRule(ctx context.Context) { for { - mutex.Lock() - LBRuleInfo, err = hooks.NetLbRuleGet() - if err != nil { - tk.LogIt(tk.LogDebug, "[Prometheus] Error occur : %v\n", err) + select { + case <-ctx.Done(): + return + default: + mutex.Lock() + LBRuleInfo, err = hooks.NetLbRuleGet() + if err != nil { + tk.LogIt(tk.LogDebug, "[Prometheus] Error occur : %v\n", err) + } + ruleCount.Set(float64(len(LBRuleInfo))) + mutex.Unlock() } - ruleCount.Set(float64(len(LBRuleInfo))) - mutex.Unlock() + time.Sleep(PromethusDefaultPeriod) } } -func RunActiveConntrackCount() { +func RunActiveConntrackCount(ctx context.Context) { for { - mutex.Lock() - // init Counts - activeFlowCountTcp.Set(0) - activeFlowCountUdp.Set(0) - activeFlowCountSctp.Set(0) - inActiveFlowCount.Set(0) + select { + case <-ctx.Done(): + return + default: + mutex.Lock() + // init Counts + activeFlowCountTcp.Set(0) + activeFlowCountUdp.Set(0) + activeFlowCountSctp.Set(0) + inActiveFlowCount.Set(0) - // Total flow count - activeConntrackCount.Set(float64(len(ConntrackInfo))) + // Total flow count + activeConntrackCount.Set(float64(len(ConntrackInfo))) - for _, ct := range ConntrackInfo { - // TCP flow count - if ct.Proto == "tcp" { - activeFlowCountTcp.Inc() - } - // UDP flow count - if ct.Proto == "udp" { - activeFlowCountUdp.Inc() - } - // SCTP flow count - if ct.Proto == "sctp" { - activeFlowCountSctp.Inc() - } - // Closed flow count - if ct.CState == "closed" { - inActiveFlowCount.Inc() + for _, ct := range ConntrackInfo { + // TCP flow count + if ct.Proto == "tcp" { + activeFlowCountTcp.Inc() + } + // UDP flow count + if ct.Proto == "udp" { + activeFlowCountUdp.Inc() + } + // SCTP flow count + if ct.Proto == "sctp" { + activeFlowCountSctp.Inc() + } + // Closed flow count + if ct.CState == "closed" { + inActiveFlowCount.Inc() + } } + mutex.Unlock() } - mutex.Unlock() + time.Sleep(PromethusDefaultPeriod) } } -func RunHostCount() { +func RunHostCount(ctx context.Context) { for { - mutex.Lock() - healthyHostCount.Set(0) - unHealthyHostCount.Set(0) - for _, ep := range EndPointInfo { - if ep.CurrState == "ok" { - healthyHostCount.Inc() - } - if ep.CurrState == "nok" { - unHealthyHostCount.Inc() + select { + case <-ctx.Done(): + return + default: + mutex.Lock() + healthyHostCount.Set(0) + unHealthyHostCount.Set(0) + for _, ep := range EndPointInfo { + if ep.CurrState == "ok" { + healthyHostCount.Inc() + } + if ep.CurrState == "nok" { + unHealthyHostCount.Inc() + } } + mutex.Unlock() } - mutex.Unlock() + time.Sleep(PromethusDefaultPeriod) } } -func RunProcessedStatistic() { +func RunProcessedStatistic(ctx context.Context) { for { - mutex.Lock() - // Init Stats - processedPackets.Set(0) - processedBytes.Set(0) - processedTCPBytes.Set(0) - processedUDPBytes.Set(0) - processedSCTPBytes.Set(0) - for k, ct := range ConntrackStats { - if strings.Contains(string(k), "tcp") { - processedTCPBytes.Add(float64(ct.Bytes)) - } - if strings.Contains(string(k), "udp") { - processedUDPBytes.Add(float64(ct.Bytes)) - } - if strings.Contains(string(k), "sctp") { - processedSCTPBytes.Add(float64(ct.Bytes)) + select { + case <-ctx.Done(): + return + default: + mutex.Lock() + // Init Stats + processedPackets.Set(0) + processedBytes.Set(0) + processedTCPBytes.Set(0) + processedUDPBytes.Set(0) + processedSCTPBytes.Set(0) + for k, ct := range ConntrackStats { + if strings.Contains(string(k), "tcp") { + processedTCPBytes.Add(float64(ct.Bytes)) + } + if strings.Contains(string(k), "udp") { + processedUDPBytes.Add(float64(ct.Bytes)) + } + if strings.Contains(string(k), "sctp") { + processedSCTPBytes.Add(float64(ct.Bytes)) + } + processedPackets.Add(float64(ct.Packets)) + processedBytes.Add(float64(ct.Bytes)) } - processedPackets.Add(float64(ct.Packets)) - processedBytes.Add(float64(ct.Bytes)) + mutex.Unlock() } - mutex.Unlock() + time.Sleep(PromethusDefaultPeriod) } } -func RunNewFlowCount() { +func RunNewFlowCount(ctx context.Context) { PreFlowCounts = 0 for { - mutex.Lock() - // Total new flow count - CurrentFlowCounts := len(ConntrackInfo) - diff := CurrentFlowCounts - PreFlowCounts - if diff > 0 { - newFlowCount.Set(float64(diff)) - } else { - newFlowCount.Set(0) + select { + case <-ctx.Done(): + return + default: + mutex.Lock() + // Total new flow count + CurrentFlowCounts := len(ConntrackInfo) + diff := CurrentFlowCounts - PreFlowCounts + if diff > 0 { + newFlowCount.Set(float64(diff)) + } else { + newFlowCount.Set(0) + } + PreFlowCounts = CurrentFlowCounts + mutex.Unlock() } - PreFlowCounts = CurrentFlowCounts - mutex.Unlock() + time.Sleep(PromethusDefaultPeriod) } } -func RunLcusCalculator() { +func RunLcusCalculator(ctx context.Context) { for { time.Sleep(PromethusDefaultPeriod) - mutex.Lock() - var LCUNewFlowCount = &dto.Metric{} - var LCUActiveFlowCount = &dto.Metric{} - var LCURuleCount = &dto.Metric{} - var LCUProcessedBytes = &dto.Metric{} - if err := newFlowCount.Write(LCUNewFlowCount); err != nil { - tk.LogIt(tk.LogDebug, "[Prometheus] Error occur : %v\n", err) - } - if err := activeConntrackCount.Write(LCUActiveFlowCount); err != nil { - tk.LogIt(tk.LogDebug, "[Prometheus] Error occur : %v\n", err) - } - if err := ruleCount.Write(LCURuleCount); err != nil { - tk.LogIt(tk.LogDebug, "[Prometheus] Error occur : %v\n", err) - } - if err := processedBytes.Write(LCUProcessedBytes); err != nil { - tk.LogIt(tk.LogDebug, "[Prometheus] Error occur : %v\n", err) + select { + case <-ctx.Done(): + return + default: + mutex.Lock() + var LCUNewFlowCount = &dto.Metric{} + var LCUActiveFlowCount = &dto.Metric{} + var LCURuleCount = &dto.Metric{} + var LCUProcessedBytes = &dto.Metric{} + if err := newFlowCount.Write(LCUNewFlowCount); err != nil { + tk.LogIt(tk.LogDebug, "[Prometheus] Error occur : %v\n", err) + } + if err := activeConntrackCount.Write(LCUActiveFlowCount); err != nil { + tk.LogIt(tk.LogDebug, "[Prometheus] Error occur : %v\n", err) + } + if err := ruleCount.Write(LCURuleCount); err != nil { + tk.LogIt(tk.LogDebug, "[Prometheus] Error occur : %v\n", err) + } + if err := processedBytes.Write(LCUProcessedBytes); err != nil { + tk.LogIt(tk.LogDebug, "[Prometheus] Error occur : %v\n", err) + } + // LCU of accumulated Flow count = Flowcount / 2160000 + // LCU of Rule = ruleCount/1000 + // LCU of Byte = processedBytes(Gb)/1h + consumedLcus.Set(float64(len(ConntrackStats))/2160000 + + *LCURuleCount.Gauge.Value/1000 + + (*LCUProcessedBytes.Gauge.Value*8)/360000000000) // (byte * 8)/ (60*60*1G)/10 + mutex.Unlock() } - // LCU of accumulated Flow count = Flowcount / 2160000 - // LCU of Rule = ruleCount/1000 - // LCU of Byte = processedBytes(Gb)/1h - consumedLcus.Set(float64(len(ConntrackStats))/2160000 + - *LCURuleCount.Gauge.Value/1000 + - (*LCUProcessedBytes.Gauge.Value*8)/360000000000) // (byte * 8)/ (60*60*1G)/10 - mutex.Unlock() } } diff --git a/api/restapi/configure_loxilb_rest_api.go b/api/restapi/configure_loxilb_rest_api.go index 8d2735c45..1db9e93e1 100644 --- a/api/restapi/configure_loxilb_rest_api.go +++ b/api/restapi/configure_loxilb_rest_api.go @@ -158,6 +158,9 @@ func configureAPI(api *operations.LoxilbRestAPIAPI) http.Handler { // Prometheus api.GetMetricsHandler = operations.GetMetricsHandlerFunc(handler.ConfigGetPrometheusCounter) + api.GetConfigMetricsHandler = operations.GetConfigMetricsHandlerFunc(handler.ConfigGetPrometheusOption) + api.PostConfigMetricsHandler = operations.PostConfigMetricsHandlerFunc(handler.ConfigPostPrometheus) + api.DeleteConfigMetricsHandler = operations.DeleteConfigMetricsHandlerFunc(handler.ConfigDeletePrometheus) // BGP Peer api.GetConfigBgpNeighAllHandler = operations.GetConfigBgpNeighAllHandlerFunc(handler.ConfigGetBGPNeigh) diff --git a/api/restapi/embedded_spec.go b/api/restapi/embedded_spec.go index 2ea838773..7925fbcac 100644 --- a/api/restapi/embedded_spec.go +++ b/api/restapi/embedded_spec.go @@ -2281,6 +2281,107 @@ func init() { } } }, + "/config/metrics": { + "get": { + "summary": "Get prometheus config value", + "responses": { + "200": { + "description": "prometheus config value", + "schema": { + "$ref": "#/definitions/MetricsConfig" + } + }, + "400": { + "description": "Malformed arguments for API call", + "schema": { + "$ref": "#/definitions/Error" + } + }, + "401": { + "description": "Invalid authentication credentials", + "schema": { + "$ref": "#/definitions/Error" + } + }, + "500": { + "description": "Internal service error", + "schema": { + "$ref": "#/definitions/Error" + } + }, + "503": { + "description": "Maintenance mode", + "schema": { + "$ref": "#/definitions/Error" + } + } + } + }, + "post": { + "summary": "turn on prometheus option", + "responses": { + "204": { + "description": "OK" + }, + "400": { + "description": "Malformed arguments for API call", + "schema": { + "$ref": "#/definitions/Error" + } + }, + "401": { + "description": "Invalid authentication credentials", + "schema": { + "$ref": "#/definitions/Error" + } + }, + "500": { + "description": "Internal service error", + "schema": { + "$ref": "#/definitions/Error" + } + }, + "503": { + "description": "Maintenance mode", + "schema": { + "$ref": "#/definitions/Error" + } + } + } + }, + "delete": { + "summary": "turn off prometheus option", + "responses": { + "204": { + "description": "OK" + }, + "400": { + "description": "Malformed arguments for API call", + "schema": { + "$ref": "#/definitions/Error" + } + }, + "401": { + "description": "Invalid authentication credentials", + "schema": { + "$ref": "#/definitions/Error" + } + }, + "500": { + "description": "Internal service error", + "schema": { + "$ref": "#/definitions/Error" + } + }, + "503": { + "description": "Maintenance mode", + "schema": { + "$ref": "#/definitions/Error" + } + } + } + } + }, "/config/mirror": { "post": { "description": "Create a new Mirror config.", @@ -5105,6 +5206,15 @@ func init() { } } }, + "MetricsConfig": { + "type": "object", + "properties": { + "prometheus": { + "description": "value for prometheus enable or not", + "type": "boolean" + } + } + }, "MirrorEntry": { "type": "object", "properties": { @@ -7992,6 +8102,107 @@ func init() { } } }, + "/config/metrics": { + "get": { + "summary": "Get prometheus config value", + "responses": { + "200": { + "description": "prometheus config value", + "schema": { + "$ref": "#/definitions/MetricsConfig" + } + }, + "400": { + "description": "Malformed arguments for API call", + "schema": { + "$ref": "#/definitions/Error" + } + }, + "401": { + "description": "Invalid authentication credentials", + "schema": { + "$ref": "#/definitions/Error" + } + }, + "500": { + "description": "Internal service error", + "schema": { + "$ref": "#/definitions/Error" + } + }, + "503": { + "description": "Maintenance mode", + "schema": { + "$ref": "#/definitions/Error" + } + } + } + }, + "post": { + "summary": "turn on prometheus option", + "responses": { + "204": { + "description": "OK" + }, + "400": { + "description": "Malformed arguments for API call", + "schema": { + "$ref": "#/definitions/Error" + } + }, + "401": { + "description": "Invalid authentication credentials", + "schema": { + "$ref": "#/definitions/Error" + } + }, + "500": { + "description": "Internal service error", + "schema": { + "$ref": "#/definitions/Error" + } + }, + "503": { + "description": "Maintenance mode", + "schema": { + "$ref": "#/definitions/Error" + } + } + } + }, + "delete": { + "summary": "turn off prometheus option", + "responses": { + "204": { + "description": "OK" + }, + "400": { + "description": "Malformed arguments for API call", + "schema": { + "$ref": "#/definitions/Error" + } + }, + "401": { + "description": "Invalid authentication credentials", + "schema": { + "$ref": "#/definitions/Error" + } + }, + "500": { + "description": "Internal service error", + "schema": { + "$ref": "#/definitions/Error" + } + }, + "503": { + "description": "Maintenance mode", + "schema": { + "$ref": "#/definitions/Error" + } + } + } + } + }, "/config/mirror": { "post": { "description": "Create a new Mirror config.", @@ -11378,6 +11589,15 @@ func init() { } } }, + "MetricsConfig": { + "type": "object", + "properties": { + "prometheus": { + "description": "value for prometheus enable or not", + "type": "boolean" + } + } + }, "MirrorEntry": { "type": "object", "properties": { diff --git a/api/restapi/handler/prometheus.go b/api/restapi/handler/prometheus.go index 603a9a855..6fc1940b0 100644 --- a/api/restapi/handler/prometheus.go +++ b/api/restapi/handler/prometheus.go @@ -16,6 +16,8 @@ package handler import ( + "github.com/loxilb-io/loxilb/api/models" + "github.com/loxilb-io/loxilb/api/prometheus" "net/http" "github.com/go-openapi/runtime" @@ -35,3 +37,29 @@ func ConfigGetPrometheusCounter(params operations.GetMetricsParams) middleware.R promhttp.Handler().ServeHTTP(w, params.HTTPRequest) }) } + +func ConfigGetPrometheusOption(params operations.GetConfigMetricsParams) middleware.Responder { + tk.LogIt(tk.LogDebug, "[API] Prometheus %s API called. url : %s\n", params.HTTPRequest.Method, params.HTTPRequest.URL) + return operations.NewGetConfigMetricsOK().WithPayload(&models.MetricsConfig{Prometheus: options.Opts.Prometheus}) +} + +func ConfigPostPrometheus(params operations.PostConfigMetricsParams) middleware.Responder { + tk.LogIt(tk.LogDebug, "[API] Prometheus %s API called. url : %s\n", params.HTTPRequest.Method, params.HTTPRequest.URL) + err := prometheus.TurnOn() + if err != nil { + tk.LogIt(tk.LogDebug, "[API] Error occur : %v\n", err) + return &ResultResponse{Result: err.Error()} + } + return &ResultResponse{Result: "Success"} +} + +func ConfigDeletePrometheus(params operations.DeleteConfigMetricsParams) middleware.Responder { + tk.LogIt(tk.LogDebug, "[API] Prometheus %s API called. url : %s\n", params.HTTPRequest.Method, params.HTTPRequest.URL) + err := prometheus.Off() + if err != nil { + tk.LogIt(tk.LogDebug, "[API] Error occur : %v\n", err) + return &ResultResponse{Result: err.Error()} + } + + return &ResultResponse{Result: "Success"} +} diff --git a/api/restapi/operations/loxilb_rest_api_api.go b/api/restapi/operations/loxilb_rest_api_api.go index 4d0909b75..fa9d67095 100644 --- a/api/restapi/operations/loxilb_rest_api_api.go +++ b/api/restapi/operations/loxilb_rest_api_api.go @@ -81,6 +81,9 @@ func NewLoxilbRestAPIAPI(spec *loads.Document) *LoxilbRestAPIAPI { DeleteConfigLoadbalancerNameLbNameHandler: DeleteConfigLoadbalancerNameLbNameHandlerFunc(func(params DeleteConfigLoadbalancerNameLbNameParams) middleware.Responder { return middleware.NotImplemented("operation DeleteConfigLoadbalancerNameLbName has not yet been implemented") }), + DeleteConfigMetricsHandler: DeleteConfigMetricsHandlerFunc(func(params DeleteConfigMetricsParams) middleware.Responder { + return middleware.NotImplemented("operation DeleteConfigMetrics has not yet been implemented") + }), DeleteConfigMirrorIdentIdentHandler: DeleteConfigMirrorIdentIdentHandlerFunc(func(params DeleteConfigMirrorIdentIdentParams) middleware.Responder { return middleware.NotImplemented("operation DeleteConfigMirrorIdentIdent has not yet been implemented") }), @@ -144,6 +147,9 @@ func NewLoxilbRestAPIAPI(spec *loads.Document) *LoxilbRestAPIAPI { GetConfigLoadbalancerAllHandler: GetConfigLoadbalancerAllHandlerFunc(func(params GetConfigLoadbalancerAllParams) middleware.Responder { return middleware.NotImplemented("operation GetConfigLoadbalancerAll has not yet been implemented") }), + GetConfigMetricsHandler: GetConfigMetricsHandlerFunc(func(params GetConfigMetricsParams) middleware.Responder { + return middleware.NotImplemented("operation GetConfigMetrics has not yet been implemented") + }), GetConfigMirrorAllHandler: GetConfigMirrorAllHandlerFunc(func(params GetConfigMirrorAllParams) middleware.Responder { return middleware.NotImplemented("operation GetConfigMirrorAll has not yet been implemented") }), @@ -222,6 +228,9 @@ func NewLoxilbRestAPIAPI(spec *loads.Document) *LoxilbRestAPIAPI { PostConfigLoadbalancerHandler: PostConfigLoadbalancerHandlerFunc(func(params PostConfigLoadbalancerParams) middleware.Responder { return middleware.NotImplemented("operation PostConfigLoadbalancer has not yet been implemented") }), + PostConfigMetricsHandler: PostConfigMetricsHandlerFunc(func(params PostConfigMetricsParams) middleware.Responder { + return middleware.NotImplemented("operation PostConfigMetrics has not yet been implemented") + }), PostConfigMirrorHandler: PostConfigMirrorHandlerFunc(func(params PostConfigMirrorParams) middleware.Responder { return middleware.NotImplemented("operation PostConfigMirror has not yet been implemented") }), @@ -317,6 +326,8 @@ type LoxilbRestAPIAPI struct { DeleteConfigLoadbalancerHosturlHosturlExternalipaddressIPAddressPortPortProtocolProtoHandler DeleteConfigLoadbalancerHosturlHosturlExternalipaddressIPAddressPortPortProtocolProtoHandler // DeleteConfigLoadbalancerNameLbNameHandler sets the operation handler for the delete config loadbalancer name lb name operation DeleteConfigLoadbalancerNameLbNameHandler DeleteConfigLoadbalancerNameLbNameHandler + // DeleteConfigMetricsHandler sets the operation handler for the delete config metrics operation + DeleteConfigMetricsHandler DeleteConfigMetricsHandler // DeleteConfigMirrorIdentIdentHandler sets the operation handler for the delete config mirror ident ident operation DeleteConfigMirrorIdentIdentHandler DeleteConfigMirrorIdentIdentHandler // DeleteConfigNeighborIPAddressDevIfNameHandler sets the operation handler for the delete config neighbor IP address dev if name operation @@ -359,6 +370,8 @@ type LoxilbRestAPIAPI struct { GetConfigIpv4addressAllHandler GetConfigIpv4addressAllHandler // GetConfigLoadbalancerAllHandler sets the operation handler for the get config loadbalancer all operation GetConfigLoadbalancerAllHandler GetConfigLoadbalancerAllHandler + // GetConfigMetricsHandler sets the operation handler for the get config metrics operation + GetConfigMetricsHandler GetConfigMetricsHandler // GetConfigMirrorAllHandler sets the operation handler for the get config mirror all operation GetConfigMirrorAllHandler GetConfigMirrorAllHandler // GetConfigNeighborAllHandler sets the operation handler for the get config neighbor all operation @@ -411,6 +424,8 @@ type LoxilbRestAPIAPI struct { PostConfigIpv4addressHandler PostConfigIpv4addressHandler // PostConfigLoadbalancerHandler sets the operation handler for the post config loadbalancer operation PostConfigLoadbalancerHandler PostConfigLoadbalancerHandler + // PostConfigMetricsHandler sets the operation handler for the post config metrics operation + PostConfigMetricsHandler PostConfigMetricsHandler // PostConfigMirrorHandler sets the operation handler for the post config mirror operation PostConfigMirrorHandler PostConfigMirrorHandler // PostConfigNeighborHandler sets the operation handler for the post config neighbor operation @@ -549,6 +564,9 @@ func (o *LoxilbRestAPIAPI) Validate() error { if o.DeleteConfigLoadbalancerNameLbNameHandler == nil { unregistered = append(unregistered, "DeleteConfigLoadbalancerNameLbNameHandler") } + if o.DeleteConfigMetricsHandler == nil { + unregistered = append(unregistered, "DeleteConfigMetricsHandler") + } if o.DeleteConfigMirrorIdentIdentHandler == nil { unregistered = append(unregistered, "DeleteConfigMirrorIdentIdentHandler") } @@ -612,6 +630,9 @@ func (o *LoxilbRestAPIAPI) Validate() error { if o.GetConfigLoadbalancerAllHandler == nil { unregistered = append(unregistered, "GetConfigLoadbalancerAllHandler") } + if o.GetConfigMetricsHandler == nil { + unregistered = append(unregistered, "GetConfigMetricsHandler") + } if o.GetConfigMirrorAllHandler == nil { unregistered = append(unregistered, "GetConfigMirrorAllHandler") } @@ -690,6 +711,9 @@ func (o *LoxilbRestAPIAPI) Validate() error { if o.PostConfigLoadbalancerHandler == nil { unregistered = append(unregistered, "PostConfigLoadbalancerHandler") } + if o.PostConfigMetricsHandler == nil { + unregistered = append(unregistered, "PostConfigMetricsHandler") + } if o.PostConfigMirrorHandler == nil { unregistered = append(unregistered, "PostConfigMirrorHandler") } @@ -866,6 +890,10 @@ func (o *LoxilbRestAPIAPI) initHandlerCache() { if o.handlers["DELETE"] == nil { o.handlers["DELETE"] = make(map[string]http.Handler) } + o.handlers["DELETE"]["/config/metrics"] = NewDeleteConfigMetrics(o.context, o.DeleteConfigMetricsHandler) + if o.handlers["DELETE"] == nil { + o.handlers["DELETE"] = make(map[string]http.Handler) + } o.handlers["DELETE"]["/config/mirror/ident/{ident}"] = NewDeleteConfigMirrorIdentIdent(o.context, o.DeleteConfigMirrorIdentIdentHandler) if o.handlers["DELETE"] == nil { o.handlers["DELETE"] = make(map[string]http.Handler) @@ -950,6 +978,10 @@ func (o *LoxilbRestAPIAPI) initHandlerCache() { if o.handlers["GET"] == nil { o.handlers["GET"] = make(map[string]http.Handler) } + o.handlers["GET"]["/config/metrics"] = NewGetConfigMetrics(o.context, o.GetConfigMetricsHandler) + if o.handlers["GET"] == nil { + o.handlers["GET"] = make(map[string]http.Handler) + } o.handlers["GET"]["/config/mirror/all"] = NewGetConfigMirrorAll(o.context, o.GetConfigMirrorAllHandler) if o.handlers["GET"] == nil { o.handlers["GET"] = make(map[string]http.Handler) @@ -1054,6 +1086,10 @@ func (o *LoxilbRestAPIAPI) initHandlerCache() { if o.handlers["POST"] == nil { o.handlers["POST"] = make(map[string]http.Handler) } + o.handlers["POST"]["/config/metrics"] = NewPostConfigMetrics(o.context, o.PostConfigMetricsHandler) + if o.handlers["POST"] == nil { + o.handlers["POST"] = make(map[string]http.Handler) + } o.handlers["POST"]["/config/mirror"] = NewPostConfigMirror(o.context, o.PostConfigMirrorHandler) if o.handlers["POST"] == nil { o.handlers["POST"] = make(map[string]http.Handler) @@ -1136,6 +1172,6 @@ func (o *LoxilbRestAPIAPI) AddMiddlewareFor(method, path string, builder middlew } o.Init() if h, ok := o.handlers[um][path]; ok { - o.handlers[method][path] = builder(h) + o.handlers[um][path] = builder(h) } } diff --git a/api/swagger.yml b/api/swagger.yml index 2e930bfef..98ebff548 100644 --- a/api/swagger.yml +++ b/api/swagger.yml @@ -2742,6 +2742,72 @@ paths: description: Metrics in prometheus text format schema: type: string + '/config/metrics': + get: + summary: Get prometheus config value + responses: + '200': + description: prometheus config value + schema: + $ref: '#/definitions/MetricsConfig' + '400': + description: Malformed arguments for API call + schema: + $ref: '#/definitions/Error' + '401': + description: Invalid authentication credentials + schema: + $ref: '#/definitions/Error' + '500': + description: Internal service error + schema: + $ref: '#/definitions/Error' + '503': + description: Maintenance mode + schema: + $ref: '#/definitions/Error' + post: + summary: turn on prometheus option + responses: + '204': + description: OK + '400': + description: Malformed arguments for API call + schema: + $ref: '#/definitions/Error' + '401': + description: Invalid authentication credentials + schema: + $ref: '#/definitions/Error' + '500': + description: Internal service error + schema: + $ref: '#/definitions/Error' + '503': + description: Maintenance mode + schema: + $ref: '#/definitions/Error' + delete: + summary: turn off prometheus option + responses: + '204': + description: OK + '400': + description: Malformed arguments for API call + schema: + $ref: '#/definitions/Error' + '401': + description: Invalid authentication credentials + schema: + $ref: '#/definitions/Error' + '500': + description: Internal service error + schema: + $ref: '#/definitions/Error' + '503': + description: Maintenance mode + schema: + $ref: '#/definitions/Error' #---------------------------------------------- # BFD @@ -4032,3 +4098,9 @@ definitions: type: integer format: uint8 description: Retry Count to detect failure + MetricsConfig: + type: object + properties: + prometheus: + type: boolean + description: value for prometheus enable or not From 61c7022a4b036d5a6eb312fa3762047052b1e412 Mon Sep 17 00:00:00 2001 From: Bangseungjae Date: Sat, 12 Oct 2024 15:02:54 +0900 Subject: [PATCH 5/5] #828 Feature Prometheus API Signed-off-by: Bangseungjae --- api/models/metrics_config.go | 50 ++++ .../operations/delete_config_metrics.go | 56 ++++ .../delete_config_metrics_parameters.go | 46 ++++ .../delete_config_metrics_responses.go | 219 ++++++++++++++++ .../delete_config_metrics_urlbuilder.go | 87 +++++++ api/restapi/operations/get_config_metrics.go | 56 ++++ .../get_config_metrics_parameters.go | 46 ++++ .../get_config_metrics_responses.go | 239 ++++++++++++++++++ .../get_config_metrics_urlbuilder.go | 87 +++++++ api/restapi/operations/post_config_metrics.go | 56 ++++ .../post_config_metrics_parameters.go | 46 ++++ .../post_config_metrics_responses.go | 219 ++++++++++++++++ 12 files changed, 1207 insertions(+) create mode 100644 api/models/metrics_config.go create mode 100644 api/restapi/operations/delete_config_metrics.go create mode 100644 api/restapi/operations/delete_config_metrics_parameters.go create mode 100644 api/restapi/operations/delete_config_metrics_responses.go create mode 100644 api/restapi/operations/delete_config_metrics_urlbuilder.go create mode 100644 api/restapi/operations/get_config_metrics.go create mode 100644 api/restapi/operations/get_config_metrics_parameters.go create mode 100644 api/restapi/operations/get_config_metrics_responses.go create mode 100644 api/restapi/operations/get_config_metrics_urlbuilder.go create mode 100644 api/restapi/operations/post_config_metrics.go create mode 100644 api/restapi/operations/post_config_metrics_parameters.go create mode 100644 api/restapi/operations/post_config_metrics_responses.go diff --git a/api/models/metrics_config.go b/api/models/metrics_config.go new file mode 100644 index 000000000..41d5cb9b2 --- /dev/null +++ b/api/models/metrics_config.go @@ -0,0 +1,50 @@ +// Code generated by go-swagger; DO NOT EDIT. + +package models + +// This file was generated by the swagger tool. +// Editing this file might prove futile when you re-run the swagger generate command + +import ( + "context" + + "github.com/go-openapi/strfmt" + "github.com/go-openapi/swag" +) + +// MetricsConfig metrics config +// +// swagger:model MetricsConfig +type MetricsConfig struct { + + // value for prometheus enable or not + Prometheus bool `json:"prometheus,omitempty"` +} + +// Validate validates this metrics config +func (m *MetricsConfig) Validate(formats strfmt.Registry) error { + return nil +} + +// ContextValidate validates this metrics config based on context it is used +func (m *MetricsConfig) ContextValidate(ctx context.Context, formats strfmt.Registry) error { + return nil +} + +// MarshalBinary interface implementation +func (m *MetricsConfig) MarshalBinary() ([]byte, error) { + if m == nil { + return nil, nil + } + return swag.WriteJSON(m) +} + +// UnmarshalBinary interface implementation +func (m *MetricsConfig) UnmarshalBinary(b []byte) error { + var res MetricsConfig + if err := swag.ReadJSON(b, &res); err != nil { + return err + } + *m = res + return nil +} diff --git a/api/restapi/operations/delete_config_metrics.go b/api/restapi/operations/delete_config_metrics.go new file mode 100644 index 000000000..eaf768f17 --- /dev/null +++ b/api/restapi/operations/delete_config_metrics.go @@ -0,0 +1,56 @@ +// Code generated by go-swagger; DO NOT EDIT. + +package operations + +// This file was generated by the swagger tool. +// Editing this file might prove futile when you re-run the generate command + +import ( + "net/http" + + "github.com/go-openapi/runtime/middleware" +) + +// DeleteConfigMetricsHandlerFunc turns a function with the right signature into a delete config metrics handler +type DeleteConfigMetricsHandlerFunc func(DeleteConfigMetricsParams) middleware.Responder + +// Handle executing the request and returning a response +func (fn DeleteConfigMetricsHandlerFunc) Handle(params DeleteConfigMetricsParams) middleware.Responder { + return fn(params) +} + +// DeleteConfigMetricsHandler interface for that can handle valid delete config metrics params +type DeleteConfigMetricsHandler interface { + Handle(DeleteConfigMetricsParams) middleware.Responder +} + +// NewDeleteConfigMetrics creates a new http.Handler for the delete config metrics operation +func NewDeleteConfigMetrics(ctx *middleware.Context, handler DeleteConfigMetricsHandler) *DeleteConfigMetrics { + return &DeleteConfigMetrics{Context: ctx, Handler: handler} +} + +/* + DeleteConfigMetrics swagger:route DELETE /config/metrics deleteConfigMetrics + +turn off prometheus option +*/ +type DeleteConfigMetrics struct { + Context *middleware.Context + Handler DeleteConfigMetricsHandler +} + +func (o *DeleteConfigMetrics) ServeHTTP(rw http.ResponseWriter, r *http.Request) { + route, rCtx, _ := o.Context.RouteInfo(r) + if rCtx != nil { + *r = *rCtx + } + var Params = NewDeleteConfigMetricsParams() + if err := o.Context.BindValidRequest(r, route, &Params); err != nil { // bind params + o.Context.Respond(rw, r, route.Produces, route, err) + return + } + + res := o.Handler.Handle(Params) // actually handle the request + o.Context.Respond(rw, r, route.Produces, route, res) + +} diff --git a/api/restapi/operations/delete_config_metrics_parameters.go b/api/restapi/operations/delete_config_metrics_parameters.go new file mode 100644 index 000000000..4059e268a --- /dev/null +++ b/api/restapi/operations/delete_config_metrics_parameters.go @@ -0,0 +1,46 @@ +// Code generated by go-swagger; DO NOT EDIT. + +package operations + +// This file was generated by the swagger tool. +// Editing this file might prove futile when you re-run the swagger generate command + +import ( + "net/http" + + "github.com/go-openapi/errors" + "github.com/go-openapi/runtime/middleware" +) + +// NewDeleteConfigMetricsParams creates a new DeleteConfigMetricsParams object +// +// There are no default values defined in the spec. +func NewDeleteConfigMetricsParams() DeleteConfigMetricsParams { + + return DeleteConfigMetricsParams{} +} + +// DeleteConfigMetricsParams contains all the bound params for the delete config metrics operation +// typically these are obtained from a http.Request +// +// swagger:parameters DeleteConfigMetrics +type DeleteConfigMetricsParams struct { + + // HTTP Request Object + HTTPRequest *http.Request `json:"-"` +} + +// BindRequest both binds and validates a request, it assumes that complex things implement a Validatable(strfmt.Registry) error interface +// for simple values it will use straight method calls. +// +// To ensure default values, the struct must have been initialized with NewDeleteConfigMetricsParams() beforehand. +func (o *DeleteConfigMetricsParams) BindRequest(r *http.Request, route *middleware.MatchedRoute) error { + var res []error + + o.HTTPRequest = r + + if len(res) > 0 { + return errors.CompositeValidationError(res...) + } + return nil +} diff --git a/api/restapi/operations/delete_config_metrics_responses.go b/api/restapi/operations/delete_config_metrics_responses.go new file mode 100644 index 000000000..e04201a0c --- /dev/null +++ b/api/restapi/operations/delete_config_metrics_responses.go @@ -0,0 +1,219 @@ +// Code generated by go-swagger; DO NOT EDIT. + +package operations + +// This file was generated by the swagger tool. +// Editing this file might prove futile when you re-run the swagger generate command + +import ( + "net/http" + + "github.com/go-openapi/runtime" + + "github.com/loxilb-io/loxilb/api/models" +) + +// DeleteConfigMetricsNoContentCode is the HTTP code returned for type DeleteConfigMetricsNoContent +const DeleteConfigMetricsNoContentCode int = 204 + +/* +DeleteConfigMetricsNoContent OK + +swagger:response deleteConfigMetricsNoContent +*/ +type DeleteConfigMetricsNoContent struct { +} + +// NewDeleteConfigMetricsNoContent creates DeleteConfigMetricsNoContent with default headers values +func NewDeleteConfigMetricsNoContent() *DeleteConfigMetricsNoContent { + + return &DeleteConfigMetricsNoContent{} +} + +// WriteResponse to the client +func (o *DeleteConfigMetricsNoContent) WriteResponse(rw http.ResponseWriter, producer runtime.Producer) { + + rw.Header().Del(runtime.HeaderContentType) //Remove Content-Type on empty responses + + rw.WriteHeader(204) +} + +// DeleteConfigMetricsBadRequestCode is the HTTP code returned for type DeleteConfigMetricsBadRequest +const DeleteConfigMetricsBadRequestCode int = 400 + +/* +DeleteConfigMetricsBadRequest Malformed arguments for API call + +swagger:response deleteConfigMetricsBadRequest +*/ +type DeleteConfigMetricsBadRequest struct { + + /* + In: Body + */ + Payload *models.Error `json:"body,omitempty"` +} + +// NewDeleteConfigMetricsBadRequest creates DeleteConfigMetricsBadRequest with default headers values +func NewDeleteConfigMetricsBadRequest() *DeleteConfigMetricsBadRequest { + + return &DeleteConfigMetricsBadRequest{} +} + +// WithPayload adds the payload to the delete config metrics bad request response +func (o *DeleteConfigMetricsBadRequest) WithPayload(payload *models.Error) *DeleteConfigMetricsBadRequest { + o.Payload = payload + return o +} + +// SetPayload sets the payload to the delete config metrics bad request response +func (o *DeleteConfigMetricsBadRequest) SetPayload(payload *models.Error) { + o.Payload = payload +} + +// WriteResponse to the client +func (o *DeleteConfigMetricsBadRequest) WriteResponse(rw http.ResponseWriter, producer runtime.Producer) { + + rw.WriteHeader(400) + if o.Payload != nil { + payload := o.Payload + if err := producer.Produce(rw, payload); err != nil { + panic(err) // let the recovery middleware deal with this + } + } +} + +// DeleteConfigMetricsUnauthorizedCode is the HTTP code returned for type DeleteConfigMetricsUnauthorized +const DeleteConfigMetricsUnauthorizedCode int = 401 + +/* +DeleteConfigMetricsUnauthorized Invalid authentication credentials + +swagger:response deleteConfigMetricsUnauthorized +*/ +type DeleteConfigMetricsUnauthorized struct { + + /* + In: Body + */ + Payload *models.Error `json:"body,omitempty"` +} + +// NewDeleteConfigMetricsUnauthorized creates DeleteConfigMetricsUnauthorized with default headers values +func NewDeleteConfigMetricsUnauthorized() *DeleteConfigMetricsUnauthorized { + + return &DeleteConfigMetricsUnauthorized{} +} + +// WithPayload adds the payload to the delete config metrics unauthorized response +func (o *DeleteConfigMetricsUnauthorized) WithPayload(payload *models.Error) *DeleteConfigMetricsUnauthorized { + o.Payload = payload + return o +} + +// SetPayload sets the payload to the delete config metrics unauthorized response +func (o *DeleteConfigMetricsUnauthorized) SetPayload(payload *models.Error) { + o.Payload = payload +} + +// WriteResponse to the client +func (o *DeleteConfigMetricsUnauthorized) WriteResponse(rw http.ResponseWriter, producer runtime.Producer) { + + rw.WriteHeader(401) + if o.Payload != nil { + payload := o.Payload + if err := producer.Produce(rw, payload); err != nil { + panic(err) // let the recovery middleware deal with this + } + } +} + +// DeleteConfigMetricsInternalServerErrorCode is the HTTP code returned for type DeleteConfigMetricsInternalServerError +const DeleteConfigMetricsInternalServerErrorCode int = 500 + +/* +DeleteConfigMetricsInternalServerError Internal service error + +swagger:response deleteConfigMetricsInternalServerError +*/ +type DeleteConfigMetricsInternalServerError struct { + + /* + In: Body + */ + Payload *models.Error `json:"body,omitempty"` +} + +// NewDeleteConfigMetricsInternalServerError creates DeleteConfigMetricsInternalServerError with default headers values +func NewDeleteConfigMetricsInternalServerError() *DeleteConfigMetricsInternalServerError { + + return &DeleteConfigMetricsInternalServerError{} +} + +// WithPayload adds the payload to the delete config metrics internal server error response +func (o *DeleteConfigMetricsInternalServerError) WithPayload(payload *models.Error) *DeleteConfigMetricsInternalServerError { + o.Payload = payload + return o +} + +// SetPayload sets the payload to the delete config metrics internal server error response +func (o *DeleteConfigMetricsInternalServerError) SetPayload(payload *models.Error) { + o.Payload = payload +} + +// WriteResponse to the client +func (o *DeleteConfigMetricsInternalServerError) WriteResponse(rw http.ResponseWriter, producer runtime.Producer) { + + rw.WriteHeader(500) + if o.Payload != nil { + payload := o.Payload + if err := producer.Produce(rw, payload); err != nil { + panic(err) // let the recovery middleware deal with this + } + } +} + +// DeleteConfigMetricsServiceUnavailableCode is the HTTP code returned for type DeleteConfigMetricsServiceUnavailable +const DeleteConfigMetricsServiceUnavailableCode int = 503 + +/* +DeleteConfigMetricsServiceUnavailable Maintenance mode + +swagger:response deleteConfigMetricsServiceUnavailable +*/ +type DeleteConfigMetricsServiceUnavailable struct { + + /* + In: Body + */ + Payload *models.Error `json:"body,omitempty"` +} + +// NewDeleteConfigMetricsServiceUnavailable creates DeleteConfigMetricsServiceUnavailable with default headers values +func NewDeleteConfigMetricsServiceUnavailable() *DeleteConfigMetricsServiceUnavailable { + + return &DeleteConfigMetricsServiceUnavailable{} +} + +// WithPayload adds the payload to the delete config metrics service unavailable response +func (o *DeleteConfigMetricsServiceUnavailable) WithPayload(payload *models.Error) *DeleteConfigMetricsServiceUnavailable { + o.Payload = payload + return o +} + +// SetPayload sets the payload to the delete config metrics service unavailable response +func (o *DeleteConfigMetricsServiceUnavailable) SetPayload(payload *models.Error) { + o.Payload = payload +} + +// WriteResponse to the client +func (o *DeleteConfigMetricsServiceUnavailable) WriteResponse(rw http.ResponseWriter, producer runtime.Producer) { + + rw.WriteHeader(503) + if o.Payload != nil { + payload := o.Payload + if err := producer.Produce(rw, payload); err != nil { + panic(err) // let the recovery middleware deal with this + } + } +} diff --git a/api/restapi/operations/delete_config_metrics_urlbuilder.go b/api/restapi/operations/delete_config_metrics_urlbuilder.go new file mode 100644 index 000000000..c42d430ba --- /dev/null +++ b/api/restapi/operations/delete_config_metrics_urlbuilder.go @@ -0,0 +1,87 @@ +// Code generated by go-swagger; DO NOT EDIT. + +package operations + +// This file was generated by the swagger tool. +// Editing this file might prove futile when you re-run the generate command + +import ( + "errors" + "net/url" + golangswaggerpaths "path" +) + +// DeleteConfigMetricsURL generates an URL for the delete config metrics operation +type DeleteConfigMetricsURL struct { + _basePath string +} + +// WithBasePath sets the base path for this url builder, only required when it's different from the +// base path specified in the swagger spec. +// When the value of the base path is an empty string +func (o *DeleteConfigMetricsURL) WithBasePath(bp string) *DeleteConfigMetricsURL { + o.SetBasePath(bp) + return o +} + +// SetBasePath sets the base path for this url builder, only required when it's different from the +// base path specified in the swagger spec. +// When the value of the base path is an empty string +func (o *DeleteConfigMetricsURL) SetBasePath(bp string) { + o._basePath = bp +} + +// Build a url path and query string +func (o *DeleteConfigMetricsURL) Build() (*url.URL, error) { + var _result url.URL + + var _path = "/config/metrics" + + _basePath := o._basePath + if _basePath == "" { + _basePath = "/netlox/v1" + } + _result.Path = golangswaggerpaths.Join(_basePath, _path) + + return &_result, nil +} + +// Must is a helper function to panic when the url builder returns an error +func (o *DeleteConfigMetricsURL) Must(u *url.URL, err error) *url.URL { + if err != nil { + panic(err) + } + if u == nil { + panic("url can't be nil") + } + return u +} + +// String returns the string representation of the path with query string +func (o *DeleteConfigMetricsURL) String() string { + return o.Must(o.Build()).String() +} + +// BuildFull builds a full url with scheme, host, path and query string +func (o *DeleteConfigMetricsURL) BuildFull(scheme, host string) (*url.URL, error) { + if scheme == "" { + return nil, errors.New("scheme is required for a full url on DeleteConfigMetricsURL") + } + if host == "" { + return nil, errors.New("host is required for a full url on DeleteConfigMetricsURL") + } + + base, err := o.Build() + if err != nil { + return nil, err + } + + base.Scheme = scheme + base.Host = host + return base, nil +} + +// StringFull returns the string representation of a complete url +func (o *DeleteConfigMetricsURL) StringFull(scheme, host string) string { + return o.Must(o.BuildFull(scheme, host)).String() +} diff --git a/api/restapi/operations/get_config_metrics.go b/api/restapi/operations/get_config_metrics.go new file mode 100644 index 000000000..85bbc6c3e --- /dev/null +++ b/api/restapi/operations/get_config_metrics.go @@ -0,0 +1,56 @@ +// Code generated by go-swagger; DO NOT EDIT. + +package operations + +// This file was generated by the swagger tool. +// Editing this file might prove futile when you re-run the generate command + +import ( + "net/http" + + "github.com/go-openapi/runtime/middleware" +) + +// GetConfigMetricsHandlerFunc turns a function with the right signature into a get config metrics handler +type GetConfigMetricsHandlerFunc func(GetConfigMetricsParams) middleware.Responder + +// Handle executing the request and returning a response +func (fn GetConfigMetricsHandlerFunc) Handle(params GetConfigMetricsParams) middleware.Responder { + return fn(params) +} + +// GetConfigMetricsHandler interface for that can handle valid get config metrics params +type GetConfigMetricsHandler interface { + Handle(GetConfigMetricsParams) middleware.Responder +} + +// NewGetConfigMetrics creates a new http.Handler for the get config metrics operation +func NewGetConfigMetrics(ctx *middleware.Context, handler GetConfigMetricsHandler) *GetConfigMetrics { + return &GetConfigMetrics{Context: ctx, Handler: handler} +} + +/* + GetConfigMetrics swagger:route GET /config/metrics getConfigMetrics + +Get prometheus config value +*/ +type GetConfigMetrics struct { + Context *middleware.Context + Handler GetConfigMetricsHandler +} + +func (o *GetConfigMetrics) ServeHTTP(rw http.ResponseWriter, r *http.Request) { + route, rCtx, _ := o.Context.RouteInfo(r) + if rCtx != nil { + *r = *rCtx + } + var Params = NewGetConfigMetricsParams() + if err := o.Context.BindValidRequest(r, route, &Params); err != nil { // bind params + o.Context.Respond(rw, r, route.Produces, route, err) + return + } + + res := o.Handler.Handle(Params) // actually handle the request + o.Context.Respond(rw, r, route.Produces, route, res) + +} diff --git a/api/restapi/operations/get_config_metrics_parameters.go b/api/restapi/operations/get_config_metrics_parameters.go new file mode 100644 index 000000000..56eb8e40f --- /dev/null +++ b/api/restapi/operations/get_config_metrics_parameters.go @@ -0,0 +1,46 @@ +// Code generated by go-swagger; DO NOT EDIT. + +package operations + +// This file was generated by the swagger tool. +// Editing this file might prove futile when you re-run the swagger generate command + +import ( + "net/http" + + "github.com/go-openapi/errors" + "github.com/go-openapi/runtime/middleware" +) + +// NewGetConfigMetricsParams creates a new GetConfigMetricsParams object +// +// There are no default values defined in the spec. +func NewGetConfigMetricsParams() GetConfigMetricsParams { + + return GetConfigMetricsParams{} +} + +// GetConfigMetricsParams contains all the bound params for the get config metrics operation +// typically these are obtained from a http.Request +// +// swagger:parameters GetConfigMetrics +type GetConfigMetricsParams struct { + + // HTTP Request Object + HTTPRequest *http.Request `json:"-"` +} + +// BindRequest both binds and validates a request, it assumes that complex things implement a Validatable(strfmt.Registry) error interface +// for simple values it will use straight method calls. +// +// To ensure default values, the struct must have been initialized with NewGetConfigMetricsParams() beforehand. +func (o *GetConfigMetricsParams) BindRequest(r *http.Request, route *middleware.MatchedRoute) error { + var res []error + + o.HTTPRequest = r + + if len(res) > 0 { + return errors.CompositeValidationError(res...) + } + return nil +} diff --git a/api/restapi/operations/get_config_metrics_responses.go b/api/restapi/operations/get_config_metrics_responses.go new file mode 100644 index 000000000..329a0dc3b --- /dev/null +++ b/api/restapi/operations/get_config_metrics_responses.go @@ -0,0 +1,239 @@ +// Code generated by go-swagger; DO NOT EDIT. + +package operations + +// This file was generated by the swagger tool. +// Editing this file might prove futile when you re-run the swagger generate command + +import ( + "net/http" + + "github.com/go-openapi/runtime" + + "github.com/loxilb-io/loxilb/api/models" +) + +// GetConfigMetricsOKCode is the HTTP code returned for type GetConfigMetricsOK +const GetConfigMetricsOKCode int = 200 + +/* +GetConfigMetricsOK prometheus config value + +swagger:response getConfigMetricsOK +*/ +type GetConfigMetricsOK struct { + + /* + In: Body + */ + Payload *models.MetricsConfig `json:"body,omitempty"` +} + +// NewGetConfigMetricsOK creates GetConfigMetricsOK with default headers values +func NewGetConfigMetricsOK() *GetConfigMetricsOK { + + return &GetConfigMetricsOK{} +} + +// WithPayload adds the payload to the get config metrics o k response +func (o *GetConfigMetricsOK) WithPayload(payload *models.MetricsConfig) *GetConfigMetricsOK { + o.Payload = payload + return o +} + +// SetPayload sets the payload to the get config metrics o k response +func (o *GetConfigMetricsOK) SetPayload(payload *models.MetricsConfig) { + o.Payload = payload +} + +// WriteResponse to the client +func (o *GetConfigMetricsOK) WriteResponse(rw http.ResponseWriter, producer runtime.Producer) { + + rw.WriteHeader(200) + if o.Payload != nil { + payload := o.Payload + if err := producer.Produce(rw, payload); err != nil { + panic(err) // let the recovery middleware deal with this + } + } +} + +// GetConfigMetricsBadRequestCode is the HTTP code returned for type GetConfigMetricsBadRequest +const GetConfigMetricsBadRequestCode int = 400 + +/* +GetConfigMetricsBadRequest Malformed arguments for API call + +swagger:response getConfigMetricsBadRequest +*/ +type GetConfigMetricsBadRequest struct { + + /* + In: Body + */ + Payload *models.Error `json:"body,omitempty"` +} + +// NewGetConfigMetricsBadRequest creates GetConfigMetricsBadRequest with default headers values +func NewGetConfigMetricsBadRequest() *GetConfigMetricsBadRequest { + + return &GetConfigMetricsBadRequest{} +} + +// WithPayload adds the payload to the get config metrics bad request response +func (o *GetConfigMetricsBadRequest) WithPayload(payload *models.Error) *GetConfigMetricsBadRequest { + o.Payload = payload + return o +} + +// SetPayload sets the payload to the get config metrics bad request response +func (o *GetConfigMetricsBadRequest) SetPayload(payload *models.Error) { + o.Payload = payload +} + +// WriteResponse to the client +func (o *GetConfigMetricsBadRequest) WriteResponse(rw http.ResponseWriter, producer runtime.Producer) { + + rw.WriteHeader(400) + if o.Payload != nil { + payload := o.Payload + if err := producer.Produce(rw, payload); err != nil { + panic(err) // let the recovery middleware deal with this + } + } +} + +// GetConfigMetricsUnauthorizedCode is the HTTP code returned for type GetConfigMetricsUnauthorized +const GetConfigMetricsUnauthorizedCode int = 401 + +/* +GetConfigMetricsUnauthorized Invalid authentication credentials + +swagger:response getConfigMetricsUnauthorized +*/ +type GetConfigMetricsUnauthorized struct { + + /* + In: Body + */ + Payload *models.Error `json:"body,omitempty"` +} + +// NewGetConfigMetricsUnauthorized creates GetConfigMetricsUnauthorized with default headers values +func NewGetConfigMetricsUnauthorized() *GetConfigMetricsUnauthorized { + + return &GetConfigMetricsUnauthorized{} +} + +// WithPayload adds the payload to the get config metrics unauthorized response +func (o *GetConfigMetricsUnauthorized) WithPayload(payload *models.Error) *GetConfigMetricsUnauthorized { + o.Payload = payload + return o +} + +// SetPayload sets the payload to the get config metrics unauthorized response +func (o *GetConfigMetricsUnauthorized) SetPayload(payload *models.Error) { + o.Payload = payload +} + +// WriteResponse to the client +func (o *GetConfigMetricsUnauthorized) WriteResponse(rw http.ResponseWriter, producer runtime.Producer) { + + rw.WriteHeader(401) + if o.Payload != nil { + payload := o.Payload + if err := producer.Produce(rw, payload); err != nil { + panic(err) // let the recovery middleware deal with this + } + } +} + +// GetConfigMetricsInternalServerErrorCode is the HTTP code returned for type GetConfigMetricsInternalServerError +const GetConfigMetricsInternalServerErrorCode int = 500 + +/* +GetConfigMetricsInternalServerError Internal service error + +swagger:response getConfigMetricsInternalServerError +*/ +type GetConfigMetricsInternalServerError struct { + + /* + In: Body + */ + Payload *models.Error `json:"body,omitempty"` +} + +// NewGetConfigMetricsInternalServerError creates GetConfigMetricsInternalServerError with default headers values +func NewGetConfigMetricsInternalServerError() *GetConfigMetricsInternalServerError { + + return &GetConfigMetricsInternalServerError{} +} + +// WithPayload adds the payload to the get config metrics internal server error response +func (o *GetConfigMetricsInternalServerError) WithPayload(payload *models.Error) *GetConfigMetricsInternalServerError { + o.Payload = payload + return o +} + +// SetPayload sets the payload to the get config metrics internal server error response +func (o *GetConfigMetricsInternalServerError) SetPayload(payload *models.Error) { + o.Payload = payload +} + +// WriteResponse to the client +func (o *GetConfigMetricsInternalServerError) WriteResponse(rw http.ResponseWriter, producer runtime.Producer) { + + rw.WriteHeader(500) + if o.Payload != nil { + payload := o.Payload + if err := producer.Produce(rw, payload); err != nil { + panic(err) // let the recovery middleware deal with this + } + } +} + +// GetConfigMetricsServiceUnavailableCode is the HTTP code returned for type GetConfigMetricsServiceUnavailable +const GetConfigMetricsServiceUnavailableCode int = 503 + +/* +GetConfigMetricsServiceUnavailable Maintenance mode + +swagger:response getConfigMetricsServiceUnavailable +*/ +type GetConfigMetricsServiceUnavailable struct { + + /* + In: Body + */ + Payload *models.Error `json:"body,omitempty"` +} + +// NewGetConfigMetricsServiceUnavailable creates GetConfigMetricsServiceUnavailable with default headers values +func NewGetConfigMetricsServiceUnavailable() *GetConfigMetricsServiceUnavailable { + + return &GetConfigMetricsServiceUnavailable{} +} + +// WithPayload adds the payload to the get config metrics service unavailable response +func (o *GetConfigMetricsServiceUnavailable) WithPayload(payload *models.Error) *GetConfigMetricsServiceUnavailable { + o.Payload = payload + return o +} + +// SetPayload sets the payload to the get config metrics service unavailable response +func (o *GetConfigMetricsServiceUnavailable) SetPayload(payload *models.Error) { + o.Payload = payload +} + +// WriteResponse to the client +func (o *GetConfigMetricsServiceUnavailable) WriteResponse(rw http.ResponseWriter, producer runtime.Producer) { + + rw.WriteHeader(503) + if o.Payload != nil { + payload := o.Payload + if err := producer.Produce(rw, payload); err != nil { + panic(err) // let the recovery middleware deal with this + } + } +} diff --git a/api/restapi/operations/get_config_metrics_urlbuilder.go b/api/restapi/operations/get_config_metrics_urlbuilder.go new file mode 100644 index 000000000..4b11d06d0 --- /dev/null +++ b/api/restapi/operations/get_config_metrics_urlbuilder.go @@ -0,0 +1,87 @@ +// Code generated by go-swagger; DO NOT EDIT. + +package operations + +// This file was generated by the swagger tool. +// Editing this file might prove futile when you re-run the generate command + +import ( + "errors" + "net/url" + golangswaggerpaths "path" +) + +// GetConfigMetricsURL generates an URL for the get config metrics operation +type GetConfigMetricsURL struct { + _basePath string +} + +// WithBasePath sets the base path for this url builder, only required when it's different from the +// base path specified in the swagger spec. +// When the value of the base path is an empty string +func (o *GetConfigMetricsURL) WithBasePath(bp string) *GetConfigMetricsURL { + o.SetBasePath(bp) + return o +} + +// SetBasePath sets the base path for this url builder, only required when it's different from the +// base path specified in the swagger spec. +// When the value of the base path is an empty string +func (o *GetConfigMetricsURL) SetBasePath(bp string) { + o._basePath = bp +} + +// Build a url path and query string +func (o *GetConfigMetricsURL) Build() (*url.URL, error) { + var _result url.URL + + var _path = "/config/metrics" + + _basePath := o._basePath + if _basePath == "" { + _basePath = "/netlox/v1" + } + _result.Path = golangswaggerpaths.Join(_basePath, _path) + + return &_result, nil +} + +// Must is a helper function to panic when the url builder returns an error +func (o *GetConfigMetricsURL) Must(u *url.URL, err error) *url.URL { + if err != nil { + panic(err) + } + if u == nil { + panic("url can't be nil") + } + return u +} + +// String returns the string representation of the path with query string +func (o *GetConfigMetricsURL) String() string { + return o.Must(o.Build()).String() +} + +// BuildFull builds a full url with scheme, host, path and query string +func (o *GetConfigMetricsURL) BuildFull(scheme, host string) (*url.URL, error) { + if scheme == "" { + return nil, errors.New("scheme is required for a full url on GetConfigMetricsURL") + } + if host == "" { + return nil, errors.New("host is required for a full url on GetConfigMetricsURL") + } + + base, err := o.Build() + if err != nil { + return nil, err + } + + base.Scheme = scheme + base.Host = host + return base, nil +} + +// StringFull returns the string representation of a complete url +func (o *GetConfigMetricsURL) StringFull(scheme, host string) string { + return o.Must(o.BuildFull(scheme, host)).String() +} diff --git a/api/restapi/operations/post_config_metrics.go b/api/restapi/operations/post_config_metrics.go new file mode 100644 index 000000000..4a71789d6 --- /dev/null +++ b/api/restapi/operations/post_config_metrics.go @@ -0,0 +1,56 @@ +// Code generated by go-swagger; DO NOT EDIT. + +package operations + +// This file was generated by the swagger tool. +// Editing this file might prove futile when you re-run the generate command + +import ( + "net/http" + + "github.com/go-openapi/runtime/middleware" +) + +// PostConfigMetricsHandlerFunc turns a function with the right signature into a post config metrics handler +type PostConfigMetricsHandlerFunc func(PostConfigMetricsParams) middleware.Responder + +// Handle executing the request and returning a response +func (fn PostConfigMetricsHandlerFunc) Handle(params PostConfigMetricsParams) middleware.Responder { + return fn(params) +} + +// PostConfigMetricsHandler interface for that can handle valid post config metrics params +type PostConfigMetricsHandler interface { + Handle(PostConfigMetricsParams) middleware.Responder +} + +// NewPostConfigMetrics creates a new http.Handler for the post config metrics operation +func NewPostConfigMetrics(ctx *middleware.Context, handler PostConfigMetricsHandler) *PostConfigMetrics { + return &PostConfigMetrics{Context: ctx, Handler: handler} +} + +/* + PostConfigMetrics swagger:route POST /config/metrics postConfigMetrics + +turn on prometheus option +*/ +type PostConfigMetrics struct { + Context *middleware.Context + Handler PostConfigMetricsHandler +} + +func (o *PostConfigMetrics) ServeHTTP(rw http.ResponseWriter, r *http.Request) { + route, rCtx, _ := o.Context.RouteInfo(r) + if rCtx != nil { + *r = *rCtx + } + var Params = NewPostConfigMetricsParams() + if err := o.Context.BindValidRequest(r, route, &Params); err != nil { // bind params + o.Context.Respond(rw, r, route.Produces, route, err) + return + } + + res := o.Handler.Handle(Params) // actually handle the request + o.Context.Respond(rw, r, route.Produces, route, res) + +} diff --git a/api/restapi/operations/post_config_metrics_parameters.go b/api/restapi/operations/post_config_metrics_parameters.go new file mode 100644 index 000000000..d36cedfb7 --- /dev/null +++ b/api/restapi/operations/post_config_metrics_parameters.go @@ -0,0 +1,46 @@ +// Code generated by go-swagger; DO NOT EDIT. + +package operations + +// This file was generated by the swagger tool. +// Editing this file might prove futile when you re-run the swagger generate command + +import ( + "net/http" + + "github.com/go-openapi/errors" + "github.com/go-openapi/runtime/middleware" +) + +// NewPostConfigMetricsParams creates a new PostConfigMetricsParams object +// +// There are no default values defined in the spec. +func NewPostConfigMetricsParams() PostConfigMetricsParams { + + return PostConfigMetricsParams{} +} + +// PostConfigMetricsParams contains all the bound params for the post config metrics operation +// typically these are obtained from a http.Request +// +// swagger:parameters PostConfigMetrics +type PostConfigMetricsParams struct { + + // HTTP Request Object + HTTPRequest *http.Request `json:"-"` +} + +// BindRequest both binds and validates a request, it assumes that complex things implement a Validatable(strfmt.Registry) error interface +// for simple values it will use straight method calls. +// +// To ensure default values, the struct must have been initialized with NewPostConfigMetricsParams() beforehand. +func (o *PostConfigMetricsParams) BindRequest(r *http.Request, route *middleware.MatchedRoute) error { + var res []error + + o.HTTPRequest = r + + if len(res) > 0 { + return errors.CompositeValidationError(res...) + } + return nil +} diff --git a/api/restapi/operations/post_config_metrics_responses.go b/api/restapi/operations/post_config_metrics_responses.go new file mode 100644 index 000000000..74462966f --- /dev/null +++ b/api/restapi/operations/post_config_metrics_responses.go @@ -0,0 +1,219 @@ +// Code generated by go-swagger; DO NOT EDIT. + +package operations + +// This file was generated by the swagger tool. +// Editing this file might prove futile when you re-run the swagger generate command + +import ( + "net/http" + + "github.com/go-openapi/runtime" + + "github.com/loxilb-io/loxilb/api/models" +) + +// PostConfigMetricsNoContentCode is the HTTP code returned for type PostConfigMetricsNoContent +const PostConfigMetricsNoContentCode int = 204 + +/* +PostConfigMetricsNoContent OK + +swagger:response postConfigMetricsNoContent +*/ +type PostConfigMetricsNoContent struct { +} + +// NewPostConfigMetricsNoContent creates PostConfigMetricsNoContent with default headers values +func NewPostConfigMetricsNoContent() *PostConfigMetricsNoContent { + + return &PostConfigMetricsNoContent{} +} + +// WriteResponse to the client +func (o *PostConfigMetricsNoContent) WriteResponse(rw http.ResponseWriter, producer runtime.Producer) { + + rw.Header().Del(runtime.HeaderContentType) //Remove Content-Type on empty responses + + rw.WriteHeader(204) +} + +// PostConfigMetricsBadRequestCode is the HTTP code returned for type PostConfigMetricsBadRequest +const PostConfigMetricsBadRequestCode int = 400 + +/* +PostConfigMetricsBadRequest Malformed arguments for API call + +swagger:response postConfigMetricsBadRequest +*/ +type PostConfigMetricsBadRequest struct { + + /* + In: Body + */ + Payload *models.Error `json:"body,omitempty"` +} + +// NewPostConfigMetricsBadRequest creates PostConfigMetricsBadRequest with default headers values +func NewPostConfigMetricsBadRequest() *PostConfigMetricsBadRequest { + + return &PostConfigMetricsBadRequest{} +} + +// WithPayload adds the payload to the post config metrics bad request response +func (o *PostConfigMetricsBadRequest) WithPayload(payload *models.Error) *PostConfigMetricsBadRequest { + o.Payload = payload + return o +} + +// SetPayload sets the payload to the post config metrics bad request response +func (o *PostConfigMetricsBadRequest) SetPayload(payload *models.Error) { + o.Payload = payload +} + +// WriteResponse to the client +func (o *PostConfigMetricsBadRequest) WriteResponse(rw http.ResponseWriter, producer runtime.Producer) { + + rw.WriteHeader(400) + if o.Payload != nil { + payload := o.Payload + if err := producer.Produce(rw, payload); err != nil { + panic(err) // let the recovery middleware deal with this + } + } +} + +// PostConfigMetricsUnauthorizedCode is the HTTP code returned for type PostConfigMetricsUnauthorized +const PostConfigMetricsUnauthorizedCode int = 401 + +/* +PostConfigMetricsUnauthorized Invalid authentication credentials + +swagger:response postConfigMetricsUnauthorized +*/ +type PostConfigMetricsUnauthorized struct { + + /* + In: Body + */ + Payload *models.Error `json:"body,omitempty"` +} + +// NewPostConfigMetricsUnauthorized creates PostConfigMetricsUnauthorized with default headers values +func NewPostConfigMetricsUnauthorized() *PostConfigMetricsUnauthorized { + + return &PostConfigMetricsUnauthorized{} +} + +// WithPayload adds the payload to the post config metrics unauthorized response +func (o *PostConfigMetricsUnauthorized) WithPayload(payload *models.Error) *PostConfigMetricsUnauthorized { + o.Payload = payload + return o +} + +// SetPayload sets the payload to the post config metrics unauthorized response +func (o *PostConfigMetricsUnauthorized) SetPayload(payload *models.Error) { + o.Payload = payload +} + +// WriteResponse to the client +func (o *PostConfigMetricsUnauthorized) WriteResponse(rw http.ResponseWriter, producer runtime.Producer) { + + rw.WriteHeader(401) + if o.Payload != nil { + payload := o.Payload + if err := producer.Produce(rw, payload); err != nil { + panic(err) // let the recovery middleware deal with this + } + } +} + +// PostConfigMetricsInternalServerErrorCode is the HTTP code returned for type PostConfigMetricsInternalServerError +const PostConfigMetricsInternalServerErrorCode int = 500 + +/* +PostConfigMetricsInternalServerError Internal service error + +swagger:response postConfigMetricsInternalServerError +*/ +type PostConfigMetricsInternalServerError struct { + + /* + In: Body + */ + Payload *models.Error `json:"body,omitempty"` +} + +// NewPostConfigMetricsInternalServerError creates PostConfigMetricsInternalServerError with default headers values +func NewPostConfigMetricsInternalServerError() *PostConfigMetricsInternalServerError { + + return &PostConfigMetricsInternalServerError{} +} + +// WithPayload adds the payload to the post config metrics internal server error response +func (o *PostConfigMetricsInternalServerError) WithPayload(payload *models.Error) *PostConfigMetricsInternalServerError { + o.Payload = payload + return o +} + +// SetPayload sets the payload to the post config metrics internal server error response +func (o *PostConfigMetricsInternalServerError) SetPayload(payload *models.Error) { + o.Payload = payload +} + +// WriteResponse to the client +func (o *PostConfigMetricsInternalServerError) WriteResponse(rw http.ResponseWriter, producer runtime.Producer) { + + rw.WriteHeader(500) + if o.Payload != nil { + payload := o.Payload + if err := producer.Produce(rw, payload); err != nil { + panic(err) // let the recovery middleware deal with this + } + } +} + +// PostConfigMetricsServiceUnavailableCode is the HTTP code returned for type PostConfigMetricsServiceUnavailable +const PostConfigMetricsServiceUnavailableCode int = 503 + +/* +PostConfigMetricsServiceUnavailable Maintenance mode + +swagger:response postConfigMetricsServiceUnavailable +*/ +type PostConfigMetricsServiceUnavailable struct { + + /* + In: Body + */ + Payload *models.Error `json:"body,omitempty"` +} + +// NewPostConfigMetricsServiceUnavailable creates PostConfigMetricsServiceUnavailable with default headers values +func NewPostConfigMetricsServiceUnavailable() *PostConfigMetricsServiceUnavailable { + + return &PostConfigMetricsServiceUnavailable{} +} + +// WithPayload adds the payload to the post config metrics service unavailable response +func (o *PostConfigMetricsServiceUnavailable) WithPayload(payload *models.Error) *PostConfigMetricsServiceUnavailable { + o.Payload = payload + return o +} + +// SetPayload sets the payload to the post config metrics service unavailable response +func (o *PostConfigMetricsServiceUnavailable) SetPayload(payload *models.Error) { + o.Payload = payload +} + +// WriteResponse to the client +func (o *PostConfigMetricsServiceUnavailable) WriteResponse(rw http.ResponseWriter, producer runtime.Producer) { + + rw.WriteHeader(503) + if o.Payload != nil { + payload := o.Payload + if err := producer.Produce(rw, payload); err != nil { + panic(err) // let the recovery middleware deal with this + } + } +}