Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Deep service insight endpoint #3261

Merged
merged 71 commits into from
Dec 5, 2022
Merged
Show file tree
Hide file tree
Changes from 67 commits
Commits
Show all changes
71 commits
Select commit Hold shift + click to select a range
c2ed0e6
poc for healthcheck inside NIC
haywoodsh Nov 6, 2022
b759a84
add listener.go
haywoodsh Nov 9, 2022
ea63d10
Add changes discussed in the meeting
jjngx Nov 9, 2022
3405d0a
Add initial implementation of the server
jjngx Nov 9, 2022
e0bebd0
Minor syntax update
jjngx Nov 9, 2022
852e577
Add a web server and a router
jjngx Nov 10, 2022
482a673
Refactor, handle err, simplify
jjngx Nov 10, 2022
b7257d1
Updated docs
jjngx Nov 10, 2022
dc5050c
Update data type expected by log message
jjngx Nov 10, 2022
0877b5a
Add TLS support for the healthcheck server
jjngx Nov 11, 2022
b2f1b91
Fix typos
jjngx Nov 11, 2022
f08179c
Add placeholders for server tests
jjngx Nov 11, 2022
2c0fc9a
Check if hostname exist and handle http 404
jjngx Nov 11, 2022
fa830da
Update url path to include 'probe'
jjngx Nov 11, 2022
ae81944
Refactor handler, add tests
jjngx Nov 12, 2022
8f56d3f
create flags for health probe
haywoodsh Nov 14, 2022
65eff0b
Handle errors from starting servers
jjngx Nov 14, 2022
1938451
Support min TLS v1.0
jjngx Nov 14, 2022
b9e2d5f
Update min TLS version to 1.2
jjngx Nov 15, 2022
3ee5480
Simplify package structure
jjngx Nov 15, 2022
dcebe1c
Remove double import
jjngx Nov 15, 2022
6ba4114
Round flow to the left
jjngx Nov 15, 2022
d4fa8fb
Sanitize hostname param
jjngx Nov 16, 2022
9c01b84
Use different var name for sanitized input
jjngx Nov 16, 2022
b4270c0
Sanitize params
jjngx Nov 17, 2022
4d8e455
Add a middleware for hostname length validation
jjngx Nov 17, 2022
04a6b61
Remove unused dependency
jjngx Nov 17, 2022
ae9c6d6
Fix linter issues
jjngx Nov 17, 2022
a7ea187
Simplify healthcheck server and tests
jjngx Nov 21, 2022
1f4575b
Merge branch 'main' into feat/deep-service-insight-endpoint
jjngx Nov 21, 2022
3982350
Update dependencies
jjngx Nov 21, 2022
0b33646
Remove hostname validation
jjngx Nov 23, 2022
c146e69
Rename healthcheck to service insight
jjngx Nov 23, 2022
bab4d7d
Update docs for Service Insight feature
jjngx Nov 23, 2022
2c9c40e
Add log info staring healthcheck
jjngx Nov 24, 2022
fb74e12
Use more meaningful var name
jjngx Nov 24, 2022
aca4d38
Update log verbosity
jjngx Nov 24, 2022
2a3fc28
Initial integration test for healthcheck service
jjngx Nov 25, 2022
18cad3a
[pre-commit.ci] auto fixes from pre-commit.com hooks
pre-commit-ci[bot] Nov 25, 2022
5604792
add service-insight port to loadbalancer svc yaml
vepatel Nov 28, 2022
094d93f
Use TLS v1.0 as a min version
jjngx Nov 28, 2022
9ffc97d
mark test as N+ only
vepatel Nov 28, 2022
df4ae66
Update err message
jjngx Nov 28, 2022
5d769a2
Merge branch 'main' into feat/deep-service-insight-endpoint
jjngx Nov 28, 2022
38d9781
Add service insight options to IC manifest
jjngx Nov 29, 2022
384cc55
Add service-insight documentation page
jjngx Nov 29, 2022
91a45d5
[pre-commit.ci] auto fixes from pre-commit.com hooks
pre-commit-ci[bot] Nov 29, 2022
e6defb3
Update Helm chart for Service Insight
jjngx Nov 29, 2022
682026f
Apply PR feedback
jjngx Nov 29, 2022
9727bb5
Merge branch 'main' into feat/deep-service-insight-endpoint
jjngx Nov 29, 2022
d93c007
Add test for https endpoint
vepatel Nov 29, 2022
ab945e8
add test for replica change
vepatel Nov 30, 2022
6c04db4
[pre-commit.ci] auto fixes from pre-commit.com hooks
pre-commit-ci[bot] Nov 30, 2022
34b9a8b
Merge branch 'main' into feat/deep-service-insight-endpoint
jjngx Nov 30, 2022
2387e1d
Update docs/content/logging-and-monitoring/service-insight.md
jjngx Dec 1, 2022
1a449fc
Update docs/content/configuration/global-configuration/command-line-a…
jjngx Dec 1, 2022
167fbfc
Update docs/content/logging-and-monitoring/service-insight.md
jjngx Dec 1, 2022
923ccb1
Update docs/content/logging-and-monitoring/service-insight.md
jjngx Dec 1, 2022
46ac908
Update docs/content/logging-and-monitoring/service-insight.md
jjngx Dec 1, 2022
9d08f94
Update docs/content/logging-and-monitoring/service-insight.md
jjngx Dec 1, 2022
ee0abb0
Update docs/content/logging-and-monitoring/service-insight.md
jjngx Dec 1, 2022
bd25635
Update docs/content/configuration/global-configuration/command-line-a…
jjngx Dec 1, 2022
0aafb41
Update docs/content/configuration/global-configuration/command-line-a…
jjngx Dec 1, 2022
2075150
Update docs/content/configuration/global-configuration/command-line-a…
jjngx Dec 1, 2022
86903f4
Update docs/content/logging-and-monitoring/service-insight.md
jjngx Dec 1, 2022
faf8245
Update docs/content/logging-and-monitoring/service-insight.md
jjngx Dec 1, 2022
aec41bc
Increase wait time in integration test
jjngx Dec 2, 2022
cbf43e6
Update deployments/helm-chart/values.yaml
jjngx Dec 2, 2022
8ba1029
add conditional check on waitTime for api response
vepatel Dec 2, 2022
48b3d72
Add docs for Helm installation
jjngx Dec 5, 2022
1ecc63e
Merge branch 'main' into feat/deep-service-insight-endpoint
jjngx Dec 5, 2022
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
19 changes: 19 additions & 0 deletions cmd/nginx-ingress/flags.go
Original file line number Diff line number Diff line change
Expand Up @@ -139,6 +139,15 @@ var (
prometheusMetricsListenPort = flag.Int("prometheus-metrics-listen-port", 9113,
"Set the port where the Prometheus metrics are exposed. [1024 - 65535]")

enableServiceInsight = flag.Bool("enable-service-insight", false,
`Enable service insight for external load balancers. Requires -nginx-plus`)

serviceInsightTLSSecretName = flag.String("service-insight-tls-secret", "",
`A Secret with a TLS certificate and key for TLS termination of the service insight.`)

serviceInsightListenPort = flag.Int("service-insight-listen-port", 9114,
"Set the port where the Service Insight stats are exposed. Requires -nginx-plus. [1024 - 65535]")

enableCustomResources = flag.Bool("enable-custom-resources", true,
"Enable custom resources")

Expand Down Expand Up @@ -250,6 +259,11 @@ func parseFlags() {
*enableLatencyMetrics = false
}

if *enableServiceInsight && !*nginxPlus {
glog.Warning("enable-service-insight flag support is for NGINX Plus, service insight endpoint will not be exposed")
*enableServiceInsight = false
}

if *enableCertManager && !*enableCustomResources {
glog.Fatal("enable-cert-manager flag requires -enable-custom-resources")
}
Expand Down Expand Up @@ -352,6 +366,11 @@ func validationChecks() {
glog.Fatalf("Invalid value for ready-status-port: %v", readyStatusPortValidationError)
}

healthProbePortValidationError := validatePort(*serviceInsightListenPort)
if healthProbePortValidationError != nil {
glog.Fatalf("Invalid value for service-insight-listen-port: %v", metricsPortValidationError)
}

var err error
allowedCIDRs, err = parseNginxStatusAllowCIDRs(*nginxStatusAllowCIDRs)
if err != nil {
Expand Down
25 changes: 25 additions & 0 deletions cmd/nginx-ingress/main.go
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,7 @@ import (
"github.com/nginxinc/kubernetes-ingress/internal/configs"
"github.com/nginxinc/kubernetes-ingress/internal/configs/version1"
"github.com/nginxinc/kubernetes-ingress/internal/configs/version2"
"github.com/nginxinc/kubernetes-ingress/internal/healthcheck"
"github.com/nginxinc/kubernetes-ingress/internal/k8s"
"github.com/nginxinc/kubernetes-ingress/internal/k8s/secrets"
"github.com/nginxinc/kubernetes-ingress/internal/metrics"
Expand Down Expand Up @@ -120,6 +121,10 @@ func main() {
transportServerValidator := cr_validation.NewTransportServerValidator(*enableTLSPassthrough, *enableSnippets, *nginxPlus)
virtualServerValidator := cr_validation.NewVirtualServerValidator(cr_validation.IsPlus(*nginxPlus), cr_validation.IsDosEnabled(*appProtectDos), cr_validation.IsCertManagerEnabled(*enableCertManager), cr_validation.IsExternalDNSEnabled(*enableExternalDNS))

if *enableServiceInsight {
createHealthProbeEndpoint(kubeClient, plusClient, cnf)
}

lbcInput := k8s.NewLoadBalancerControllerInput{
KubeClient: kubeClient,
ConfClient: confClient,
Expand Down Expand Up @@ -446,6 +451,10 @@ func createGlobalConfigurationValidator() *cr_validation.GlobalConfigurationVali
forbiddenListenerPorts[*prometheusMetricsListenPort] = true
}

if *enableServiceInsight {
forbiddenListenerPorts[*serviceInsightListenPort] = true
}

return cr_validation.NewGlobalConfigurationValidator(forbiddenListenerPorts)
}

Expand Down Expand Up @@ -674,6 +683,22 @@ func createPlusAndLatencyCollectors(
return plusCollector, syslogListener, lc
}

func createHealthProbeEndpoint(kubeClient *kubernetes.Clientset, plusClient *client.NginxClient, cnf *configs.Configurator) {
if !*enableServiceInsight {
return
}
var serviceInsightSecret *api_v1.Secret
var err error

jjngx marked this conversation as resolved.
Show resolved Hide resolved
if *serviceInsightTLSSecretName != "" {
serviceInsightSecret, err = getAndValidateSecret(kubeClient, *serviceInsightTLSSecretName)
if err != nil {
glog.Fatalf("Error trying to get the service insight TLS secret %v: %v", *serviceInsightTLSSecretName, err)
}
}
go healthcheck.RunHealthCheck(*serviceInsightListenPort, plusClient, cnf, serviceInsightSecret)
}

func processGlobalConfiguration() {
if *globalConfiguration != "" {
_, _, err := k8s.ParseNamespaceName(*globalConfiguration)
Expand Down
3 changes: 3 additions & 0 deletions deployments/deployment/nginx-plus-ingress.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -32,6 +32,8 @@ spec:
containerPort: 8081
- name: prometheus
containerPort: 9113
- name: service-insight
containerPort: 9114
readinessProbe:
httpGet:
path: /nginx-ready
Expand Down Expand Up @@ -75,4 +77,5 @@ spec:
#- -report-ingress-status
#- -external-service=nginx-ingress
#- -enable-prometheus-metrics
#- -enable-service-insight
#- -global-configuration=$(POD_NAMESPACE)/nginx-configuration
7 changes: 7 additions & 0 deletions deployments/helm-chart/templates/controller-daemonset.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -94,6 +94,10 @@ spec:
- name: prometheus
containerPort: {{ .Values.prometheus.port }}
{{- end }}
{{- if .Values.serviceInsight.create }}
- name: service-insight
containerPort: {{ .Values.serviceInsight.port }}
{{- end }}
{{- if .Values.controller.readyStatus.enable }}
- name: readiness-port
containerPort: {{ .Values.controller.readyStatus.port }}
Expand Down Expand Up @@ -199,6 +203,9 @@ spec:
- -enable-prometheus-metrics={{ .Values.prometheus.create }}
- -prometheus-metrics-listen-port={{ .Values.prometheus.port }}
- -prometheus-tls-secret={{ .Values.prometheus.secret }}
- -enable-service-insight={{ .Values.serviceInsight.create }}
- -service-insight-listen-port={{ .Values.serviceInsight.port }}
- -service-insight-tls-secret={{ .Values.serviceInsight.secret }}
- -enable-custom-resources={{ .Values.controller.enableCustomResources }}
- -enable-snippets={{ .Values.controller.enableSnippets }}
- -include-year={{ .Values.controller.includeYear }}
Expand Down
7 changes: 7 additions & 0 deletions deployments/helm-chart/templates/controller-deployment.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -97,6 +97,10 @@ spec:
- name: prometheus
containerPort: {{ .Values.prometheus.port }}
{{- end }}
{{- if .Values.serviceInsight.create }}
- name: service-insight
containerPort: {{ .Values.serviceInsight.port }}
{{- end }}
{{- if .Values.controller.readyStatus.enable }}
- name: readiness-port
containerPort: {{ .Values.controller.readyStatus.port }}
Expand Down Expand Up @@ -202,6 +206,9 @@ spec:
- -enable-prometheus-metrics={{ .Values.prometheus.create }}
- -prometheus-metrics-listen-port={{ .Values.prometheus.port }}
- -prometheus-tls-secret={{ .Values.prometheus.secret }}
- -enable-service-insight={{ .Values.serviceInsight.create }}
- -service-insight-listen-port={{ .Values.serviceInsight.port }}
- -service-insight-tls-secret={{ .Values.serviceInsight.secret }}
- -enable-custom-resources={{ .Values.controller.enableCustomResources }}
- -enable-snippets={{ .Values.controller.enableSnippets }}
- -include-year={{ .Values.controller.includeYear }}
Expand Down
57 changes: 57 additions & 0 deletions deployments/helm-chart/values.schema.json
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@
"controller",
"rbac",
"prometheus",
"serviceInsight",
"nginxServiceMesh"
],
"properties": {
Expand Down Expand Up @@ -1436,6 +1437,56 @@
}
]
},
"serviceInsight": {
"type": "object",
"default": {},
"title": "The Service Insight Schema",
"required": [
"create"
],
"properties": {
"create": {
"type": "boolean",
"default": false,
"title": "The create",
"examples": [
true
]
},
"port": {
"type": "integer",
"default": 9114,
"title": "The port",
"examples": [
9114
]
},
"secret": {
"type": "string",
"default": "",
"title": "The secret",
"examples": [
""
]
},
"scheme": {
"type": "string",
"default": "http",
"title": "The scheme",
"examples": [
"http"
]
}
},
"examples": [
{
"create": true,
"port": 9114,
"secret": "",
"scheme": "http"
}
]
},
"nginxServiceMesh": {
"type": "object",
"default": {},
Expand Down Expand Up @@ -1622,6 +1673,12 @@
"secret": "",
"scheme": "http"
},
"serviceInsight": {
"create": true,
"port": 9114,
"secret": "",
"scheme": "http"
},
"nginxServiceMesh": {
"enable": false,
"enableEgress": false
Expand Down
13 changes: 13 additions & 0 deletions deployments/helm-chart/values.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -433,6 +433,19 @@ prometheus:
## Configures the HTTP scheme used.
scheme: http

serviceInsight:
## Expose NGINX Plus Service Insight endpoint.
create: true
jjngx marked this conversation as resolved.
Show resolved Hide resolved

## Configures the port to expose endpoint.
port: 9114

## Specifies the namespace/name of a Kubernetes TLS Secret which will be used to protect the Service Insight endpoint.
secret: ""

## Configures the HTTP scheme used.
scheme: http

nginxServiceMesh:
## Enables integration with NGINX Service Mesh.
## Requires controller.nginxplus
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -336,7 +336,30 @@ Format: `[1024 - 65535]` (default `9113`)

A Secret with a TLS certificate and key for TLS termination of the Prometheus metrics endpoint.

* If the argument is not set, the prometheus endpoint will not use a TLS connection.
* If the argument is not set, the Prometheus endpoint will not use a TLS connection.
* If the argument is set, but the Ingress Controller is not able to fetch the Secret from Kubernetes API, the Ingress Controller will fail to start.
 
<a name="cmdoption-enable-service-insight"></a>

### -enable-service-insight

Exposes the Service Insight endpoint for Ingress Controller.
&nbsp;
<a name="cmdoption-service-insight-listen-port"></a>

### -service-insight-listen-port `<int>`

Sets the port where the Service Insight is exposed.

Format: `[1024 - 65535]` (default `9114`)
&nbsp;
<a name="cmdoption-service-insight-tls-secret"></a>

### -service-insight-tls-secret `<string>`

A Secret with a TLS certificate and key for TLS termination of the Service Insight endpoint.

* If the argument is not set, the Service Insight endpoint will not use a TLS connection.
* If the argument is set, but the Ingress Controller is not able to fetch the Secret from Kubernetes API, the Ingress Controller will fail to start.

Format: `<namespace>/<name>`
Expand Down
58 changes: 58 additions & 0 deletions docs/content/logging-and-monitoring/service-insight.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,58 @@
---
title: Service Insight

description: "The Ingress Controller exposes the Service Insight endpoint."
weight: 2100
doctypes: [""]
aliases:
- /service-insight/
toc: true
docs: "DOCS-000"
---


The Ingress Controller exposes an endpoint and provides host statistics for Virtual Servers (VS).
It exposes data in JSON format and returns HTTP status codes.
The response body holds information about the total, down and the unhealthy number of
upstreams associated with the hostname.
Returned HTTP codes indicate the health of the upstreams (service).

The service is not healthy (HTTP response code different than 200 OK) if all upstreams are unhealthy.
The service is healthy if at least one upstream is healthy. In this case, the endpoint returns HTTP code 200 OK.



## Enabling Service Insight Endpoint

If you're using *Kubernetes manifests* (Deployment or DaemonSet) to install the Ingress Controller, to enable the Service Insight endpoint:
1. Run the Ingress Controller with the `-enable-service-insight` [command-line argument](/nginx-ingress-controller/configuration/global-configuration/command-line-arguments). This will expose the Ingress Controller endpoint via the path `/probe/{hostname}` on port `9114` (customizable with the `-service-insight-listen-port` command-line argument).
1. To enable TLS for the Service Insight endpoint, configure the `-service-insight-tls-secret` cli argument with the namespace and name of a TLS Secret.
1. Add the Service Insight port to the list of the ports of the Ingress Controller container in the template of the Ingress Controller pod:
```yaml
- name: service-insight
containerPort: 9114
```

If you're using *Helm* to install the Ingress Controller, to enable Service Insight endpoint, configure the `serviceinsight.*` parameters of the Helm chart. See the [Installation with Helm](/nginx-ingress-controller/installation/installation-with-helm) doc.

## Available Statistics and HTTP Response Codes

The Service Insight provides the following statistics:

* Total number of VS
* Number of VS in 'Down' state
* Number of VS in 'Healthy' state

These statistics are returned as JSON:

```json
{ "Total": <int>, "Up": <int>, "Unhealthy": <int> }
```

Response codes:

* HTTP 200 OK - Service is healthy
* HTTP 404 - No upstreams/VS found for the requested hostname
* HTTP 503 Service Unavailable - The service is down (All upstreams/VS are "Unhealthy")

**Note**: wildcards in hostnames are not supported at the moment.
1 change: 1 addition & 0 deletions go.mod
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,7 @@ require (
github.com/aws/aws-sdk-go-v2/config v1.18.3
github.com/aws/aws-sdk-go-v2/service/marketplacemetering v1.13.24
github.com/cert-manager/cert-manager v1.10.1
github.com/go-chi/chi v1.5.4
github.com/golang-jwt/jwt/v4 v4.4.2
github.com/golang/glog v1.0.0
github.com/google/go-cmp v0.5.9
Expand Down
2 changes: 2 additions & 0 deletions go.sum
Original file line number Diff line number Diff line change
Expand Up @@ -112,6 +112,8 @@ github.com/fsnotify/fsnotify v1.5.4 h1:jRbGcIw6P2Meqdwuo0H1p6JVLbL5DHKAKlYndzMwV
github.com/ghodss/yaml v1.0.0/go.mod h1:4dBDuWmgqj2HViK6kFavaiC9ZROes6MMH2rRYeMEF04=
github.com/go-asn1-ber/asn1-ber v1.5.4 h1:vXT6d/FNDiELJnLb6hGNa309LMsrCoYFvpwHDF0+Y1A=
github.com/go-asn1-ber/asn1-ber v1.5.4/go.mod h1:hEBeB/ic+5LoWskz+yKT7vGhhPYkProFKoKdwZRWMe0=
github.com/go-chi/chi v1.5.4 h1:QHdzF2szwjqVV4wmByUnTcsbIg7UGaQ0tPF2t5GcAIs=
github.com/go-chi/chi v1.5.4/go.mod h1:uaf8YgoFazUOkPBG7fxPftUylNumIev9awIWOENIuEg=
github.com/go-kit/kit v0.8.0/go.mod h1:xBxKIO96dXMWWy0MnWVtmwkA9/13aqxPnvrjFYMA2as=
github.com/go-kit/kit v0.9.0/go.mod h1:xBxKIO96dXMWWy0MnWVtmwkA9/13aqxPnvrjFYMA2as=
github.com/go-kit/log v0.1.0/go.mod h1:zbhenjAZHb184qTLMA9ZjW7ThYL0H2mk7Q6pNt4vbaY=
Expand Down
37 changes: 37 additions & 0 deletions internal/configs/configurator.go
Original file line number Diff line number Diff line change
Expand Up @@ -265,6 +265,43 @@ func (cnf *Configurator) AddOrUpdateIngress(ingEx *IngressEx) (Warnings, error)
return warnings, nil
}

// GetVirtualServerForHost takes a hostname and returns a VS for the given hostname.
func (cnf *Configurator) GetVirtualServerForHost(hostname string) *conf_v1.VirtualServer {
for _, vsEx := range cnf.virtualServers {
if vsEx.VirtualServer.Spec.Host == hostname {
return vsEx.VirtualServer
}
}
return nil
}

// GetUpstreamsforVirtualServer takes VS and returns a slice of upstreams.
func (cnf *Configurator) GetUpstreamsforVirtualServer(vs *conf_v1.VirtualServer) []string {
glog.V(3).Infof("Get upstreamName for vs: %s", vs.Spec.Host)
upstreamNames := make([]string, 0, len(vs.Spec.Upstreams))

virtualServerUpstreamNamer := NewUpstreamNamerForVirtualServer(vs)

for _, u := range vs.Spec.Upstreams {
upstreamName := virtualServerUpstreamNamer.GetNameForUpstream(u.Name)
glog.V(3).Infof("upstream: %s, upstreamName: %s", u.Name, upstreamName)
upstreamNames = append(upstreamNames, upstreamName)
}
return upstreamNames
}

// GetUpstreamsforHost takes a hostname and returns a slice of upstreams
// for the given hostname.
func (cnf *Configurator) GetUpstreamsforHost(hostname string) []string {
glog.V(3).Infof("Get upstream for host: %s", hostname)
vs := cnf.GetVirtualServerForHost(hostname)

if vs != nil {
return cnf.GetUpstreamsforVirtualServer(vs)
}
return nil
}

func (cnf *Configurator) addOrUpdateIngress(ingEx *IngressEx) (Warnings, error) {
apResources := cnf.updateApResources(ingEx)

Expand Down
Loading