Skipper Ingress Usage¶
This documentation is meant for people deploying to Kubernetes Clusters and describes to use Ingress and low level and high level features Skipper provides.
RouteGroups, Skipper Kubernetes native routing object that supports all Skipper features. If you need to create more than one route to your application, RouteGroups should be the default choice, instead of ingress. The documentation contains a section with mapping Ingress to RouteGroups.
Skipper Ingress Annotations¶
Annotation | example data | usage |
---|---|---|
zalando.org/backend-weights | {"my-app-1": 80, "my-app-2": 20} |
blue-green deployments, see also StackSet for more high-level integration |
zalando.org/skipper-filter | consecutiveBreaker(15) |
arbitrary filters |
zalando.org/skipper-predicate | QueryParam("version", "^alpha$") |
arbitrary predicates |
zalando.org/skipper-routes | Method("OPTIONS") -> status(200) -> <shunt> |
extra custom routes, please consider using RouteGroups instead |
zalando.org/ratelimit | ratelimit(50, "1m") |
deprecated, use zalando.org/skipper-filter instead |
zalando.org/skipper-ingress-redirect | "true" |
change the default HTTPS redirect behavior for specific ingresses (true/false) |
zalando.org/skipper-ingress-redirect-code | 301 |
change the default HTTPS redirect code for specific ingresses |
zalando.org/skipper-loadbalancer | consistentHash |
defaults to roundRobin , see available choices |
zalando.org/skipper-backend-protocol | fastcgi |
(experimental) defaults to http , see available choices |
zalando.org/skipper-ingress-path-mode | path-prefix |
(deprecated) please use Ingress version 1 pathType option, which defaults to ImplementationSpecific and does not change the behavior. Skipper’s path-mode defaults to kubernetes-ingress , see available choices, to change the default use -kubernetes-path-mode . |
Supported Service types¶
Ingress backend definitions are services, which have different service types. ClusterIP should be the default for all backend applications that expose via ingress.
Service type | supported | workaround |
---|---|---|
ClusterIP | yes | — |
NodePort | yes | — |
ExternalName | yes | — |
LoadBalancer | no | it should not, because Kubernetes cloud-controller-manager will maintain it |
HTTP Host header routing¶
HTTP host header is defined within the rules host
section and this
route will match by http Host: app-default.example.org
and route to
endpoints selected by the Kubernetes service app-svc
on port 80
.
apiVersion: networking.k8s.io/v1
kind: Ingress
metadata:
name: app
spec:
rules:
- host: app-default.example.org
http:
paths:
- backend:
service:
name: app-svc
port:
number: 80
pathType: ImplementationSpecific
To have 2 routes with different Host
headers serving the same
backends, you have to specify 2 entries in the rules section, as
Kubernetes defined the ingress spec. This is often used in cases of
migrations from one domain to another one or migrations to or from
bare metal datacenters to cloud providers or inter cloud or intra
cloud providers migrations. Examples are AWS account migration, AWS to
GCP migration, GCP to bare metal migration or bare metal to Alibaba
Cloud migration.
apiVersion: networking.k8s.io/v1
kind: Ingress
metadata:
name: app
spec:
rules:
- host: app-default.example.org
http:
paths:
- backend:
service:
name: app-svc
port:
number: 80
pathType: ImplementationSpecific
- host: foo.example.org
http:
paths:
- backend:
service:
name: app-svc
port:
number: 80
pathType: ImplementationSpecific
Multiple Ingresses defining the same route¶
Warning
If multiple ingresses define the same host and the same predicates, traffic routing may become non-deterministic.
Consider the following two ingresses which have the same hostname and therefore overlap. In Skipper the routing of this is currently undefined as skipper doesn’t pick one over the other, but just creates routes (possible overlapping) for each of the ingresses.
In this example (taken from the issues we saw in production clusters) one ingress points to a service with no endpoints and the other to a service with endpoints. (Most likely service-x was renamed to service-x-live and the old ingress was forgot).
apiVersion: networking.k8s.io/v1
kind: Ingress
metadata:
name: service-x
spec:
rules:
- host: service-x.example.org
http:
paths:
- backend:
service:
name: service-x # this service has 0 endpoints
port:
number: 80
pathType: ImplementationSpecific
apiVersion: networking.k8s.io/v1
kind: Ingress
metadata:
name: service-x-live
spec:
rules:
- host: service-x.example.org
http:
paths:
- backend:
service:
name: service-x-live
port:
number: 80
pathType: ImplementationSpecific
Ingress path handling¶
Skipper supports all Kubernetes path-types as documented in Kubernetes documentation.
Ingress paths can be interpreted in five different modes:
pathType: Prefix
results in PathSubtree predicate)pathType: Exact
results in Path predicate)pathType: ImplementationSpecific
- based on the Kubernetes ingress specification
- as plain regular expression
- as a path prefix (same as
pathType: Prefix
and results in PathSubtree)
The default is 3.1 the Kubernetes ingress mode. It can be changed by startup option -kubernetes-path-mode
to any of the other modes. The individual ingress rules can also override the
default behavior with the zalando.org/skipper-ingress-path-mode
annotation. You can
also set for each path rule a different Kubernetes pathType
like Prefix
and Exact
.
E.g.:
zalando.org/skipper-ingress-path-mode: path-prefix # Skipper specific
pathType: Prefix # ingress v1
Kubernetes ingress specification base path¶
By default, the ingress path mode is set to kubernetes-ingress
,
which is interpreted as a regular expression with a mandatory leading
/
, and is automatically prepended by a ^
control character,
enforcing that the path has to be at the start of the incoming request
path.
Plain regular expression¶
When the path mode is set to path-regexp
, the ingress path is interpreted similar
to the default Kubernetes ingress specification way, but is not prepended by the ^
control character.
Path prefix¶
When the path mode is set to path-prefix
, the ingress path is not a regular
expression. As an example, /foo/bar
will match /foo/bar
or /foo/bar/baz
, but
won’t match /foo/barooz
.
When PathPrefix is used, the path matching becomes deterministic when a request could match more than one ingress routes otherwise.
In PathPrefix mode, when a Path or PathSubtree predicate is set in an annotation, the predicate in the annotation takes precedence over the normal ingress path.
Filters and Predicates¶
- Filters can manipulate http data, which is not possible in the ingress spec.
- Predicates change the route matching, beyond normal ingress definitions
This example shows how to add predicates and filters:
apiVersion: networking.k8s.io/v1
kind: Ingress
metadata:
annotations:
zalando.org/skipper-predicate: predicate1 && predicate2 && .. && predicateN
zalando.org/skipper-filter: filter1 -> filter2 -> .. -> filterN
name: app
spec:
rules:
- host: app-default.example.org
http:
paths:
- backend:
service:
name: app-svc
port:
number: 80
pathType: ImplementationSpecific
Custom Routes¶
Please consider using RouteGroups, instead of custom routes!
Custom routes is a way of extending the default routes configured for an ingress resource.
Sometimes you just want to return a header, redirect or even static
html content. You can return from Skipper without doing a proxy call
to a backend, if you end your filter chain with <shunt>
. The use of
<shunt>
recommends the use in combination with status()
filter, to
not respond with the default http code, which defaults to 404. To
match your custom route with higher priority than your ingress you
also have to add another predicate, for example the Method(“GET”)
predicate to match the route with higher
priority.
Custom routes specified in ingress will always add the Host()
predicate to match the host header specified in
the ingress rules:
. If there is a path:
definition in your
ingress, then it will be based on the Skipper command line parameter
-kubernetes-path-mode
set one of these predicates:
If you have a path:
value defined in your ingress resource, a custom
route is not allowed to use Path()
nor PathSubtree()
predicates.
You will get an error in Skipper logs, similar to:
[APP]time="2019-01-02T13:30:16Z" level=error msg="Failed to add route having 2 path routes: Path(\"/foo/bar\") -> inlineContent(\"custom route\") -> status(200) -> <shunt>"
Note: Predicates coming from
zalando.org/skipper-predicate
won’t be appended to routes generated fromzalando.org/skipper-routes
Redirects¶
Overwrite the current ingress with a redirect¶
Sometimes you want to overwrite the current ingress with a redirect to a nicer downtime page.
The following example shows how to create a temporary redirect with status
code 307 to https://outage.example.org. No requests will pass to your
backend defined, because the created route from the annotation
zalando.org/skipper-routes
will get 3 Predicates
Host("^app-default[.]example[.]org$") && Path("/") && PathRegexp("/")
,
instead of the 2 Predicates
Host("^app-default[.]example[.]org$") && Path("/")
, that will be
created for the ingress backend.
apiVersion: networking.k8s.io/v1
kind: Ingress
metadata:
name: app
namespace: default
annotations:
zalando.org/skipper-routes: |
redirect_app_default: PathRegexp("/") -> redirectTo(307, "https://outage.example.org/") -> <shunt>;
spec:
rules:
- host: "app-default.example.org"
http:
paths:
- path: /
pathType: Prefix
backend:
service:
name: app-svc
port:
number: 80
Redirect a specific path from ingress¶
Sometimes you want to have a redirect from
http://app-default.example.org/myredirect
to
https://somewhere.example.org/another/path
.
The following example shows how to create a permanent redirect with status
code 308 from http://app-default.example.org/myredirect
to
https://somewhere.example.org/another/path
, other paths will not be
redirected and passed to the backend selected by serviceName=app-svc
and
servicePort=80
:
apiVersion: networking.k8s.io/v1
kind: Ingress
metadata:
name: app
namespace: default
annotations:
zalando.org/skipper-routes: |
redirect_app_default: PathRegexp("/myredirect") -> redirectTo(308, "https://somewhere.example.org/another/path") -> <shunt>;
spec:
rules:
- host: "app-default.example.org"
http:
paths:
- path: /
pathType: Prefix
backend:
service:
name: app-svc
port:
number: 80
Return static content¶
The following example sets a response header X: bar
, a response body
<html><body>hello</body></html>
and respond from the ingress
directly with a HTTP status code 200:
zalando.org/skipper-routes: |
PathRegexp("/") -> setResponseHeader("X", "bar") -> inlineContent("<html><body>hello</body></html>") -> status(200) -> <shunt>
Keep in mind that you need a valid backend definition to backends which are available, otherwise Skipper would not accept the entire route definition from the ingress object for safety reasons.
CORS example¶
This example shows how to add a custom route for handling OPTIONS
requests.
apiVersion: networking.k8s.io/v1
kind: Ingress
metadata:
annotations:
zalando.org/skipper-routes: |
Method("OPTIONS") ->
setResponseHeader("Access-Control-Allow-Origin", "*") ->
setResponseHeader("Access-Control-Allow-Methods", "GET, OPTIONS") ->
setResponseHeader("Access-Control-Allow-Headers", "Authorization") ->
status(200) -> <shunt>
name: app
spec:
rules:
- host: app-default.example.org
http:
paths:
- backend:
service:
name: app-svc
port:
number: 80
pathType: ImplementationSpecific
This will generate a custom route for the ingress which looks like this:
Host(/^app-default[.]example[.]org$/) && Method("OPTIONS") ->
setResponseHeader("Access-Control-Allow-Origin", "*") ->
setResponseHeader("Access-Control-Allow-Methods", "GET, OPTIONS") ->
setResponseHeader("Access-Control-Allow-Headers", "Authorization") ->
status(200) -> <shunt>
Multiple routes¶
You can also set multiple routes, but you have to set the IDs
(routename1
, routename2
) of the route as defined in eskip:
zalando.org/skipper-routes: |
routename1: Path("/") -> clientRatelimit(2, "1h") -> inlineContent("A") -> status(200) -> <shunt>;
routename2: Path("/foo") -> clientRatelimit(5, "1h") -> inlineContent("B") -> status(200) -> <shunt>;
Make sure the ;
semicolon is used to terminate the routes, if you
use multiple routes definitions.
Disclaimer: This feature works only with having different Path*
predicates in ingress, if there are no paths rules defined. For
example this will not work:
apiVersion: networking.k8s.io/v1
kind: Ingress
metadata:
name: skipper-ingress
annotations:
kubernetes.io/ingress.class: skipper
zalando.org/skipper-routes: |
redirect1: Path("/foo/") -> redirectTo(308, "/bar/") -> <shunt>;
spec:
rules:
- host: foo.bar
http:
paths:
- path: /something
pathType: Prefix
backend:
service:
name: something
port:
number: 80
- path: /else
pathType: Prefix
backend:
service:
name: else
port:
number: 80
A possible solution is to use skipper’s RouteGroups.
Filters - Basic HTTP manipulations¶
HTTP manipulations are done by using Skipper filters. Changes can be done in the request path, meaning request to your backend or in the response path to the client, which made the request.
The following examples can be used within zalando.org/skipper-filter
annotation.
Add a request Header¶
Add a HTTP header in the request path to your backend.
setRequestHeader("X-Foo", "bar")
Add a response Header¶
Add a HTTP header in the response path of your clients.
setResponseHeader("X-Foo", "bar")
Enable compression¶
Compress responses with accepted encoding (more details here).
compress() // compress all valid MIME types
compress("text/html") // only compress HTML files
compress(11, "text/html") // control the level of compression, 1 = fastest, 11 = best compression (fallback to 9 for gzip), 0 = no compression
Set the Path¶
Change the path in the request path to your backend to /newPath/
.
setPath("/newPath/")
Modify Path¶
Modify the path in the request path from /api/foo
to your backend to /foo
.
modPath("^/api/", "/")
Set the Querystring¶
Set the Querystring in the request path to your backend to ?text=godoc%20skipper
.
setQuery("text", "godoc skipper")
Redirect¶
Create a redirect with HTTP code 301 to https://foo.example.org/.
redirectTo(301, "https://foo.example.org/")
Cookies¶
Set a Cookie in the request path to your backend.
requestCookie("test-session", "abc")
Set a Cookie in the response path of your clients.
responseCookie("test-session", "abc", 31536000)
responseCookie("test-session", "abc", 31536000, "change-only")
// response cookie without HttpOnly:
jsCookie("test-session-info", "abc-debug", 31536000, "change-only")
Authorization¶
Our authentication and authorization tutorial or filter auth godoc shows how to use filters for authorization.
Basic Auth¶
% htpasswd -nbm myName myPassword
basicAuth("/path/to/htpasswd")
basicAuth("/path/to/htpasswd", "My Website")
Bearer Token (OAuth/JWT)¶
OAuth2/JWT tokens can be validated and allowed based on different content of the token. Please check the filter documentation for that:
There are also auth predicates, which will allow you to match a route based on the content of a token:
JWTPayloadAnyKV()
JWTPayloadAllKV()
These are not validating the tokens, which should be done separately by the filters mentioned above.
Diagnosis - Throttling Bandwidth - Latency¶
For diagnosis purpose there are more than 20 filters that enable you to throttle the bandwidth or add latency. For the full list of filters see our diagnostics filters. Examples:
bandwidth(30) // incoming in kb/s
backendBandwidth(30) // outgoing in kb/s
backendLatency(120) // in ms
normalRequestLatency("10ms", "5ms") // normal distribution for request latency as time duration string
logHeader("request") // log all request headers
logBody("response", 1024) // log up to 1024 Bytes of the response body
Flow Id to trace request flows¶
To trace request flows Skipper can generate a unique Flow Id for every HTTP request that it receives. You can then find the trace of the request in all your access logs. Skipper sets the X-Flow-Id header to a unique value. Read more about this in our flowid filter and godoc.
flowId("reuse")
Filters - reliability features¶
Filters can modify http requests and responses. There are plenty of things you can do with them.
Circuitbreaker¶
Consecutive Breaker¶
The consecutiveBreaker filter is a breaker for the ingress route that open if the backend failures for the route reach a value of N (in this example N=15), where N is a mandatory argument of the filter and there are some more optional arguments documented.
consecutiveBreaker(15)
The ingress spec would look like this:
apiVersion: networking.k8s.io/v1
kind: Ingress
metadata:
annotations:
zalando.org/skipper-filter: consecutiveBreaker(15)
name: app
spec:
rules:
- host: app-default.example.org
http:
paths:
- backend:
service:
name: app-svc
port:
number: 80
pathType: ImplementationSpecific
Rate Breaker¶
The rateBreaker filter is a breaker for the ingress route that open if the backend failures for the route reach a value of N within a window of the last M requests, where N (in this example 30) and M (in this example 300) are mandatory arguments of the filter and there are some more optional arguments documented.
rateBreaker(30, 300)
The ingress spec would look like this:
apiVersion: networking.k8s.io/v1
kind: Ingress
metadata:
annotations:
zalando.org/skipper-filter: rateBreaker(30, 300)
name: app
spec:
rules:
- host: app-default.example.org
http:
paths:
- backend:
service:
name: app-svc
port:
number: 80
pathType: ImplementationSpecific
Admission Control¶
The admissionControl filter is a dynamic circuit breaker that works based on HTTP error codes observed by backends. It will dynamically adjust the shedding of load to the maximum throughput possible.
apiVersion: networking.k8s.io/v1
kind: Ingress
metadata:
annotations:
zalando.org/skipper-filter: admissionControl("myapp", "active", "1s", 5, 10, 0.95, 0.9, 0.5)
name: app
spec:
rules:
- host: app-default.example.org
http:
paths:
- backend:
service:
name: app-svc
port:
number: 80
pathType: ImplementationSpecific
Ratelimits¶
There are two kind of ratelimits:
- Client side ratelimits are used to slow down login enumeration attacks, that targets your login pages. This is a security protection for DDoS or login attacks.
- Service or backend side ratelimits are used to protect your services due too much traffic. This can be used in an emergency situation to make sure you calm down ingress traffic or in general if you know how much calls per duration your backend is able to handle.
- Cluster ratelimits can be enforced either on client or on service side as described above.
Ratelimits are enforced per route.
More details you will find in rate limit filters section and in our ratelimit tutorial.
Client Ratelimits¶
The example shows 20 calls per hour per client, based on X-Forwarded-For header or IP in case there is no X-Forwarded-For header set, are allowed to each Skipper instance for the given ingress.
apiVersion: networking.k8s.io/v1
kind: Ingress
metadata:
annotations:
zalando.org/skipper-filter: clientRatelimit(20, "1h")
name: app
spec:
rules:
- host: app-default.example.org
http:
paths:
- backend:
service:
name: app-svc
port:
number: 80
pathType: ImplementationSpecific
If you need to rate limit service to service communication and you use Authorization headers to protect your backend from your clients, then you can pass a 3 parameter to group clients by “Authorization Header”:
apiVersion: networking.k8s.io/v1
kind: Ingress
metadata:
annotations:
zalando.org/skipper-filter: clientRatelimit(20, "1h", "authorization")
name: app
spec:
rules:
- host: app-default.example.org
http:
paths:
- backend:
service:
name: app-svc
port:
number: 80
pathType: ImplementationSpecific
Service Ratelimits¶
The example shows 50 calls per minute are allowed to each skipper instance for the given ingress.
apiVersion: networking.k8s.io/v1
kind: Ingress
metadata:
annotations:
zalando.org/skipper-filter: ratelimit(50, "1m")
name: app
spec:
rules:
- host: app-default.example.org
http:
paths:
- backend:
service:
name: app-svc
port:
number: 80
pathType: ImplementationSpecific
Cluster Ratelimits¶
Cluster ratelimits are eventual consistent and require the flag
-enable-swarm
to be set.
Service¶
The example shows 50 calls per minute are allowed to pass this ingress rule to the backend.
apiVersion: networking.k8s.io/v1
kind: Ingress
metadata:
annotations:
zalando.org/skipper-filter: clusterRatelimit("groupSvcApp", 50, "1m")
name: app
spec:
rules:
- host: app-default.example.org
http:
paths:
- backend:
service:
name: app-svc
port:
number: 80
pathType: ImplementationSpecific
Client¶
The example shows 10 calls per hour are allowed per client, X-Forwarded-For header, to pass this ingress rule to the backend.
apiVersion: networking.k8s.io/v1
kind: Ingress
metadata:
annotations:
zalando.org/skipper-filter: clusterClientRatelimit("groupSvcApp", 10, "1h")
name: app
spec:
rules:
- host: app-default.example.org
http:
paths:
- backend:
service:
name: app-svc
port:
number: 80
pathType: ImplementationSpecific
Path ratelimit¶
To ratelimit a specific path use a second ingress definition like
apiVersion: networking.k8s.io/v1
kind: Ingress
metadata:
name: app-default
spec:
rules:
- host: app-default.example.org
http:
paths:
- backend:
service:
name: app-svc
port:
number: 80
pathType: ImplementationSpecific
---
apiVersion: networking.k8s.io/v1
kind: Ingress
metadata:
name: app-login
annotations:
zalando.org/skipper-predicate: Path("/login")
zalando.org/skipper-filter: clusterClientRatelimit("login-ratelimit", 10, "1h")
spec:
rules:
- host: app-default.example.org
http:
paths:
- backend:
service:
name: app-svc
port:
number: 80
pathType: ImplementationSpecific
Shadow Traffic¶
If you want to test a new replacement of a production service with production load, you can copy incoming requests to your new endpoint and ignore the responses from your new backend. This can be done by the tee() and teenf() filters. See also our shadow traffic tutorial.
apiVersion: networking.k8s.io/v1
kind: Ingress
metadata:
annotations:
zalando.org/skipper-filter: teenf("https://app-new.example.org")
name: app
spec:
rules:
- host: app-default.example.org
http:
paths:
- backend:
service:
name: app-svc
port:
number: 80
pathType: ImplementationSpecific
Predicates¶
Predicates are influencing the route matching, which you might want to carefully test before using it in production. This enables you to do feature toggles or time based enabling endpoints.
You can use all kinds of predicates with filters together.
Feature Toggle¶
Feature toggles are often implemented as query string to select a new feature. Normally you would have to implement this in your application, but Skipper can help you with that and you can select routes with an ingress definition.
You create 2 ingresses that matches the same route, here host header
match to app-default.example.org
and one ingress has a defined query
parameter to select the route to the alpha version deployment. If the
query string in the URL has version=alpha
set, for example
https://app-default.example.org/mypath?version=alpha
, the service
alpha-svc
will get the traffic, if not prod-svc
.
alpha-svc:
apiVersion: networking.k8s.io/v1
kind: Ingress
metadata:
annotations:
zalando.org/skipper-predicate: QueryParam("version", "^alpha$")
name: alpha-app
spec:
rules:
- host: app-default.example.org
http:
paths:
- backend:
service:
name: alpha-svc
port:
number: 80
pathType: ImplementationSpecific
prod-svc:
apiVersion: networking.k8s.io/v1
kind: Ingress
metadata:
name: prod-app
spec:
rules:
- host: app-default.example.org
http:
paths:
- backend:
service:
name: prod-svc
port:
number: 80
pathType: ImplementationSpecific
IP Allow Listing¶
This ingress route will only allow traffic from networks 1.2.3.0/24 and 195.168.0.0/17 Before you use this in production please understand your deployment and check the difference between the following options:
apiVersion: networking.k8s.io/v1
kind: Ingress
metadata:
annotations:
zalando.org/skipper-predicate: ClientIP("1.2.3.0/24", "195.168.0.0/17")
name: app
spec:
rules:
- host: app-default.example.org
http:
paths:
- backend:
service:
name: app-svc
port:
number: 80
pathType: ImplementationSpecific
A/B test¶
Implementing A/B testing is heavy. Skipper can help you to do that. You need to have a traffic split somewhere and have your customers sticky to either A or B flavor of your application. Most likely people would implement using cookies. Skipper can set a cookie with responseCookie() in a response to the client and the cookie predicate can be used to match the route based on the cookie. Like this you can have sticky sessions to either A or B for your clients. This example shows to have 10% traffic using A and the rest using B.
10% choice of setting the Cookie “flavor” to “A”:
apiVersion: networking.k8s.io/v1
kind: Ingress
metadata:
annotations:
zalando.org/skipper-predicate: Traffic(.1, "flavor", "A")
zalando.org/skipper-filter: responseCookie("flavor", "A", 31536000)
name: app
spec:
rules:
- host: app-default.example.org
http:
paths:
- backend:
service:
name: a-app-svc
port:
number: 80
pathType: ImplementationSpecific
Rest is setting Cookie “flavor” to “B”:
apiVersion: networking.k8s.io/v1
kind: Ingress
metadata:
annotations:
zalando.org/skipper-filter: responseCookie("flavor, "B", 31536000)
name: app
spec:
rules:
- host: app-default.example.org
http:
paths:
- backend:
service:
name: b-app-svc
port:
number: 80
pathType: ImplementationSpecific
To be sticky, you have to create 2 ingress with predicate to match routes with the cookie we set before. For “A” this would be:
apiVersion: networking.k8s.io/v1
kind: Ingress
metadata:
annotations:
zalando.org/skipper-predicate: Cookie("flavor", /^A$/)
name: app
spec:
rules:
- host: app-default.example.org
http:
paths:
- backend:
service:
name: a-app-svc
port:
number: 80
pathType: ImplementationSpecific
For “B” this would be:
apiVersion: networking.k8s.io/v1
kind: Ingress
metadata:
annotations:
zalando.org/skipper-predicate: Cookie("flavor", /^B$/)
name: app
spec:
rules:
- host: app-default.example.org
http:
paths:
- backend:
service:
name: b-app-svc
port:
number: 80
pathType: ImplementationSpecific
Blue-Green deployments¶
To do blue-green deployments you have to have control over traffic
switching. Skipper gives you the opportunity to set weights to backend
services in your ingress specification. zalando.org/backend-weights
is a hash map, which key relates to the serviceName
of the backend
and the value is the weight of traffic you want to send to the
particular backend. It works for more than 2 backends, but for
simplicity this example shows 2 backends, which should be the default
case for supporting blue-green deployments.
In the following example my-app-1 service will get 80% of the traffic and my-app-2 will get 20% of the traffic:
apiVersion: networking.k8s.io/v1
kind: Ingress
metadata:
name: my-app
labels:
application: my-app
annotations:
zalando.org/backend-weights: |
{"my-app-1": 80, "my-app-2": 20}
spec:
rules:
- host: my-app.example.org
http:
paths:
- backend:
service:
name: my-app-1
port:
name: http
pathType: Prefix
path: /
- backend:
service:
name: my-app-2
port:
name: http
pathType: Prefix
path: /
For more advanced blue-green deployments, check out our stackset-controller.
Chaining Filters and Predicates¶
You can set multiple filters in a chain similar to the eskip format.
apiVersion: networking.k8s.io/v1
kind: Ingress
metadata:
annotations:
zalando.org/skipper-predicate: Cookie("flavor", /^B$/) && Source("1.2.3.0/24", "195.168.0.0/17")
zalando.org/skipper-filter: clientRatelimit(50, "10m") -> requestCookie("test-session", "abc")
name: app
spec:
rules:
- host: app-default.example.org
http:
paths:
- backend:
service:
name: app-svc
port:
number: 80
pathType: ImplementationSpecific
Controlling HTTPS redirect¶
Skipper Ingress can provide HTTP->HTTPS redirection. Enabling it and setting the status code used by default can
be done with the command line options: -kubernetes-https-redirect
and -kubernetes-https-redirect-code
. By using
annotations, this behavior can be overridden from the individual ingress specs for the scope of routes generated
based on these ingresses specs.
Annotations:
zalando.org/skipper-ingress-redirect
: the possible values are true or false. When the global HTTPS redirect is disabled, the value true enables it for the current ingress. When the global redirect is enabled, the value false disables it for the current ingress.zalando.org/skipper-ingress-redirect-code
: the possible values are integers300 <= x < 400
. Sets the redirect status code for the current ingress.
Example:
apiVersion: networking.k8s.io/v1
kind: Ingress
metadata:
annotations:
zalando.org/skipper-ingress-redirect: "true"
zalando.org/skipper-ingress-redirect-code: 301
name: app
spec:
rules:
- host: mobile-api.example.org
http:
paths:
- backend:
service:
name: app-svc
port:
number: 80
pathType: ImplementationSpecific
Load Balancer Algorithm¶
You can set the loadbalancer algorithm, which is used to find the next
endpoint for a given request with the ingress annotation
zalando.org/skipper-loadbalancer
.
For example, for some workloads you might want to have always the same endpoint for the same client. For this use case there is the consistent hash algorithm, that finds for a client detected by the IP or X-Forwarded-For header, the same backend. If the backend is not available it would switch to another one.
Annotations:
zalando.org/skipper-loadbalancer
see available choices
Example:
apiVersion: networking.k8s.io/v1
kind: Ingress
metadata:
annotations:
zalando.org/skipper-loadbalancer: consistentHash
name: app
spec:
rules:
- host: websocket.example.org
http:
paths:
- backend:
service:
name: app-svc
port:
number: 80
pathType: ImplementationSpecific